はじめに
前回の記事で、Flexible Programming Modelを使って以下を実現しようとしました。
- List Reportで"Create"ボタンを押したらカスタムページを開く
- テーブル行をクリックしたらObject Pageを開く
しかし、この構成では「Object Pageを編集モードに切り替えると、カスタムページが表示されてしまう」という問題がありました。前回の記事では代替案として以下の方法を考えました。
案1
- 登録用のカスタムページは既存のページとは別のURLパターンで開くようにする
- カスタムの「登録」ボタンを作成し、カスタムボタンが押されたらカスタムページを開く
案2
- 登録画面はBuilding Blocksを使用して独立したアプリケーションとして作り、カスタムの「登録」ボタンが押されたら登録画面にナビゲーションする
今回の構成
今回は案2をベースに実装することとしました。カスタムの登録ボタンを出すにあたり標準の登録ボタンをどうやって消そうかと考えた結果、List Reportもカスタムページとすることにしました。当初は案1(リストと登録画面が1つのアプリ)で考えていましたが、それだと登録画面に独自のURLパターンを使えない(※)ことがわかったため、登録画面を別アプリとしました。
※登録画面に独自のURLパターンを使えない制約について
Flexible Programming Modelで、新規エンティティの登録はEditFlowのcreateDocumentメソッドで行います。
createDocumentのパラメータであるcreationModeには以下の3つがあります。
- NewPage: Object Pageを開く
- Inline: List Reportに行追加する
- External: 外部のアプリケーションを開く
NewPageの場合、createDocumentを実行した時点でURLパターンが自動的にEntity名(key)
になります。このURLパターンではもともとのObjedct Pageが開いてしまうため、Externalを使用する必要があるという結論になりました。
前提
- OData V4を使用していること
- ドラフトが有効であること
以下のリポジトリにあるCAPのプロジェクトをベースに作成します。
https://github.com/miyasuta/flex-orders
UIアノテーションは以下のファイルで設定しています。
https://github.com/miyasuta/flex-orders/blob/main/srv/order-srv-ui.cds
ステップ
アプリ1
- Custom Pageのテンプレートでアプリを生成
- List Report風のカスタムページを作成
- 登録ボタン押下時の処理を実装
- Object Pageへのナビゲーションを実装
アプリ2
- Custom Pageのテンプレートでアプリを生成
- 登録処理の実装
- ルーティングの設定
- 登録用のWizardを作成
1. アプリ1
1.1. Custom Pageのテンプレートでアプリを生成
ウィザードで"Custom Page"を選択してアプリを生成します。
Custom Pageのテンプレートで作ると、extフォルダにビュー、コントローラーが作成されます。
1.2. List Report風のカスタムページを作成
TechEd 2022のハンズオンを参考に、カスタムページをList Report風にしていきます。
https://github.com/SAP-samples/teched2022-DT181/blob/main/exercises/ex3/README.md
- namespaceに
xmlns:f="sap.f"
を追加
<mvc:View xmlns:core="sap.ui.core"
xmlns:mvc="sap.ui.core.mvc"
xmlns="sap.m"
xmlns:macros="sap.fe.macros"
xmlns:f="sap.f"
xmlns:html="http://www.w3.org/1999/xhtml" controllerName="flex.customlistreport.ext.main.Main">
<f:DynamicPage id="FilterBarDefault" class="sapUiResponsiveContentPadding">
<f:title>
<f:DynamicPageTitle id="_IDGenDynamicPageTitle1">
<f:heading>
<Title id="_IDGenTitle1" text="Manage Orders" level="H2" />
</f:heading>
</f:DynamicPageTitle>
</f:title>
<f:header>
<f:DynamicPageHeader id="_IDGenDynamicPageHeader1" pinnable="true">
<VBox id="_IDGenVBox1">
<macros:FilterBar
metaPath="@com.sap.vocabularies.UI.v1.SelectionFields"
id="FilterBar"
filterChanged=".handlers.onFiltersChanged" />
</VBox>
</f:DynamicPageHeader>
</f:header>
<f:content>
<macros:Table
id="myTable"
filterBar="FilterBar"
readOnly="true"
metaPath="@com.sap.vocabularies.UI.v1.LineItem" />
</f:content>
</f:DynamicPage>
実行すると、すでにList Reportらしい見た目になっています。
1.3. 登録ボタン押下時の処理を実装
- ビュー
フィルターバーに"Create"ボタンを表示させます。
<macros:Table
id="myTable"
filterBar="FilterBar"
readOnly="true"
metaPath="@com.sap.vocabularies.UI.v1.LineItem">
<macros:actions>
<macros:Action
key="customAction"
text="Create"
press=".onCreate"
requiresSelection="false"
/>
</macros:actions>
</macros:Table>
- コントローラー
コントローラーにボタン押下時の処理を実装します。EditFlowのcreateDocumentメソッドで、creationModeに"External"を指定します。outboundにはmanifest.json(後述)で定義するoutboundのナビゲーション先を指定します。creationModeを"External"にした場合、ドキュメントには記載がありませんでしたがhandlers.onChevronPressNavigateOutBound
というメソッドの実装がないとエラーになってしまいました。ここで何をしたらよいかわかりませんでしたが、とりあえず外部アプリへのナビゲーションを実行しました(結果、うまくいきました)。
※creationModeが"External"の場合、ナビゲーション元ではエンティティを作成しないので、普通に外部へのナビゲーションを実行してもよいです。たとえば、LineItemでDataFieldForIntentBasedNavigation
アノテーションを使用してナビゲーションの設定をしても同じ結果になります。
sap.ui.define(
[
'sap/fe/core/PageController'
],
function(PageController) {
'use strict';
return PageController.extend('flex.customlistreport.ext.main.Main', {
onCreate: function() {
const listBinding = this.getView().getModel().bindList("/Orders");
this.editFlow.createDocument(listBinding, {
creationMode: "External",
outbound: "create-wizard"
})
},
handlers: {
onChevronPressNavigateOutBound: function(oController, outbound, undefined, sCreatePath) {
oController.intentBasedNavigation.navigateOutbound(outbound);
}
}
});
}
);
- manifest.json
crossNavigationを以下のように設定します。
"crossNavigation": {
"inbounds": {
"flex-customlistreport-inbound": {
"signature": {
"parameters": {},
"additionalParameters": "allowed"
},
"semanticObject": "Order",
"action": "customlistreport",
"title": "{{flpTitle}}",
"subTitle": "{{flpSubtitle}}",
"icon": ""
}
},
"outbounds": {
"create-wizard": {
"semanticObject": "Order",
"action": "createWithWizard",
"additionalParameters": "allowed"
}
}
}
1.4. Object Pageへのナビゲーションを実装
Object Pageの追加およびナビゲーションはPage Mapの設定だけで完了し、コーディングは不要です。
Application Infoページから、Page Mapを開きます。
+をクリックしてページを追加します。
Page TypeにObjectPageを選択し、NavigationにCustom Pageと同じエンティティを選択してAddをクリックします。
manifest.jsonに自動的にObject Page用のルートが追加されます。
"routing": {
"routes": [
{
"name": "OrdersMain",
"pattern": ":?query:",
"target": "OrdersMain"
},
{
"name": "OrdersObjectPage",
"pattern": "Orders({OrdersKey}):?query:",
"target": "OrdersObjectPage"
}
],
"targets": {
"OrdersMain": {
"type": "Component",
"id": "OrdersMain",
"name": "sap.fe.core.fpm",
"options": {
"settings": {
"viewName": "flex.customlistreport.ext.main.Main",
"entitySet": "Orders",
"navigation": {
"Orders": {
"detail": {
"route": "OrdersObjectPage"
}
}
}
}
}
},
"OrdersObjectPage": {
"type": "Component",
"id": "OrdersObjectPage",
"name": "sap.fe.templates.ObjectPage",
"options": {
"settings": {
"entitySet": "Orders",
"navigation": {}
}
}
}
}
}
2. アプリ2
2.1. Custom Pageのテンプレートでアプリを生成
アプリ1と同様に、"Custom Page"のテンプレートでアプリを作成します。
2.2. 登録処理の実装
Mainのビューはアプリ起動時に最初に呼ばれます。ここで登録処理を実行して、登録用のWizard画面に遷移します。
Mainのビューは空です。
<mvc:View xmlns:core="sap.ui.core" xmlns:mvc="sap.ui.core.mvc" xmlns:macros="sap.fe.macros" controllerName="flex.customformentry.ext.main.Main" xmlns="sap.m">
<App id="app" />
</mvc:View>
Mainのコントローラーで登録処理を実行します。ポイントはonInitではなくrouterのattachPatternMatchedのタイミングでcreateDocumentを呼ぶことです。onInitのタイミングでは早すぎるためか、フレームワーク側の処理でエラーが起きていました。
sap.ui.define(
[
'sap/fe/core/PageController'
],
function(PageController) {
'use strict';
return PageController.extend('flex.customformentry.ext.main.Main', {
onInit: function() {
PageController.prototype.onInit.apply(this);
const router = this.getAppComponent().getRouter();
router.getRoute("OrdersMain").attachPatternMatched(this._onObjectMatched, this);
},
_onObjectMatched: function() {
const listBinding = this.getAppComponent().getModel().bindList("/Orders");
this.editFlow.createDocument(listBinding, {
creationMode: "NewPage"
})
}
});
}
);
2.3. ルーティングの設定
creationModeが"NewPage"の場合、URLパターンが自動的に"/Orders(key)"となります。よってこのパターンの時に登録用のWizard画面を開くようにルーティングの設定をします。
"routing": {
"routes": [
{
"name": "OrdersMain",
"pattern": ":?query:",
"target": "OrdersMain"
},
{
"name": "Wizard",
"pattern": "Orders({key}):?query:",
"target": "Wizard"
}
],
"targets": {
"OrdersMain": {
"type": "Component",
"id": "Main",
"name": "sap.fe.core.fpm",
"options": {
"settings": {
"viewName": "flex.customformentry.ext.main.Main",
"entitySet": "Orders"
}
}
},
"Wizard": {
"type": "Component",
"id": "Wizard",
"name": "sap.fe.core.fpm",
"options": {
"settings": {
"viewName": "flex.customformentry.ext.wizard.Wizard",
"entitySet": "Orders"
}
}
}
}
}
2.4. 登録用のWizardを作成
内容は前回の記事で作成したものと基本的には同じです(一部うまく動かない部分もありますが、ここではカスタムページが開ければよしとします。)前回の記事と変えたのは、Create, Cancelなどのボタンが登録/キャンセル後は押せないようにViewモデルのプロパティでコントロールしている点です。
- ビュー
<mvc:View xmlns:core="sap.ui.core" xmlns:mvc="sap.ui.core.mvc" xmlns:macros="sap.fe.macros" xmlns:form="sap.ui.layout.form" id="review" controllerName="flex.customformentry.ext.wizard.Wizard" xmlns="sap.m">
<NavContainer id="wizardNavContainer">
<pages>
<Page id="createOrderPage" title="Guided Order Creation">
<content>
<Wizard id="CreateOrderWizard" class="sapUiResponsivePadding--header sapUiResponsivePadding--content" complete="reviewOrder">
<WizardStep id="stepCustomer" title="Customer" validated="true">
<VBox width="800px">
<Text text="Select the customer." />
<macros:Field metaPath="customer_ID" id="customer" />
</VBox>
</WizardStep>
<WizardStep id="stepItems" validated="true" title="Items">
<macros:Table metaPath="to_Items/@com.sap.vocabularies.UI.v1.LineItem" id="items" busy="{ui>/isBusy}" />
</WizardStep>
</Wizard>
</content>
<footer>
<OverflowToolbar>
<ToolbarSpacer />
<Button id="cancelButton" text="Cancel" press="cancelDocument"
visible="{viewModel>/editable}"/>
</OverflowToolbar>
</footer>
</Page>
<Page id="reviewPage" title="Review New Order">
<content>
<macros:Form metaPath="@com.sap.vocabularies.UI.v1.FieldGroup#main" id="reviewGeneral" />
<macros:Table enableExport="false" isSearchable="false" personalization="false"
metaPath="to_Items/@com.sap.vocabularies.UI.v1.LineItem" id="itemsDisplay" />
<MessageStrip class="sapUiSmallMarginBottom"
type="Warning"
text="By clicking on Create New Order you accept you read and accept our internal order guidelines."
showIcon="true"
visible="{viewModel>/editable}"/>
<Button text="Create New Order" press="createOrder" type="Emphasized"
visible="{viewModel>/editable}" />
</content>
</Page>
</pages>
</NavContainer>
</mvc:View>
- コントローラー
sap.ui.define(
[
'sap/fe/core/PageController',
'sap/ui/model/json/JSONModel',
'sap/m/MessageToast'
],
function(PageController, JSONModel, MessageToast) {
'use strict';
return PageController.extend('flex.customformentry.ext.wizard.Wizard', {
onInit: function() {
PageController.prototype.onInit.apply(this);
let model = {
editable : true
};
this.getView().setModel(new JSONModel(model), "viewModel");
},
reviewOrder: function () {
this.byId("wizardNavContainer").to(this.byId("reviewPage"));
},
createOrder: function () {
var that = this;
this.editFlow.saveDocument(this.getView().getBindingContext()).then(function(){
MessageToast.show("Order created!");
that.getView().getModel("viewModel").setProperty("/editable", false);
})
},
cancelDocument: function () {
var that = this;
this.editFlow.cancelDocument(this.getView().getBindingContext(), {
control: this.byId("cancelButton")
}).then(function(){
that.getView().getModel("viewModel").setProperty("/editable", false);
})
}
});
}
);
動作確認
事前設定
外部アプリへのナビゲーションの確認はLaunchpadからでないとできないため、以下の記事を参考にCAPローカルのLaunchpadの設定を行いました。
ポイントは、server.jsの設定でUI5のバージョンを指定しない(または最新のバージョンを指定する)ことです。Building Blocksは比較的新しいバージョンでないと動作しないためです。今回作成したアプリはバージョン1.109.0で動作確認しています。
const cds = require ('@sap/cds');
if (process.env.NODE_ENV !== 'production') {
const {cds_launchpad_plugin} = require('cds-launchpad-plugin');
// Enable launchpad plugin
cds.once('bootstrap',(app)=>{
const handler = new cds_launchpad_plugin();
app.use(handler.setup({theme:'sap_horizon'})); //versionは設定しない
});
}
動作確認
CAPのサービスを実行し、Launchpadのリンクをクリックします。
"Create"ボタンをクリックします。(Create wizardボタンはLineItemのDataFieldForIntentBasedNavigation
により表示させたボタンです。こちらもアプリ2へ遷移します)
登録後、一覧に戻って行をクリックします。
Object Pageに遷移します。
"Edit"をクリックしても(当然)Object Pageのままです。
よって、以下の動作が実現できました。
- List Reportで"Create"ボタンを押したらカスタムページを開く
- テーブル行をクリックしたらObject Pageを開く
感想
Flexible Programming Modelはドキュメントに具体的な使い方が書いていない場合もあり(例:createDocument)、試行錯誤での実装になりました。ナビゲーションやエンティティの登録、保存などは少ないコーディングで実装でき、「こういうケースではこのように実装する」というパターンを確立すれば効率的にアプリが作成できるのかもしれません。SAP公式やコミュニティで、使い方の紹介がもっと出てくることを期待します。