Skip to content

Latest commit

 

History

History
405 lines (290 loc) · 15.9 KB

top-legacy-blades-template-pdl.md

File metadata and controls

405 lines (290 loc) · 15.9 KB

Legacy TemplateBlades with PDL

Disclaimer

Note that this document covers a legacy programming model -- PDL -- that should not be used when implementing new blades in the Azure portal. You should only need this document if you are maintaining an existing blade that was implemented using the PDL-based programming model.

Overview

The TemplateBlade is the recommended way of authoring blades in Ibiza. It is the equivalent to windows or pages in other systems.

You can think of a TemplateBlade as an HTML page. Authoring template blades requires:

  • an HTML template
  • a ViewModel
  • an optional CSS file
  • a use of <TemplateBlade> in a corresponding PDL file, naming the HTML template filename and view model class, supplying additional TemplateBlade configuration.

What follows is a walk-through of creating a TemplateBlade.


Creating the TemplateBlade

Use the following three steps to create a template blade.

  1. Add the blade definition to the PDL file, as in the following example.

    <TemplateBlade
                Name="MyTemplateBlade"
                ViewModel="{ ViewModel Name=MyTemplateBladeViewModel, Module=./ViewModels/MyTemplateBladeViewModel }"
                InitialDisplayState="Maximized"
                Template="{ Html Source='Templates\\MyTemplateBlade.html' }">
    </TemplateBlade>

    The PDL file can contain several options. The following is a list of the most relevant parameters.

    Name: Name of the blade. This name is used later to refer to this blade.

    ViewModel: Required field. The ViewModel that is associated with this blade.

    Template: Required field. The HTML template for the blade.

    Size: Optional field. The width of the blade. The default value is Medium.

    InitialDisplayState: Optional field. Specifies whether the blade is opened maximized or not. The default value is Normal.

    Style: Optional field. Visual style for the blade. The default value is Basic.

    Pinnable: Optional field. Flag that specifies whether the blade can be pinned or not. The default value is false.

    ParameterProvider: Optional field. Flag that specifies whether the blade provides parameters to other objects. The default value is false.

    Export: Optional field. Flag that specifies whether this blade is exported in the extension so that it can be opened by other extensions. As a result, a strongly typed blade reference is created.

  2. Create a ViewModel TypeScript class. The following example demonstrates the ViewModel that is associated with the blade from the PDL file in the previous step. This model exposes two observable properties, but more complex behavior can be added as appropriate.

    export class MyTemplateBladeViewModel extends MsPortalFx.ViewModels.Blade {
    
        public text: KnockoutObservable<string>;
        public url: KnockoutObservable<string>;
    
        constructor(container: MsPortalFx.ViewModels.ContainerContract, initialState: any, dataContext: any) {
            super();
            this.title("InfoBox");
            this.subtitle("InfoBox Playground");
    
            this.text = ko.observable<string>("Go to the Azure Portal");
            this.url = ko.observable<string>("https://portal.azure.com");
        }
    
        public onInputsSet(inputs: any): Promise<any> {
            return Promise.resolve();
        }
    }
  3. Create a template for the blade using regular HTML and Knockout. The Knockout bindings are bound to the public properties in the ViewModel in the previous step.

    <div>This is an example template blade that shows a link.</div>
    
    <a data-bind="text: text, attr: { href: url }" target="_blank"></a>

For more information about Knockout, see https://knockoutjs.com.

Adding controls

Ibiza provides an extensive controls library that can be used in the HTML template. The following example uses the InfoBox control instead of a regular HTML link.

  1. Change the link element in the HTML template to a control container.

    <div>This is an example template blade that shows a link.</div>
    
    <div data-bind="pcControl:infoBox"></div>
  2. Update the blade ViewModel to expose and instantiate the control ViewModel, as in the following code.

    export class MyTemplateBladeViewModel extends MsPortalFx.ViewModels.Blade {
    
        // view-model for the infoBox control
        public infoBox: MsPortalFx.ViewModels.Controls.InfoBox.BaseViewModel;
    
        constructor(container: MsPortalFx.ViewModels.ContainerContract, initialState: any, dataContext: any) {
            super();
            this.title("InfoBox");
            this.subtitle("InfoBox Playground");
    
            // initialization of the InfoBox view-model
            this.infoBox = new MsPortalFx.ViewModels.Controls.InfoBox.LinkViewModel(container, {
                text: ko.observable<string>('Go to the Azure Portal'),
                image: ko.observable(MsPortalFx.Base.Images.Info()),
                clickableLink: ko.observable(MsPortalFx.ViewModels.Part.createClickableLinkViewModel(ko.observable<string>('https://portal.azure.com'))
            });
        }
    
        public onInputsSet(inputs: any): Promise<any> {
            return Promise.resolve();
        }
    }
  3. This example uses the PDL file from the section named #creating-the-templateblade.

Supplying parameters to the TemplateBlade

Blades can receive input parameters that are part of the signature for the blade. The following code adds an "id" input parameter to the template blade. It reuses the HTML template from the previous steps.

  1. Include the parameters in the signature of the blade in the PDL definition.

    <TemplateBlade
                Name="MyTemplateBlade"
                ViewModel="{ ViewModel Name=MyTemplateBladeViewModel, Module=./ViewModels/MyTemplateBladeViewModel }"
                Template="{ Html Source='Templates\\MyTemplateBlade.html' }">
        <TemplateBlade.Parameters>
            <Parameter Name="id" />
        </TemplateBlade.Parameters>
    </TemplateBlade>
  2. Define the signature in the ViewModel.

    import Def = ExtensionDefinition.ViewModels.Resource.MyTemplateBladeViewModel;
    
    export class MyTemplateBladeViewModel extends MsPortalFx.ViewModels.Blade {
    
        // this property is part of the blade signature and is passed into onInputSet
        public id: KnockoutObservable<string>;
    
        public infoBox: MsPortalFx.ViewModels.Controls.InfoBox.BaseViewModel;
    
        constructor(container: MsPortalFx.ViewModels.ContainerContract, initialState: any, dataContext: any) {
            super();
            this.title("InfoBox");
            this.subtitle("InfoBox Playground");
    
            this.infoBox = new MsPortalFx.ViewModels.Controls.InfoBox.LinkViewModel(container, {
                text: ko.observable<string>('Go to the Azure Portal'),
                image: ko.observable(MsPortalFx.Base.Images.Info()),
                clickableLink: ko.observable(MsPortalFx.ViewModels.Part.createClickableLinkViewModel(ko.observable<string>('https://portal.azure.com'))
            });
        }
    
        public onInputsSet(inputs: Def.InputsContract): Promise<any> {
            // write the input property to the console
            console.log(inputs.id);
            return Promise.resolve();
        }
    }

Adding commands

Commands are typically displayed at the top of the template blade. To add the commands, add a toolbar to the TemplateBlade, and then define its contents in the ViewModel.

The working copy of the sample in the Dogfood environment is located at https://df.onecloud.azure-test.net/?SamplesExtension=true#blade/SamplesExtension/SDKMenuBlade/bladewithtoolbar.

  1. Add a CommmandBar element to the PDL template.

    <TemplateBlade
                Name="MyTemplateBlade"
                ViewModel="{ ViewModel Name=MyTemplateBladeViewModel, Module=./ViewModels/MyTemplateBladeViewModel }"
                Template="{ Html Source='Templates\\MyTemplateBlade.html' }">
        <TemplateBlade.Parameters>
            <Parameter Name="id" />
        </TemplateBlade.Parameters>
        <CommandBar />
    </TemplateBlade>
  2. Instantiate the CommandBar in the ViewModel, as in the following example.

    import Def = ExtensionDefinition.ViewModels.Resource.MyTemplateBladeViewModel;
    
    export class MyTemplateBladeViewModel extends MsPortalFx.ViewModels.Blade {
    
        public id: KnockoutObservable<string>;
        public infoBox: MsPortalFx.ViewModels.Controls.InfoBox.BaseViewModel;
    
        // toolbar view-model
        public commandBar: MsPortalFx.ViewModels.Toolbars.ToolbarContract;
    
        constructor(container: MsPortalFx.ViewModels.ContainerContract, initialState: any, dataContext: any) {
            super();
            this.title("InfoBox");
            this.subtitle("InfoBox Playground");
    
            this.infoBox = new MsPortalFx.ViewModels.Controls.InfoBox.LinkViewModel(container, {
                text: ko.observable<string>('Go to the Azure Portal'),
                image: ko.observable(MsPortalFx.Base.Images.Info()),
                clickableLink: ko.observable(MsPortalFx.ViewModels.Part.createClickableLinkViewModel(ko.observable<string>('https://portal.azure.com'))
            });
    
            // initialize the toolbar
            var button = new Toolbars.OpenLinkButton("https://azure.com");
            button.label("azure.com");
            button.icon(MsPortalFx.Base.Images.Hyperlink());
            this.commandBar = new Toolbars.Toolbar(container);
            this.commandBar.setItems( [ button ] );
        }
    
        public onInputsSet(inputs: Def.InputsContract): Promise<any> {
            return Promise.resolve();
        }
    }

Adding buttons

Blades can display buttons that are docked at the base of the blade. The following code demonstrates how to add buttons to the blade.

  1. Add an ActionBar element in the PDL template. The ActionBar is docked to the bottom of the blade and contains buttons, as in the following example.

    <TemplateBlade
                Name="MyTemplateBlade"
                ViewModel="{ ViewModel Name=MyTemplateBladeViewModel, Module=./ViewModels/MyTemplateBladeViewModel }"
                Template="{ Html Source='Templates\\MyTemplateBlade.html' }">
        <TemplateBlade.Parameters>
            <Parameter Name="id" />
        </TemplateBlade.Parameters>
        <ActionBar ActionBarKind="Generic" />
    </TemplateBlade>
  2. Instantiate the ActionBar in the ViewModel.

        export class MyTemplateBladeViewModel extends MsPortalFx.ViewModels.Blade {
    
            public id: KnockoutObservable<string>;
            public infoBox: MsPortalFx.ViewModels.Controls.InfoBox.BaseViewModel;
    
            // define the actionBar view-demol
            public actionBar: MsPortalFx.ViewModels.ActionBars.GenericActionBar.ViewModel;
    
            constructor(container: MsPortalFx.ViewModels.ContainerContract, initialState: any, dataContext: any) {
                super();
                this.title("InfoBox");
                this.subtitle("InfoBox Playground");
    
                this.infoBox = new MsPortalFx.ViewModels.Controls.InfoBox.LinkViewModel(container, {
                    text: this.text,
                    image: ko.observable(MsPortalFx.Base.Images.Info()),
                    clickableLink: ko.observable(MsPortalFx.ViewModels.Part.createClickableLinkViewModel(this.url))
                });
    
          // initialize the ActionBar
    
                this.actionBar = new MsPortalFx.ViewModels.ActionBars.GenericActionBar.ViewModel(container);
                this.actionBar.actionButtonClick = () => {
                    console.log("Clicked!!!");
                };
            }
    
            public onInputsSet(inputs: Def.InputsContract): Promise<any> {
                return Promise.resolve();
            }
        }

Displaying a full screen blade

To open the blade using the full screen, add InitialState="Maximized" to the PDL definition of the blade, as in the following code.

<TemplateBlade
            Name="MyTemplateBlade"
            ViewModel="{ ViewModel Name=MyTemplateBladeViewModel, Module=./ViewModels/MyTemplateBladeViewModel }"
            InitialDisplayState="Maximized"
            Template="{ Html Source='Templates\\MyTemplateBlade.html' }">
</TemplateBlade>

Displaying a loading indicator UX

Sometimes, interaction with a blade should be prevented while it is initializing. In those cases, a shield that contains a loading indicator UX is displayed in the blade to block the display. The shield can be fully transparent or opaque. The following code demonstrates how to set an opaque filter in the blade.

constructor(container: FxCompositionBlade.Container, initialState: any, dataContext: BladesArea.DataContext) {
    super();

    var operation = Q.defer<any>();

    // display the shield while the operation promise is not resolved
    container.operations.add(operation.promise, { blockUi: true, shieldType: MsPortalFx.ViewModels.ShieldType.Opaque });

    // wait for 3 seconds and resolve the promise (which will remove the shield)
    window.setTimeout(() => { operation.resolve(); }, 3000);
}

The following code snippet demonstrates how to apply a filter on a timer. The filter slowly changes from opaque to transparent. The sample is also located at <dir>\Client\V1\Blades/Template/ViewModels/TemplateBladeViewModels.ts.

@Di.Class("viewModel")
export class TemplateBladeWithShieldViewModel
extends Blade
implements Def.TemplateBladeWithShieldViewModel.Contract
{
/**
 * The blade's title.
 */
public title: KnockoutObservable<string>;

/**
 * TextBox form field.
 */
public myTextBox: TextBox.Contract;

private _timerHandle: number;

constructor(container: FxCompositionBlade.Container) {
    super();

    this.title(ClientResources.templateBladeWithShield);

    const translucent = MsPortalFx.ViewModels.ShieldType.Translucent;
    const opaque = MsPortalFx.ViewModels.ShieldType.Opaque;
    let isTranslucent = true;

    const op = () => {
        const operation = Fx.defer<any>();
        const shieldType = isTranslucent ? translucent : opaque;
        container.operations.add(operation.promise, { blockUi: true, shieldType: shieldType });

        isTranslucent = !isTranslucent;
        window.setTimeout(() => { operation.resolve(); }, 3000);
    };

    op();

    window.setInterval(op, 5000);

    // TextBox
    this.myTextBox = TextBox.create(container, {
        label: ClientResources.formsSampleBasicTextBox,
    });
}

/**
 * Clean up any resources.
 */
public dispose(): void {
    window.clearInterval(this._timerHandle);
}
}