Parts, also known as tiles, are a framework feature that integrates the UI of an extension on dashboards. Parts were more prevalent on blades, but this older pattern is being obsoleted by TemplateBlades that do not contain parts. For more information about template blades, see top-extensions-blades.md.
Intrinsic parts are composed of existing controls in the portal, and provide patterns for solving common patterns in the portal.
Many of the intrinsics can be found in the playground located at https://aka.ms/portalfx/playground.
The remainder of them can be located in the working copy located at https://aka.ms/portalfx/samples/#blade/SamplesExtension/IntrinsicPartsIndexBlade.
The following sections cover these topics.
Previously, Ibiza blades contained customizable parts, or tiles, that served as the main way for users to navigate the UI. However, the UI's were difficult to navigate, caused excessive navigation, caused excessive horizontal scrolling, and were not very performant.
Newer UI experiences combine traditional menu blades with TemplateBlades to display features and content.
There are still many cases where extensions may display rich monitoring experiences or at-a-glance data. To support those cases, Ibiza now supports multiple, shareable dashboards that were once known as the single start board.
It is highly recommended that developers who still need to build parts should build them for use on dashboards, instead of blades.
Built-in parts, also known as intrinsic parts, let extension developers create parts that contain pre-defined views and interaction patterns, but lets them plug in their own data.
NOTE: In this discussion, <dir>
is the SamplesExtension\Extension\
directory, and <dirParent>
is the SamplesExtension\
directory, based on where the samples were installed when the developer set up the SDK. If there is a working copy of the sample in the Dogfood environment, it is also included.
The following example of a Button part contains an icon and a label that navigates to a blade when the user clicks it. The three files used for the button part example are the following.
-
<dir>\Client\V1\Parts\Intrinsic\ButtonPart\ViewModels\ButtonPartIntrinsicInstructionsPartViewModel.ts
-
<dir>\Client\V1\Parts\Intrinsic\ButtonPart\ButtonPartIntrinsicInstructions.pdl
-
<dir>\Client\V1\Parts\Intrinsic\ViewModels\ButtonPartViewModel.ts
A working copy of the sample is located at https://aka.ms/portalfx/samples#blade/SamplesExtension/ButtonPartIntrinsicInstructions/selectedItem/ButtonPartIntrinsicInstructions/selectedValue/ButtonPartIntrinsicInstructions.
The following procedure demonstrates how to use a button part.
- Declare the part in the global
<Definition>
section of the PDL for the extension, as in the following example.
<!-- Name - Give your part a unique name -->
<!-- PartKind - This is where you declare that you're using a built in part type -->
<!-- ViewModel A pointer to the view model type that will customize the view (label, icon, etc) -->
<!-- InitialSize - The initial size of the part, in this case 2 X 1 (Small) -->
<Part Name="ButtonPartSmall"
PartKind="Button"
ViewModel="{ViewModel Name=ButtonPartViewModel, Module=./Intrinsic/ViewModels/ButtonPartViewModel}"
InitialSize="Small" />
- The ViewModel that is associated with the pdl will plug data into the part. The ViewModel is located at
<dir>\Client\V1\Parts\Intrinsic\ViewModels\ButtonPartViewModel.ts
For this step, the data is just the label and icon, but for more data-oriented parts, the data can be gathered from a server, like a resource provider. The ViewModel is in the following code.
/**
* This sample uses the base class implementation. You can also implement the
* interface MsPortalFx.ViewModels.ButtonPartContract.
*/
@Di.Class("viewModel")
export class ButtonPartViewModel extends MsPortalFx.ViewModels.ButtonPart {
/**
* Initialize the part.
*/
constructor(container: MsPortalFx.ViewModels.PartContainerContract) {
super();
this.title(ClientResources.AssetTypeNames.Robot.singular);
this.shortTitle(ClientResources.AssetTypeNames.Robot.singular);
this.description(ClientResources.robotDescription);
this.icon(CommonImages.robot);
container.assetName(ClientResources.robotManufacturerBotsAreUs);
}
}
Unlike intrinsic parts, custom parts use html
templates that are bound to the view model. The developer defines the look and feel in addition to the data loading for the part. Templates also can use other controls that are provided by the framework, as specified in top-extensions-controls.md.
The following is an example of a custom part. The three files used for the custom part example are the following.
-
<dir>\Client\V1\Parts\Custom\CustomParts.pdl
-
<dir>\Client\V1\Parts\Custom\Templates\ExampleCustomPart.html
-
<dir>\Client\V1\Parts\Custom\ViewModels\ExampleCustomPartViewModel.ts
A working copy is located at https://aka.ms/portalfx/samples#blade/SamplesExtension/SDKMenuBlade/custompart.
The following procedure demonstrates how to use a custom part.
- Declare the part in the global
<Definition>
section of the PDL for the extension, as in the following example.
<CustomPart Name="ExampleCustomPart"
ViewModel="{ViewModel Name=ExampleCustomPartViewModel, Module=./Custom/ViewModels/ExampleCustomPartViewModel}"
Template="{Html Source='Templates\\ExampleCustomPart.html'}"
InitialSize="Large">
<StyleSheet Source="{Css Source='Styles\\ExampleStyles.css'}" />
</CustomPart>
- The pdl points to the html template.
<h3>This is a custom part</h3>
<p>
Number of clicks: <strong data-bind="text: numberOfClicks"></strong>
</p>
<div data-bind="visible: allowMoreClicks">
<button data-bind="fxclick: increaseClickCount">Click me</button>
</div>
<div class="ext-too-many-clicks-box" data-bind="visible: !allowMoreClicks()">
That's too many clicks!
<button data-bind="fxclick: resetClickCount">Reset</button>
</div>
<ul data-bind="foreach: myButtons">
<li>
<button data-bind="text: displayName, fxclick: $parent.buttonClickHandler"></button>
Number of clicks: <strong data-bind="text: clicked"></strong>
</li>
</ul>
- The HTML template is bound to the following ViewModel by using Knockout, which is also referred to in the pdl.
/**
* Example view model for a custom part
*/
@Di.Class("viewModel")
export class ExampleCustomPartViewModel {
// Public properties bound to the UI in the part's template
public numberOfClicks = ko.observable(0);
public allowMoreClicks = ko.pureComputed(() => {
return this.numberOfClicks() < 3;
});
/**
* Constructs an instance of the custom part view model.
*/
constructor() {
}
public increaseClickCount(): void {
const currentCount = this.numberOfClicks();
this.numberOfClicks(currentCount + 1);
}
public resetClickCount(): void {
this.numberOfClicks(0);
}
public myButtons = ko.observableArray([
{
displayName: ko.observable("First button"),
clicked: ko.observable(0),
},
{
displayName: ko.observable("Second button"),
clicked: ko.observable(0),
},
]);
public buttonClickHandler = () => {
this.numberOfClicks(this.numberOfClicks() + 1);
};
}
Tiles are added to dashboards by using the part gallery, which is also known as the tile gallery. The tile gallery is displayed when the Edit Dashboard command is clicked, or when parts on the dashboard are rearranged or resized.
To register a part with the gallery, you need to add the PartGalleryInfo
tag inside the <Part>
or <CustomPart>
tag, as in the file located at <dir>\Client/V1/Parts/TileGallery/TileGallery.pdl
and in the following code.
<PartGalleryInfo
Title="{Resource generalGalleryPartTitle, Module=ClientResources}"
Category="{Resource partGalleryCategorySample, Module=ClientResources}"
Thumbnail="{Svg Type=Favorite}"
AutoConfigSelectablePath="configOnDropSelectable"/>
The Title, Category, and Thumbnail parts are reasonably intuitive.
Some tile experiences require that parts are configured when they are dropped from the tile gallery. The AutoConfigSelectablePath property performs this task for parts that have this requirement.
The following is the path to a selectable that is located inside the ViewMmodel and is immediately set to true
by the Framework when the part is dropped on a dashboard. The selectable can be configured to open a context blade for configuration. The example located at <dir>/Client/V1/Parts/TileGallery/ViewModels/GeneralGalleryPart.ts
demonstrates how this type of selectable is configured. The example code is in the following code.
// Configure the HotSpot's Selectable so it will be implicitly activated when the user drops this Part on a Dashboard.
const bladeSelection: FxViewModels.DynamicBladeSelection = {
detailBlade: ExtensionDefinition.BladeNames.pdlGeneralGalleryPartConfigurationBlade,
detailBladeInputs: {},
};
const hotSpotSelectable = new FxViewModels.Selectable({
selectedValue: bladeSelection,
});
hotSpotSelectable.getDefaultSelection = () => {
return Promise.resolve(bladeSelection);
};
this.configureHotSpot.selectable = hotSpotSelectable;
this.configOnDropSelectable = hotSpotSelectable;
// Create a ParameterCollector that will open the configure Blade to modify 'configuration' -- this Part's Configuration.
const configuration = container.activateConfiguration<Inputs, Def.SettingsContract>();
const collector = new FxViewModels.ParameterCollector<PartConfiguration>(container, {
selectable: hotSpotSelectable,
// The Parts Configuration values are sent to the Provider Blade to be edited by the user.
supplyInitialData: configuration.getValues.bind(configuration),
// The edited Configuration values are returned from the Provider Blade and updated in this Part.
// Any edits will cause 'onInputsSet' to be called again, since this is the method where the Part receives a new, consistent
// set of inputs/settings.
receiveResult: configuration.updateValues.bind(configuration),
});
// This Selectable must be dynamically registered due to a PDL compiler bug that rejects any <BladeAction> that opens a
// <ContextBlade> from a HotSpot.
container.registerSelectable(
container,
"GeneralGalleryPartConfigSelectable",
hotSpotSelectable,
{
openInContextPane: true,
parameterCollector: collector,
});
If the part that is being developed is associated with an Ibiza asset like an ARM resource, then it should be associated with an asset type and have a single input definition whose IsAssetId
property is true
. If this is not the case then the part will appear in the General category of the part gallery.
There is a significant amount of flexibility when sizing extension tiles. All size options are included in the <CustomPart>
tag in the PDL file located at <dir>\Client/V1/Parts/PartSizes/PartSizes.pdl
.
A working copy is located at https://aka.ms/portalfx/samples#blade/SamplesExtension/SDKMenuBlade/partsizes.
-
The following part supports only the large, standard size.
<CustomPart Name="LargePart"
ViewModel="{ViewModel Name=PartSizesLargePartViewModel, Module=./PartSizes/ViewModels/PartSizesViewModels}"
Template="{Html Source='Templates\\SizeAwarePart.html'}"
InitialSize="Large">
-
The following part supports multiple, standard sizes.
<CustomPart Name="MiniPart"
ViewModel="{ViewModel Name=PartSizesMiniPartViewModel, Module=./PartSizes/ViewModels/PartSizesViewModels}"
Template="{Html Source='Templates\\SizeAwarePart.html'}"
InitialSize="Mini">
<CustomPart.SupportedSizes>
<PartSize>Mini</PartSize>
<PartSize>Normal</PartSize>
</CustomPart.SupportedSizes>
</CustomPart>
- The following part defaults to an arbitrary size, and can be resized by the user. The Framework automatically adds a drag handle to this part because of the value
ResizeMode="User"
.
<CustomPart Name="CustomSizeUserResizePart"
ViewModel="{ViewModel Name=PartSizesCustomSizeUserResizePartViewModel, Module=./PartSizes/ViewModels/PartSizesViewModels}"
Template="{Html Source='Templates\\SizeAwarePart.html'}"
InitialSize="Custom"
InitialWidth="5"
InitialHeight="2"
ResizeMode="User">
</CustomPart>
- The following part defaults to an arbitrary size, and can be resized programatically.
<CustomPart Name="CustomSizeProgrammaticResizePart"
ViewModel="{ViewModel Name=PartSizesCustomSizeProgrammaticResizePartViewModel, Module=./PartSizes/ViewModels/PartSizesViewModels}"
Template="{Html Source='Templates\\SizeAwareResizablePart.html'}"
InitialSize="Custom"
InitialWidth="6"
InitialHeight="3"
ResizeMode="Programmatic">
<CustomPart.SupportedSizes>
<PartSize>Tall</PartSize>
<PartSize>Mini</PartSize>
<PartSize>Wide</PartSize>
<PartSize>Large</PartSize>
</CustomPart.SupportedSizes>
</CustomPart>
- The following code demonstrates how to programatically resize the part from within the associated ViewModel. The parameters are specified in grid units instead of pixels. The code is located at
<dir>\Client\V1\Parts\PartSizes\ViewModels\PartSizesViewModels.ts
.
onClick: () => {
container.resizeTo(resizeA.width, resizeA.height);
},
});
Extensions can read and write settings that are saved whenever the user saves a dashboard. Private dashboards are located in the Ibiza user settings service. Shared dashboards are stored in ARM as Azure resources inside the MS.Portal resource provider.
The dashboard is saved as a JSON
document that contains the list of parts on the dashboard, in addition to their sizes, positions, and settings. It also contains dashboard-level settings like the time range that is shared by many monitoring charts.
The following example located at <dir>/Client/V1/Parts/TileGallery/TileGallery.pdl
demonstrates the use of the <CustomPart.PartSettings>
tag to declare settings in the PDL file. It is also in the following code.
<CustomPart.Settings>
<Setting Property="colorSettingValue" />
<Setting Property="fontSettingValue" />
</CustomPart.Settings>
The following is the TypeScript code that reads and writes settings. It is also located at <dir>Client/V1/Parts/TileGallery/ViewModels/GeneralGalleryPart.ts
.
import * as ClientResources from "ClientResources";
import * as ExtensionDefinition from "_generated/ExtensionDefinition";
import Def = ExtensionDefinition.ViewModels.V1$$Parts.GeneralGalleryPart;
import FxViewModels = MsPortalFx.ViewModels;
import PartContainerContract = FxViewModels.PartContainerContract;
import FxConfiguration = MsPortalFx.Composition.Configuration;
import TimeUnit = FxConfiguration.TimeUnit;
// OBSOLETE API IMPORTS
const ObsoleteHotSpot: any = MsPortalFx.ViewModels.Controls.HotSpot;
type ObsoleteHotSpotViewModel = any;
// We have to explicitly define our Inputs contract here rather than use Def.InputsContract since there is a PDL
// compiler bug where <Part.InputDefinitions> are not represented on Def.InputsContract.
export interface Inputs {
timeRange: FxConfiguration.TimeRange;
otherParameter: string;
}
// We have to use this over Def.Settings because Def.Settings includes an old 'content' property that is no longer
// important to the Part Configuration design re: Part Settings.
export type Settings = Def.Settings$content$0;
// This will be used by the GeneralGalleryPartConfigurationPart too.
export type PartConfiguration = FxConfiguration.Part.ValuesWithMetadata<Inputs, Settings>;
export enum BackgroundColor {
Default,
Blue,
Green,
Yellow,
}
export enum FontStyle {
Default,
Muted,
AllCaps,
}
@Di.Class("viewModel")
export class GeneralGalleryPart implements Def.Contract {
public configOnDropSelectable: FxViewModels.Selectable<FxViewModels.DynamicBladeSelection>;
public configureHotSpot: ObsoleteHotSpotViewModel;
public timeRange = ko.observable<string>();
public otherParameter = ko.observable<string>();
public css: KnockoutObservableBase<any>; // For the 'css' binding in the corresponding HTML template.
public location: string;
// These are required by the portal presently. Re: Part Settings, the Part below works exclusively in terms of
// 'configuration.updateValues' to update settings values and 'onInputsSet(..., settings)' to receive settings values.
public colorSettingValue = ko.observable<BackgroundColor>();
public fontSettingValue = ko.observable<FontStyle>();
// These store the raw color and font style values supplied to the Part.
private _colorSetting = ko.observable<BackgroundColor>();
private _fontSetting = ko.observable<FontStyle>();
constructor(container: PartContainerContract) {
container.partTitle(ClientResources.generalGalleryPartTitle);
// Create the HotSpot control that the user will click.
this.configureHotSpot = new ObsoleteHotSpot.ViewModel(container);
this.configureHotSpot.clickableDuringCustomize = true;
//parts#PartGalleryConfigOnDropDoc
// Configure the HotSpot's Selectable so it will be implicitly activated when the user drops this Part on a Dashboard.
const bladeSelection: FxViewModels.DynamicBladeSelection = {
detailBlade: ExtensionDefinition.BladeNames.pdlGeneralGalleryPartConfigurationBlade,
detailBladeInputs: {},
};
const hotSpotSelectable = new FxViewModels.Selectable({
selectedValue: bladeSelection,
});
hotSpotSelectable.getDefaultSelection = () => {
return Promise.resolve(bladeSelection);
};
this.configureHotSpot.selectable = hotSpotSelectable;
this.configOnDropSelectable = hotSpotSelectable;
// Create a ParameterCollector that will open the configure Blade to modify 'configuration' -- this Part's Configuration.
const configuration = container.activateConfiguration<Inputs, Def.SettingsContract>();
const collector = new FxViewModels.ParameterCollector<PartConfiguration>(container, {
selectable: hotSpotSelectable,
// The Parts Configuration values are sent to the Provider Blade to be edited by the user.
supplyInitialData: configuration.getValues.bind(configuration),
// The edited Configuration values are returned from the Provider Blade and updated in this Part.
// Any edits will cause 'onInputsSet' to be called again, since this is the method where the Part receives a new, consistent
// set of inputs/settings.
receiveResult: configuration.updateValues.bind(configuration),
});
// This Selectable must be dynamically registered due to a PDL compiler bug that rejects any <BladeAction> that opens a
// <ContextBlade> from a HotSpot.
container.registerSelectable(
container,
"GeneralGalleryPartConfigSelectable",
hotSpotSelectable,
{
openInContextPane: true,
parameterCollector: collector,
});
//parts#PartGalleryConfigOnDropDoc
// For fringe cases, this illustrates how the Part can understand whether it is located on a Dashboard or a Blade.
// Importantly, the Part behavior shouldn't change between Dashboard and Blade.
this.location = container.location === FxViewModels.PartLocation.Dashboard ?
ClientResources.generalGalleryPartDashboardLocation :
ClientResources.generalGalleryPartBladeLocation;
// Data-driven styling for the Part.
this.css = ko.computed(container, () => {
const colorSetting = this._colorSetting();
const fontSetting = this._fontSetting();
return {
"msportalfx-bgcolor-h2": colorSetting === BackgroundColor.Blue,
"msportalfx-bgcolor-i2": colorSetting === BackgroundColor.Green,
"msportalfx-bgcolor-j2": colorSetting === BackgroundColor.Yellow,
"msportalfx-text-muted-50": fontSetting === FontStyle.Muted,
"msportalfx-text-header-small": fontSetting === FontStyle.AllCaps,
};
});
}
public onInputsSet(inputs: Inputs, settings: Def.SettingsContract): Promise<any> {
// Any changes to the Part's Configuration values (see 'updateValues' above) will cause 'onInputsSet' to be called with the
// new inputs/settings values.
this.timeRange(inputs.timeRange ? timeRangeToString(inputs.timeRange) : ClientResources.generalGalleryPartNone);
this.otherParameter(inputs.otherParameter || ClientResources.generalGalleryPartNone);
this._colorSetting(settings && settings.content && settings.content.colorSettingValue || BackgroundColor.Default);
this._fontSetting(settings && settings.content && settings.content.fontSettingValue || FontStyle.Default);
return null;
}
}
function timeRangeToString(timeRange: FxConfiguration.TimeRange): string {
if (timeRange.relative) {
const duration = timeRange.relative.duration;
const plural = duration > 1;
let timeUnit: string;
switch (timeRange.relative.timeUnit) {
case TimeUnit.Minute:
timeUnit = plural ? ClientResources.timeUnitMinutes : ClientResources.timeUnitMinute;
break;
case TimeUnit.Hour:
timeUnit = plural ? ClientResources.timeUnitHours : ClientResources.timeUnitHour;
break;
case TimeUnit.Day:
timeUnit = plural ? ClientResources.timeUnitDays : ClientResources.timeUnitDay;
break;
case TimeUnit.Week:
timeUnit = plural ? ClientResources.timeUnitWeeks : ClientResources.timeUnitWeek;
break;
case TimeUnit.Month:
timeUnit = plural ? ClientResources.timeUnitMonths : ClientResources.timeUnitMonth;
break;
case TimeUnit.Year:
timeUnit = plural ? ClientResources.timeUnitYears : ClientResources.timeUnitYear;
break;
}
return "{0} {1} {2}".format(ClientResources.timeRangeLast, duration, timeUnit);
} else {
return "{0} - {1}".format(timeRange.absolute.from, timeRange.absolute.to);
}
}
Sometimes parts are displayed for which no data is available. For example, an extension may display a prototype 'deployment history' that contains sample data, previous to the time when the user enables deployments for the extension. To support this, part container
objects use the noDataMessage
property.
The following example populates the part with sample data and set noDataMessage
to a non-empty string.
container.noDataMessage("Enable deployments to see your history");
In this example, the part is grayed-out and is non-interactive. The message is displayed on top of the part. To remove the message, set the noDataMessage
value back to null
.
This feature informs the user that the feature exists, although no data is available yet. If the extension needs to disable a part while the data is loading, it should return a promise from the onInputsSet
method or use the container.operations
queue.
By default, all blades and parts are 'pinnable'. Pinning a part creates a copy of that part on the dashboard. The part on the dashboard provides a shortcut for users, allowing them to get their most used blades, as in the following example.
All parts are pinnable by default, at little or no cost to the developer. Part settings are copied to the new part, and the current size is maintained. The new part is a complete copy of the original part, and can be independently configured, resized, or moved around on the dashboard.
When a part is pinned or moved to another blade, a new ViewModel for that part is created. This ViewModel will have the exact same contract. The inputs to the part are stored in the Portal's cloud settings, and replayed onto the part when the dashboard loads. For example, the robots part in this sample takes a single input named id
, which comes from the blade. When the part is pinned to the dashboard, the blade will obviously not be available, therefore the id
of the part is stored in persistent storage. The id
of the part is in the pdl file located at <dir>\SamplesExtension\Extension\Client\Hubs\Browse\Browse.pdl
, and it is also in the following code.
<CustomPart Name="RobotSummary"
ViewModel="{ViewModel Name=RobotSummaryViewModel, Module=./Browse/ViewModels/RobotSummaryViewModel}"
Template="{Html Source='Templates\\Robot.html'}"
InitialSize="FullWidthFitHeight"
AssetType="Robot"
AssetIdProperty="name">
<CustomPart.Properties>
<Property Name="name"
Source="{BladeParameter Name=id}" />
</CustomPart.Properties>
</CustomPart>
Model data is not sent in bindings because the extension is required to store specific types of data in persistent storage. The only objects that should be sent as bindings between the extension and the various parts are the keys and ids that are used to load data from a back end server. If data that specifies the robot changes, like the name, those changes should be reflected in the part. This provides the 'live tile' feel of the Portal, and ensures that the part data is not stale.
For more information about keys and data loading in the Portal, see top-legacy-data.md.
In some cases, a part should not be pinnable. Some instances are as follows.
- The part is showing an editable form
- The part is not performant enough to run persistently
- The part should not be displayed in a constrained UX
- The part has no value when pinned
Parts are not individually flagged as not being pinnable. Rather, a blade that contains those parts is locked
, as in the sample located at <dir>\SamplesExtension\Extension\Client\extension.pdl
and in the following code.
<Blade Name="SamplesBlade"
ViewModel="SamplesBladeViewModel"
Pinnable="True"
Locked="True">
NOTE: All parts in the previous SamplesBlade
will not provide the ability to be pinned. However, the blade itself can still be pinned, as specified in portalfx-blades-pinning.md.
For more information, about sharing parts, see portalfx-extensibility-pde.md.
When users customize or pin a part, the following states are stored and used the next time the part is loaded from a customized context.
- Basic part metadata, like part name or extension name
- Part inputs, like resource id
- Part settings, like time range for a chart
Because these states are stored, these parts need to be backwards-compatible.
Supporting new functionality may require the modification of the schema of a part's inputs and settings.
The Azure Portal always calls the most recent edition of an extension, which is specified in the extensions configuration file. However, older versions of inputs and settings that were stored by earlier editions of an extension may still exist, and they may be incompatible with the most recent edition of the extension. Consequently, users may experience unexpected results when the extension or part is called with previous settings.
Likewise, other extensions may have taken dependencies on less-recent editions of the extension or part. For example, another extension may use a .pde file that contains a <PartReference/>
. Those other extensions may also experience unexpected results when they call the extension or part with previous inputs.
NOTE: In this discussion, <dir>
is the SamplesExtension\Extension\
directory, and <dirParent>
is the SamplesExtension\
directory, based on where the samples were installed when the developer set up the SDK. If there is a working copy of the sample in the Dogfood environment, it is also included.
This example is based on the sample located at <dir>\Client\V1\Hubs\Browse\Browse.pdl
. It builds on the ViewModel located at <dir>\Client\V1\Hubs\Browse\ViewModels\RobotPartViewmodel.ts
.
The CanUseOldInputVersion attribute can be set to true
to specify that the part can process older versions of inputs. It should be used in conjunction with the part property named version
, as in the following example.
<Part Name="RobotPart"
ViewModel="{ViewModel Name=RobotPartViewModel, Module=./Browse/ViewModels/RobotPartViewModel}"
CanUseOldInputVersions="true"
PartKind="Button"
AssetType="Robot"
AssetIdProperty="name">
<Part.Properties>
<Property Name="version"
Source="{Constant 2}" />
<Property Name="name"
Source="{DataInput Property=id}" />
</Part.Properties>
</Part>
NOTE: Inline parts can specify the current version as a constant. Although this is the first explicit version, we recommend using a value of 2
the first time it is used.
Globally-defined parts can not specify constant bindings, but the flow is mostly the same.
<CustomPart Name="RobotSummary2"
Export="true"
ViewModel="RobotSummaryViewModel"
CanUseOldInputVersions="true"
Template="{Html Source='Templates\\Robot.html'}"
InitialSize="FullWidthFitHeight">
<CustomPart.Properties>
<Property Name="name"
Source="{DataInput Property=name}" />
<Property Name="version"
Source="{DataInput Property=version}" /> <!-- currently 2 -->
</CustomPart.Properties>
</CustomPart>
The following code demonstrates how to process explicitly-versioned inputs, in addition to the version of the parts that existed previous to the addition of explicit versioning support.
public onInputsSet(inputs: Def.InputsContract, settings: Def.SettingsContract): Promise<any> {
var name: string;
if (inputs.version === "2") { // this block explicitly handles version 2, which is the latest
name = inputs.name;
} else if (inputs.version === "1") { // this block explicitly handles version 1, which is now old, but was an explicit version
var oldInputs: any = inputs;
name = oldInputs.oldName;
} else if (MsPortalFx.Util.isNullOrUndefined(inputs.version)) { // this block handles any version of the inputs
var oldestInputs: any = inputs; // that existed before the version property was added
name = oldestInputs.oldestName;
} else {
throw new Error("Unexpected version. This should not happen, but there is one edge case where you temporarily deploy a new version, say version 3, and then roll back your code to version 2. Any tiles pinned before you roll back will hit this block.");
}
return this._view.fetch(name);
}
The same technique can be used for part settings as in the following example.
public onInputsSet(inputs: Def.InputsContract, settings: Def.SettingsContract): Promise<any> {
var someSetting: string;
if (settings.version === "2") { // this block explicitly handles version 2, which is the latest
someSetting = settings.someSetting;
} else if (settings.version === "1") { // this block explicitly handles version 1, which is now old, but was an explicit version
var oldSettings: any = settings;
someSetting = oldSettings.oldSetting;
} else if (MsPortalFx.Util.isNullOrUndefined(settings.version)) { // this block handles any version of the settings
var oldestSettings: any = string; // that existed before the version property was added
someSetting = oldestSettings.oldestSetting;
} else {
throw new Error("Unexpected version. This should not happen, but there is one edge case where you temporarily deploy a new version, say version 3, and then roll back your code to version 2. Any tiles pinned before you roll back will hit this block.");
}
return this._view.fetch(name);
}
Developers occasionally build and ship parts, and later discontinue their functionality. However, there may be cases where these parts were pinned and incorporated into the layout of a user's dashboard.
Azure customers have informed the Azure team that parts disappearing automatically from their startboards is an extremely dissatisfactory experience. To address this customer request, Azure has created a process that allows developers to permanently discontinue a part while providing a good user experience that uses customizations.
To discontinue a part, developers delete the majority of the code from the ViewModel
, but leave the constructor and onInputsSet
. in place so that the tile still loads.
Then use the container.noDataMessage()
in the constructor to inform the user that the part is no longer supported, and return empty promise from onInputsSet
.
This ensures that customers are informed that this part is no longer supported, and that parts that fail will not be displayed on their dashboards. An example is in the following code.
export class DocumentCountUsagePartViewModel extends MsPortalFx.ViewModels.Parts.SingleValueGauge.ViewModel {
constructor(container: MsPortalFx.ViewModels.PartContainerContract, initialState: any, dataContext: DataContext) {
super();
container.noDataMessage(ClientResources.tileRemoved);
}
public onInputsSet(inputs: any): Promise<any> {
return Q();
}
}
In the PDL code, make the part global by moving it from inside <Blade>/<Lens>
tags to be the child of the <Definition />
tag in the global *.pdl
file. Then, rename this part, and create a <RedirectPart />
tag that uses the old name.
The following example is the code before and after it was rewritten to obsolete the old part.
-
Before
<Definition...> <Blade Name="MyBlade" ...> <Lens Name="MyLens" ...> <Part Name="DocumentCountUsagePartViewModel" PartKind="SingleValueGauge" ViewModel="DocumentCountUsagePartViewModel" InitialSize="Wide" AssetType="SearchService" AssetIdProperty="resourceId"> <Part.Permissions> <PermissionReference AssetType="SearchService" Permission="read"/> </Part.Permissions> <Part.Properties> <Property Name="resourceId"> <BladeParameter Name="id"/> </Property> </Part.Properties> </Part> </Lens> </Blade> </Definition>
-
After
<Definition...> <Blade Name="MyBlade" ...> <Lens Name="MyLens" ...> </Lens> </Blade> <RedirectPart Name="DocumentCountUsagePart" Blade="MyBlade" Lens="MyLens"> <NewPart PartType="DocumentCountUsagePartGlobal" /> </RedirectPart> <Part Name="DocumentCountUsagePartViewModelGlobal" PartKind="SingleValueGauge" ViewModel="DocumentCountUsagePartViewModel" InitialSize="Wide" AssetType="SearchService" AssetIdProperty="resourceId"> <Part.Permissions> <PermissionReference AssetType="SearchService" Permission="read"/> </Part.Permissions> <Part.Properties> <Property Name="resourceId"/> </Part.Properties> </Part> </Definition>
NOTE: The <BladeParameter />
element was removed from the <Property />
tag because it is not allowed for global parts.
An unlocked blade's default layout should consist of tiles that provide the most value to users and still meet extension performance goals out-of-the-box. That layout may change over time, and your team may decide that a part that was included in a blade's default layout should be removed.
If the part was defined inline as a <Part/>
or <CustomPart>
element within a <Blade/>
and <Lens/>
, then the part should be moved out of the blade and into the global part catalog for the extension. Otherwise, if the part is already defined in the global part catalog, or is defined in another extension, then the pdl file may contain a <PartReference/>
tag for the blade, instead of a <Part/>
tag.
NOTE: It is best practice to use Typescript or no-pdl parts.
The following procedure to remove a part from a blade layout.
-
Remove the
<Part/>
or<PartReference/>
tag from the extension configuration or pdl file. -
If the goal was to permanently discontinue a part, including removing support for pinned instances and the tile gallery, then follow the procedure specified in #permanently-discontinue-a-part. Otherwise, to continue supporting instances of the part that have been pinned to user startboards, or to allow new users to find the part in the tile gallery, replace the part tag with a global definition for a
<RedirectPart/>
tag, as in the following xml.
<RedirectPart Name="SAME EXACT PART NAME THAT IS BEING REDIRECTED FROM"
Blade="EXACT BLADE NAME THAT THE PART WAS DEFINED IN"
Lens="OPTIONAL - EXACT LENS NAME THE PART WAS DEFINED IN"
Extension="OPTIONAL - ONLY APPLICABLE IF THE PART IS DEFINED IN A DIFFERENT EXTENSION">
<NewPart PartType="NAME OF THE NEW GLOBAL PART THAT DEFINES THE PART BEHAVIOR" />
</RedirectPart>
As a Part loads, by default, the user is presented with a blocking loading indicator:
![Part with blocking loading indicator][opaque] [opaque]: ../media/portalfx-parts-opaquespinner.png
By default, the lifetime of this blocking loading indicator is controlled by the promise returned from the Part's onInputsSet
method:
public onInputsSet(inputs: Def.InputsContract): Promise<any> {
// When this promise is resolved, the loading indicator is removed.
return this._view.fetch(inputs.websiteId);
}
Now, it is quite common that a Part's content can be revealed (that is, the blocking loading indicator can be removed) earlier, allowing the user to interact with the Part while its data continues to load. An example of this looks like:
![Part with non-blocking loading indicator][translucent] [translucent]: ../media/portalfx-parts-translucentspinner.png
Here, essential content - like the "Accounts" SQL Database name - is displayed to the user while non-essential "status" (bottom left) loads in the background. While "status" loads, a non-blocking loading indicator is displayed at the top of the Part. Crucially, the user can activate the Part (can interact with the Part) while the Part is in this non-blocking loading state.
To optimize your Part to behave in this more responsive way, you'll use the container.revealContent()
API from within your Part's view model. A call to this API will:
- remove the blocking loading indicator
- reveal the Part's content
- apply the non-blocking loading indicator
- allow the user to interact with your Part
Depending on the nature of your Part, you will call container.revealContent()
:
- from your Part view model's
constructor
or - from your Part view model's
onInputsSet
function:- in a '.then(() => ...)' callback, once essential data has loaded
- in 'onInputsSet' directly, before initiating data-loading
You'll call container.revealContent()
from your view model's constructor
in scenarios where the Part has interesting content to display even before any data is loaded. An example of this would be a chart that can show its X and Y-axis immediately:
export class BarChartPartViewModel implements Def.BarChartPartViewModel.Contract {
public barChartVM: MsPortalFx.ViewModels.Controls.Visualization.Chart.Contract<string, number>;
constructor(container: MsPortalFx.ViewModels.PartContainerContract, initialState: any, dataContext: ControlsArea.DataContext) {
// Initialize the chart view model.
this.barChartVM = new MsPortalFx.ViewModels.Controls.Visualization.Chart.ViewModel<string, number>(container);
// Configure the chart view model (incomplete as shown).
this.barChartVM.yAxis.showGridLines(true);
container.revealContent();
}
}
More often, you'll call container.revealContent()
once some essential, fast-loading data is loaded:
public onInputsSet(inputs: MyPartInputs): Promise {
// This larger Promise still controls the lifetime of all loading indicators (the
// non-blocking one in this case, since 'container.revealContent()' has been called).
return Q.all([
this._essentialDataView.fetch(inputs.resourceId).then(() => {
// Show the Part content once essential, fast-loading data loads.
this._container.revealContent();
}),
this._slowLoadingNonEssentialDataView.fetch(inputs.resourceId)
]);
}
Less commonly, you'll call container.revealContent()
when the essential data you'll display can be computed synchronously from the Part/Blade inputs:
public onInputsSet(inputs: MyPartInputs): Promise {
// In this case, the 'resourceGroupName' is sufficient to allow the user to interact with the Part/Blade.
var resourceDescriptor = ResourceTypes.parseResourceManagerDescriptor(inputs.resourceId);
this.resourceGroupName(resourceDescriptor.resourceGroup);
this._container.revealContent();
// This Promise controls the lifetime of all loading indicators (the
// non-blocking one in this case, since 'container.revealContent()' has been called).
return this._dataView.fetch(inputs.resourceId);
}
In all cases above, the promise returned from onInputsSet
still determines the visibility/presence of loading indicators. Once the promise returned from onInputsSet
is resolved, all loading indicators are removed:
![Fully loaded Part with no loading indicator][nospinner] [nospinner]: ../media/portalfx-parts-nospinner.png
Also, if the promise returned from onInputsSet
is rejected (due to the rejection of either the fast- or slow-loading data promise), the Part will transition to show the default error UX (a "sad cloud").
This treatment of the promise returned from onInputsSet
behaves consistently, whether or not the Part makes use of container.revealContent()
. In this way, container.revealContent()
is a simple, additive change you should use to optimize your Part's behavior.
It is important that loading indicators are consistently applied across the Parts/Blades of all extensions. To achieve this:
Do call container.revealContent()
to limit the time where the user sees blocking loading indicators.
Do return a Promise from onInputsSet
that reflects all data-loading for your Part (or for your Blade, if the Blade is locked or <TemplateBlade>
).
Do not return a Promise from onInputsSet
that removes loading indicators before all Part data is loaded. While your Part data loads, if the user sees no loading indicator, your Part will appear broken and/or unresponsive. For instance,
public onInputsSet(inputs: MyPartInputs): Promise {
this._view.fetch(inputs.resourceId);
// DO NOT DO THIS! Removes all loading indicators.
// Your Part will look broken while the `fetch` above completes.
return Q();
}
Occasionally while loading parts, an extension may encounter an unrecoverable error. In that case, the part may be placed into a failure state, as in the following image.
Parts should only be placed into a failed state if there was a system fault and no action can be taken by the user to correct the error. If the user can correct the error, then the extension should display guidance about the error, as in the example located at <dir>\Client\V1\Parts\Lifecycle\ViewModels\PartLifecycleViewModels.ts
, and in the following code.
constructor(container: MsPortalFx.ViewModels.PartContainer, initialState: any, dataContext: DataContext) {
container.fail(SamplesExtension.Resources.Strings.failedToLoad);
}
When the error is fixed, then the extension can call container.recover()
to return the part to its normal display state. One example is that the extension is polling for data, and the first poll does not retrieve results, but a subsequent poll returns valid results.
Many parts represent assets like ARM resources that can be deleted from the UI, PowerShell, or the calling REST APIs. A stateless UI system handles this deletion by loading only assets that exist at the time the UI starts up. Because Ibiza contains the state for all user customizations, this 'Not Found' case is handled in a few specific places. Some examples are as follows.
-
A VM part that was pinned to the startboard represents an asset that no longer exists
-
The CPU chart for a VM part that was pinned to the startboard depends on information provided by an asset that no longer exists
If this is the case, see portalfx-extensions-status-codes.md#server-error-404.
NOTE: Instances of 'Not Found' do not count against a part's reliability KPI.
Portal development patterns or architectures that are recommended based on customer feedback and usability studies may be categorized by the type of part.
Loading indicators should be consistently applied across all blades and parts of the extension. To achieve this:
-
Call
container.revealContent()
to limit the time when the part displays blocking loading indicators. -
Return a
Promise
from theonInputsSet
method that reflects all data-loading for the part. Return thePromise
from the blade if it is locked or is of type<TemplateBlade>
. -
Do not return a
Promise
from theonInputsSet
method previous to the loading of all part data if it removes loading indicators. The part will seem to be broken or unresponsive if no loading indicator is displayed while the data is loading, as in the following code.
public onInputsSet(inputs: MyPartInputs): Promise {
this._view.fetch(inputs.resourceId);
// DO NOT DO THIS! Removes all loading indicators.
// Your Part will look broken while the `fetch` above completes.
return Q();
}
The sad cloud UX is displayed when there is no meaningful error to display to the user. Typically this occures when the error is unexpected and the only option the user has is to try again.
If an error occurs that the user can do something about, then the extension should launch the UX that allows them to correct the issue. Extension developers and domain owners are aware of how to handle many types of errors.
For example, if the error is caused because the user's credentials are not known to the extension, then it is best practice to use one of the following options instead of failing the part.
-
The part can handle the error and change its content to show the credentials input form
-
The part can handle the error and show a message that says ‘click here to enter credentials’. Clicking the part would launch a blade with the credentials form.