Workflow framework engine


Keywords
angular, workflow
License
Other
Install
Install-Package BizDoc.Core -Version 8.29.0

Documentation

bizdoc.core

BizDoc is a software developer framework for delivering form-flow solution. It includes a workflow engine, a mailbox -like user interface, and set of built-in components for data analysis.

Setting up

BizDoc is built as a .Net web application, running Angular on the client side. To author a new BizDoc environment, create a new project from BizDoc template.

Prerequisites

Visual Studio or Visual Code, .Net 8.0, Latest Node.js, Angular CLI and EF Core.

Installing

To install BizDoc, open Visual Studio and choose Extensions menu, Manage Extensions. Select Online and search for BizDoc Core.

Install the package and restart Visual Studio.

Create a new project and choose BizDoc as the template.

Update BizDoc Nuget package from Package Manager Console:

Update-Package bizdoc.core

Update npm package from PowerShell command-line:

npm i bizdoc.core@latest

Create a database of your choice and set it's connectionString in appsettings.json.

BizDoc will automatically create the database objects.

Architecture

Architecture

BizDoc can be broken into two major parts: backend server objects, responsible for the business logic, and front-end user interface, built as Angular components. Commonly, a front-end component has a matching backing server object. For example, BizDoc form object will have a backend class that manages its lifetime events, while a corresponding front-end Angular component will take care of presenting form data to user and respond to form input. Communication between the front-end and the backend is done through BizDoc APIs.

BizDoc configures both managed and none-managed server-side objects in bizdoc.json file.

Open bizdoc.json and review your app's configuration.

If a BizDoc object has a front-end Angular component, the two is coupled by annotating the back-end class with the [Template()] attribute, and decorating the Angular component with the @BizDoc() attribute with a matching value.

BizDoc also maintains non managed objects such as Roles. A role is declared per type and assigned positions per type keys. In addition to assigning roles to type keys, positions can be assigned to patterns, a regular expression, and to groups of keys.

BizDoc maintains its data in a database. Available database providers are SqlServer, MySQl and Oracle. The BizDoc database objects are created under the [BizDoc] schema. In addition to the core database, developer commonly create their own database context to access custom database objects. Refer to .Net Core EF on how to create and maintain Entity Framework context.

BizDoc web user interface is aligned with Material Design guidelines.

User interface is responsive, supporting mobile browsers. If you wish to develop a dedicate Mobile app, we offer libraries for Flutter, Kotlin and Xamarin.

Configuration

BizDoc server-side behavior is set in startup.cs. You first register BizDoc services using AddBizDoc(), then consume it using UseBizDoc().

BizDoc configuration includes licensing, SMTP and more.

Setup database provider by installing the relevant Nuget, from either PostgreSQL, SqlServer, Oracle or MySql.

    services.AddBizDoc().
      UseSqlServer(connectionString);

Set up authentication from one of: FormIdentity for managing users in database, DirectoryServices, Azure PingOne or Okta.

Install the relevant Nuget and add it to services in startup.cs.

    services.AddBizDoc().
       UseSqlServer(connectionString).
       AddAspIdentity(options =>
          {
              options.Password.RequireLowercase = false;
              options.Password.RequireUppercase = false;
              options.Password.RequireDigit = false;
              ...
          });

If the available default identity managers doesn't answer your organization needs, you can implement your own identity manager. See below how to create a Custom Identity Manager.

BizDoc offers additional services, like:

  • AddEscalate() - Escalate unattended documents.
  • AddSummaryMail() - Send @ of pending documents, notifications and chat summary.
  • AddMailExecute() - Analyze @ and forward document workflow. Service arrives in three configurations: IMAP, POP3 and Exchange. Exchange require an additional reference to Exchange.
  • AddExchangeRate() - Update currency exchange rates from the web.
  • AddSwagger() - Support Swagger.

See full extensions list here.

You can set BizDoc client app behavior using the BizDocModule.forRoot() function in app.module.ts file.

  imports: [BizDocModule.forRoot({
    currencyCode: 'EUR'
    ...
  })]

Find product updates in release notes.

Managed Objects

A managed object is a unit of code that implements one of BizDoc base classes. Base classes are found in BizDoc.Configuration namespace.

Dependency Injection

A BizDoc object supports Dependency Injection. Object may consume services added in your startup.cs as well as BizDoc services.

public class MyForm : FormBase<MyFormModel> {
    private readonly IWorkflowInstance _workflowInstance;
    public MyForm(IWorkflowInstance workflowInstance) {
        _workflowInstance = workflowInstance;
    }
}

The above code uses the IWorkflowInstance service to access the currently running workflow.

BizDoc provides list here.

Angular DI

In your Angular app, you can access BizDoc functionality by injecting BizDoc services.

See full Angular services list here.

class MyClass {
  constructor(private _cube: CubeInfo) {
  }
  open() {
    this._cube.open({
      company: 201
    }, {
     xAxis: 'month',
     series: 'balance'
    })
  }
}

The above example uses CubeInfo to show a matrix of cube axes with company 201 showing balance per month.

Using the MapInfo requires configuring the maps settings in BizDocModule.forRoot({maps: {apiKey: ...}}), in app.module.ts file.

Forms

A form object consists of three parts:

  • A data model, representing form properties,
  • A backing class, derived from FormBase, responsible for managing form lifetime events,
  • And an Angular component, displaying form data.

If you choose to develop a dedicated mobile app, you'll need to develop the form again in the respective environment.

Declare a data model

public class MyFormModel {
  public string Purpose { get; set; }
  public DateTime? Due { get; set; }
  ...
}

You may annotate your model with attributes to instruct BizDoc how to handle it.

See full list here.

In addition to BizDoc behavior attributes above, model properties may be annotated with .Net attributes such as Required, DataType, MaxLength and Display.

For example, to use the model Purpose property as the document title, annotate it with the Subject attribute.

using BizDoc.ComponentModel.Annotations;

public class MyFormModel {
  [Subject]
  public string Purpose { get; set; }
  ...
}

Data model declare sub models, to implement more complex scenarios:

public class MyFormModel {
  ...
  public IList<MyFormModelLine> Lines { get; set; }
}

Declare baking object

Create a class that derived from FormBase<T>. Override its methods to respond to form events.

using BizDoc.Configuration;

[Form(title: "My form")]
public class MyForm : FormBase<MyFormModel>
{
  public override Task FlowEndAsync(MyFormModel model)
  {
    ...
  }
}

The above example uses the Form attribute to instruct BizDoc to set a title to form when registering it in bizdoc.json configuration file. Title an other properties can be later altered by editing the configuration file.

You may assign an icon to the form from Material Icons.

Mapping database tables

BizDoc stores form's data internally. If you wish to store form data as database tables in addition to the internal representation for reporting purposes, annotate the class with Table attribute.

Assign the DocumentId attribute to instruct BizDoc to set its value from the document identity, and the Line attribute to set am ordinal to a sub model collections.

[Table("MyTableLine")]
public class MyFormLine {
  [DocumentId]
  public int FormId { get; set; }
  [Line]
  public short Line { get; set; }
  ...
}

Use NotMapped attribute on properties you wish to exclude.

Mapping a cube

Map form model to cube by annotating your model class with CubeMapping attribute. This creates an index from model properties, which can later be accessed from various reports and widgets.

[CubeMapping(nameof(Year), nameof(Quarter), nameof(Month), nameof(Balance))]
public class Line {
  public DateTime Date { get; set; }
  [JsonIgnore, ListType(typeof(Years))]
  public short Year => (short)Date.Year;
  [JsonIgnore, ListType(typeof(Months))]
  public byte Month => (byte)Date.CubeMonth(); // extension method
  [JsonIgnore, ListType(typeof(Quarters))]
  public byte? Quarter => Date.Quarter(); // extension method

  [ListType(typeof(Balances)), ValueResolver(typeof(StateAxisResolver<Balance?>))]
  public Balance? Balance { get; set; }
  [Value, JsonIgnore]
  public decimal Total => Amount * Qty;
  public decimal Amount { get; set; }
  public decimal Qty { get; set; }
}

Axes mapped has to be a known model property - or parent model property.

The Value attribute directs BizDoc which property holds the cube value. You can also set the Currency, ExchangeRate and Percentage attributes.

The StateAxisResolver above finds the Balance in the configuration file that matches the document state. Set the Axis attribute of state Options:

"States": [
{
  "Past": "Opened",
  "Color": "green",
  "Name": "open",
  "Title": "Open",
  "ResourceType": "BizDoc.Core.Resources.Strings",
  "Options": {
    "Axis": "Open"
  }
}]

CubeMapping axes should match the ordinal of the Axes declared in configuration file.

Mapping scheduled tasks

Schedule one or more events from model properties, allowing documents to be presented on a calendar.

[ScheduleMapping(nameof(EventDate))]
public class Line {
  public DateTime EventDate {get; set;}
}

Designing the User Interface

A BizDoc form front-end is represented as an Angular component.

To create an Angular component, run the following from terminal in project ./ClientApp folder:

ng g c myForm

Open ./app/src/my-Form/my-form.component.ts and replace the content of the file with the following:

import { FormComponent, RecipientModel, ViewMode, BizDoc } from '@bizdoc/core';

@Component({
  templateUrl: './my-form.component.html',
  styleUrls: ['./my-form.component.scss']
})
@BizDoc({
  selector: 'app-my-form'
})
export class MyFormComponent implements FormComponent<MyFormModel> {
  readonly form = this.fb.group({
    subject: this._fb.control([], [Validators.required])
  });
  mode: ViewMode;
  constructor(private _fb: FormBuilder) {
  }
  onBind(model: RecipientModel<MyFormModel>): void {
  }
}
interface MyFormModel {
    subject: string;
}

The above code declares MyFormModel interface matching the server-side data model, with Subject property using camel-casing. The interface is then given as a generic parameter to the FormComponent<T> interface.

Note the the client-side model properties names use camel casing.

Access form data from the onBind() function.

Open my-form.component.html and replace the content of the file with the following:

<form [formGroup]="form" autocomplete="off">
  <mat-form-field>
    <input matInput placeholder="Subject" required formControlName="subject" />
  </mat-form-field>
</form>

The formGroup attribute maps to the FormGroup declared in the .ts file, while the input tag maps to the subject form control.

See Angular reactive forms on forms options and validations.

Complex forms may separate behavior to more than one Angular component.

You can embed BizDoc components in your form. Components can be Select, Autocomplete, CombinationPicker, CombinationPool, TimePicker and AddressInput.

<mat-form-field>
  <bizdoc-select type="users" placeholder="My property" formControlName="requesterId"></bizdoc-select>
</mat-form-field>

In the above example, the type attribute matches a predefined object declared in the configuration file types.

The CombinationPicker and CombinationPool tags allows the user to pick combinations of segments.

<mat-form-field>
  <bizdoc-combination-picker placeholder="My property"
  (optionSelected)='accountPicked($event)'></bizdoc-combination-picker>
</mat-form-field>

An axis is considered a segment if its Combination attribute is set to true.

By default, combinations are stored in BizDoc.Combinations database table. You can override CombinationsAsync() method of the cube backend object to populate combinations from a different source.

The CombinationPool tag allows the user to pick combinations of segments by constraints, set too in configuration file.

  <bizdoc-combination-pool [formGroup]="form"><bizdoc-combination-pool>
{"Cubes": [{
  "Name": "myCube",
  "Constraints": [
    {
      "Condition": {
        "company": "company1"
      },
      "Projection": {
        "region": ["100-200"]
      }
    }
  ]
}]
}

Conditions and projections accepts axis expression, presenting either a value, a range of values, an array of values or a mask. For explanation on axis options see Understanding Patterns.

Supporting view mode

A form may be visible in one of three ways: editing, previewing or version compare. Template may handle the different cases using an ngSwitch:

<ng-container [ngSwitch]="mode">
  <div *ngSwitchCase='"compose"'>
    <!-- editing mode -->
  </div>
  <div *ngSwitchDefault>
    <!-- preview and version mode -->
  </div>
</ng-container>

Mapping Server Object To Angular Component

The server-side object maps to the Angular component using the Template attribute.

using BizDoc.ComponentModel.Annotations;

[Template("app-my-form")]
public class MyFormModel {
  ...
}

The corresponding Angular component use the @BizDoc decorator, with the same value, here app-my-form.

import { FormComponent, RecipientModel, ViewMode, BizDoc } from '@bizdoc/core';

@BizDoc({
  selector: 'app-my-form'
})
export class MyFormComponent implements FormComponent<MyFormModel> {
  ...
}

In addition, you'll need to register the component in your app.module.ts:

  imports: [BizDocModule.forRoot({
    components: [MyFormComponent]
    ...
  })]

Types

A type represents a source of key-value pairs. For example, an Account type may retrieve accounts from a database, and make their values available using a type.

[Table("Accounts")]
public class Account
{
    public string Id { get; set; }
    public string Name { get; set; }
}
/// <summary>
/// Accounts data source
/// <summary>
public class Accounts : TypeBase<string>
{
    private readonly CustomStore _store;

    public Accounts(CustomStore customStore) => _store = customStore;

    public override Task<Dictionary<string, string>> GetValuesAsync(Void args) =>
        _store.Accounts.ToDictionaryAsync(a => a.Id, a => a.Name);
}

Type can be linked to a form model property by setting its ListType attribute.

[ListType(typeof(Accounts))]
public string AccountId {get; set }

Built-In Types

BizDoc has several built-in types, including Years, Quarters, Months and Users. See BizDoc.Configuration.Types namespace for the complete list of available types.

You control Type object behavior by setting its Options in the configuration file.

{"Types": [
  {
    "Name": "products",
    "Type": "BizDoc.Configuration.Types.SqlDataSource",
    "Options":{
      "ConnectionString": "<connection name here>",
      "CommandText": "SELECT Id, Name FROM Products"
    }
  }
]
}

The code above queries a database for products.

Alternatively, you may manage type values in the configuration file itself.

{
  "Types":[{
    "Type": "BizDoc.Configuration.Types.ConfigurationDataSource",
    "Name": "types",
    "Options": {
      "Items": {
          "Type1": "Type 1",
          "Type2": "Type 2"
      }
    }
  }]
}

You can make your own object configurable by adding public properties to it and setting default values to those properties in configuration Option.

Reports

A report component consists of a data model, a backing class, and an arguments class.

public class MyReportDataModel
{
    [Key]
    public int Id { get; set; }
    public string Product { get; set; }
    public decimal? Price { get; set; }
}
public struct MyReportArgsModel
{
    public DateTime Starting { get; set; }
}
[Report(title: "My report")]
public class MyReport : ReportBase<MyReportArgsModel, MyReportDataModel>
{
  private readonly CustomStore _store;
  public MyReport(CustomStore store) {
    _store = store;
  }

  public override async Task<IEnumerable<MyReportDataModel>>
    PopulateAsync(MyReportArgsModel args, IProgress<int> progress) =>
      await _store.Lines.Select(l=> new MyReportDataModel {
        Product = l.Product,
        Price = l.Price,
        ...
      }).ToListAsync();
}

The above PopulateAsync method accepts MyReportArgsModel and returns a list of MyReportDataModel retrieved from database.

A report object may have an Angular component presenting it. In which case, the component should implement the IReportComponent<T>.

import {ReportComponent, BizDoc, ReportRef} from '@bizdoc/core';

@Component({...})
@BizDoc({ selector: 'my-report' })
export class MyReport implements ReportComponent<MyReportModel, MyReportArgs> {
  constructor(@Inject(ReportRef) private _ref: ReportRef) { }
  onBind(data: MyReportModel[]) {
    ...
  }
}
interface MyReportModel { ... }
interface MyReportArgs { ... }

The MyReportModel and MyReportArgs above maps to the server-side models.

The ReportRef enables the report to access executing context, such as subscribing to report progress events of the server-side.

If no Angular component is registered for the report, BizDoc treats the model properties as columns and the arguments as filters.

Annotate model properties with Display and DataType attributes to control their layout.

You can also customize the arguments pane by implementing IArgumentComponent<T> and annotating the arguments model with the Template attribute.

Use one of the built-in pipes to display BizDoc keys as values.

Customizing built-in Reports

BizDoc has several built-in reports which can be customized by setting Options in the configuration.

{"Reports": [{
  "Type": "BizDoc.Configuration.Reports.CubeChartUsage",
  "Name": "cube-chart-usage",
  "Title": "ChartUsage",
  "Options": {
    "Series": [
      "year",
      "month"
    ],
    "ChartType": "Column",
    "Collapse": true
  }
}]
}

To find out which reports accepts which options, use the Object Browser to discover class properties.

Cube

A cube is a BI for indexing forms data. Data can then be visualized using cube views or one of the built-in reports and widgets.

Axes

A cube declares columns as axes. Each axis maps to a type, representing possible values.

You map a cube to form data model by annotating the CubeMapping attribute as explained above in the Form section.

Backing Cube Object

Create a class that inherits from CubeBase.

public class MyCube : CubeBase
{
}

Override base methods, such as CanView(), to control cube behavior.

Configuring

A cube uses views to show a segment of its data. A view typically has an X axis and series, and can present data as either chart, grid, pivot or spreadsheet.

"Cubes": [
  {
    "Axes": [
      {
        "DataType": "years",
        "Name": "year",
        "Title": "Year"
      },
      {
        "DataType": "quarters",
        "Name": "quarter",
        "Title": "Quarter"
      }
    ],
    "Views": [
      {
        "XAxis": [
          "year",
          "month"
        ],
        "Series": "balance",
        "Indices": "budget",
        "ChartType": "StackingColumn",
        "Name": "balance",
        "Title": "Balance"
      }
    ],
    "Indices": [
      {
        "Name": "budget",
        "Title": "Budget"
      }
    ],
    "Name": "myCube",
    "Title": "My cube"
  }]

An index represents a linear data such as budgeted values or target performance. Index data are stored in BizDoc.Indices database table.

As per document, cube database table is BizDoc.Entries.

Open bizdoc.json and find Cubes section. Add, modify and reorder axes and views.

Understanding Patterns

A cube may declare patterns. Each pattern represents a set of axes values which can be assigned permissions to one or more user roles and rules.

Use patterns to restrict access to cube data. For example, to restrict view to years 20` for companies 201 and 202 and regions 11 through 99 to system role, add the following configuration:

    "Patterns": [
      {
        "Axes": {
          "year": "20*",
          "company": [201, 202],
          "region": "11-99"
        },
        "Roles": [
          "System"
        ],
        "Name": "y2",
        "Title": "20`"
      }
    ]

Mask value accepts a dot (.) for a single character, asterisk (*) for more than one character and hashtag (#) for replacing an individual character with a character at the same position in a previously selected combination. Hashtags patterns enable drilling up from a base combination.

In addition to roles, a pattern can be set a programmatic rule expression. See Rules section on how to add an expression.

The built-in CubePatternUsage report uses patterns to selectively show data relevant to the user.

You can test if the user has privileges to a certain set of axes using the Authorize() method of the ICubeService service.

By default, the cube CanView() method uses patterns to restrict access to cube data.

Patterns can be modified using the designated administration utility or by manually editing the configuration file.

Explore data

If you use the cube to store 3rd party app data, such as POs from ERP, you can provide drill down to the 3rd party app by implementing the IBrowsable interface.

API

Querying

Use the ICubeService to preform queries on cubes, indices and currencies.

public class MyForm : FormBase<MyFormModel> {
  private readonly ICubeService _cubeService;
  public MyForm(ICubeService cubeService) {
      _cubeService = cubeService;
  }

  public override async Task FlowStartAsync(MyFormModel model) {
    var usage = await _cubeService.GetUsage("myCube",
      2020 /* year axis */,
      Axis.FromArray(1, 2) /* quarter axis */,
      default /* skip months axis */,
      Balance.Opened /* open only */);
      ...
  }
}

The above retrieves the usage for myCube, year 2020 1st through 2nd quarters of all Opened balance.

The Axis struct can be utilized to specify a range, an array, a mask or - combination of them using the static Join() method.

To show cube from your Angular app, use Angular ICubeService. See Angular DI example.

Define anomaly

Anomaly is when the usage accedes index value. By default, anomaly is calculated as the value of all cube records that map to the document entries deducted from all indices values.

BizDoc notifies recipients of any anomaly found in any of the axes. To make BizDoc ignore one of more segment, set the Sensitive to false for one of more axes in configuration file.

If the 3rd axis is, for example, months, the anomaly will be calculated in a higher lever of quarters.

You may also change the default behavior of notifying recipients to notifying users who where assigned to roles by setting the AnomalyPolicy to Positions.

To refine how anomaly is calculated, override cube CalculateAnomalyAsync() method. A negative value is considered an anomaly, whereas a positive or zero indicates no anomaly.

public class MyCube : CubeBase
{
  public override Task<decimal> CalculateAnomalyAsync(params Axis[] axes)
  {
    ...
  }
}

Widgets

A widget represents a component displayed in user dashboard, commonly showing relevant data summary. Widgets have an Angular component and a backing server-side object.

Core widgets server-side objects can be overridden. See How To example.

API

Views

A view presents information in the context of specific document. Built-in views can be configured to present cube -related data, like matrix and pivot, derived from the cube arguments provided in the document form.

Developer can create custom view, by implementing IViewComponent on the Angular side and inheriting ViewBase on the backing server code.

Commonly, views options are set in the Options node of Views in bizdoc.config.

API

Rules

A rule declares a programmatic value. For example, the Anomaly rule returns the cube anomaly for the currently processed document. Rules can then be evaluated in scenarios like a workflow if condition or object privileges.

API

How To

Enable version compare

BizDoc logs changes made by users to documents model. Changes can then be viewed from document trace. To enable a form to show version compare, use the bizdocCompareGroup, bizdocCompareContext and bizdocCompareName directives.

API

Enable navigation in forms

You communicate with form container by injecting FormRef.

export class MyFormComponent implements FormComponent<MyModel> {
  constructor(@Optional() @Inject(FormRef) private _formRef: FormRef) {
    this._formRef.navigating.subscribe(page => {
      ...
    });
  }
  page() {
    this._formRef.navigate('page1');
  }
}

This code pattern is recommended if your form opens internal portals, and the user should be able to navigate back to previous view.

Inject custom pane

BizDoc uses panes to display different content parts to desktop users. A pane can be opened either next to the currently displayed info, or as a tab.

If you wish to open a custom component on a new pane next to your form, use the PanesRouter service.

constructor(private _router: PanesRouter) {}
open() {
  this._router.navigate(MyPaneComponent);
}

You may pre-register panes with paths in advance. See panes for more.

Providing a user guide

BizDoc can present the user with a guide of component functionality and use.

You can also set a guide at runtime. For example, a form may provide different guide for line view and header view, or a different guide in preview and edit mode.

API

Format delivered emails

BizDoc delivers emails to users. Emails can be customized using XSLT.

To change the default XSLT, create a new XSLT file, choose 'Copy Always' in its file properties.

Edit file:

<?xml version="1.0" encoding="utf-8" ?>
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
    xmlns="https://raw.githubusercontent.com/moding-il/bizdoc.core/master/message.xsd">
  <xsl:output method="html" indent="no"/>
  <xsl:template match="/Message">
    <html>
      <body>
          #<xsl:value-of select="Number"/>
          ...
        </body>
    </html>
</xsl>

In startup.cs services.AddBizDoc(), set BodyTemplate to your xslt file path.

Study the schema to learn about the structure of the XML representing document data. xsd here.

For more information, see emails.

Provide a Custom Identity Manager

Create new class in your project implementing BizDoc.Core.Identity.IIdentityManager interface. Register in startup.cs.

  AddBizDoc(...).
    AddIdentityManager<MyManager>();

Customizing built-in components

BizDoc built-in components, such as widgets and reports, can be extended to alter their behavior. For example, an existing widget component implementation can be overridden to determine data scope.

public class MyDepartmentsCompare: DepartmentsCompareBase {
  private readonly IIdentityContext _identityContext;
  protected override async Task<string[]> GetUsersAsync(string groupId) => ... // provide list of identities
}

In BizDoc configuration file, change the Type of the widget to your object implementation.

{"Widgets": [{
  "Type": "MyProject.MyDepartmentsCompare, MyProject, Version=1.0.0.0, Culture=neutral, PublicKeyToken=null",
  "Name": "myDepartmentsCompare"
}]
}

BizDoc built-in objects can be found under the following namespaces:

  • BizDoc.Workflow.Activities,
  • BizDoc.Workflow.Actions,
  • BizDoc.Configuration.Types,
  • BizDoc.Configuration.Rules,
  • BizDoc.Configuration.Views,
  • BizDoc.Configuration.Reports and
  • BizDoc.Configuration.Widgets.

Support internationalization

Server-side objects internationalization may be applied by setting the ResourceType attribute to a .resx public file.

"Folders": [
  {
    "Icon": "drafts",
    "Name": "df",
    "Title": "Drafts",
    "ResourceType": "BizDoc.Core.Resources.Strings"
  }]

In a custom object code:

[Form(Title = "MyForm", ResourceType = typeof(Properties.Resources))]
public class MyForm : FormBase<MyFormModel> {
}

The above Properties.Resources maps to project-level resources, found under project properties.

In Angular side, you can introduce a string resource using the TranslationService:

TranslationService.Set('es', {
  myField: 'My Field {0}'
});

Consume the resource by using the translate pipe:

<div>{{'myField' | translate : '!'}}</div>

In the above code, translate pipe accepts ! as a parameter, replacing the {0} in the resource string.

Disabling objects

In bizdoc.json file, locate the object you wish to disable and set the Disabled property.

{
  "Disabled": true
}

Set privileges to forms

Create rules in form configuration. Use administration utility to assign roles to rules.

{
  "Forms":[{
  "Rules": {
    "myField": {
      "Roles": [
        "role1"
      ]
    }
  },
  "Name": "myForm"
}]
}

In your form component, test privileges for a field or a section of your code by adding bizdocDisabled or bizdocHidden directive.

<input bizdocDisabled="myField" />
...
<div bizdocHidden="myField"></div>

You can gain programmatic access to rules from form component onBind() method.

onBind(data: RecipientModel<MyFormModel>): void {
    if (!data.rules['myField']) {
      ...
    }
}

In addition to roles, a rule may grant privilege per rules expression. See rules above on how to endorse a new rule or use the existing ones.

Run code after transaction completes

To run code after transaction safely committed, use the ScheduledTasks service.

Store custom user settings

Use BizDoc IProfileManager service to store user specific settings.

In your object constructor, consume IProfileManager and use Get() and Set() methods to retrieve and set a predefined model. Persist your changes using the Persist() method.

public class MyForm: FormBase<MyFormModel> {
    private readonly IProfileManager _profileManager;
    private readonly IHttpContext _httpContext;
    public MyForm(IHttpContext httpContext, IProfileManager profileManager) {
        _profileManager = profileManager;
        _httpContext = httpContext;
    }
    private void ApplySettings() {
        var profile = _profileManager[_httpContext.UserId];
        var settings = profile.Get<MySettings>();
        settings.Like = false;
        _profileManager.Persist(profile);
    }
    public struct MySettings {
        public bool Like {get; set; }
    }
}

Database

BizDoc database tables are self-maintained under the BizDoc schema. You can access database objects using the IDocumentStore service. If you wish to access the cube to query data, we recommend using the ICubeService as explained in the Cube section.

You can access a document data model using the Document GetModel<TModel>() method. This practice is not intended for large queries. Use TableMapping as explained above when dealing with a large datasets.

As Cube and Indices tables grow significantly large, consider adding indexes to the axes you use.

If you add 3rd party information to Cube table, use separate records than the ones BizDoc creates.

Segments and Combinations tables are used by Segments built-in type, CombinationPicker and CombinationPool. You may override cube CombinationsAsync() method to retrieve combinations from a source of your choice.

References

Issus can be submitted here.

Product updates are released regularly via npm and Nuget packages. Keep your project up to date!