はじめに
SAPUI5関係での2023年の大きなニュースの一つは、TypeScriptが公式にサポートされるようになったことです(バージョン116.0から)。
それまでは「ベータ」のステータスで、型定義は使えるけれども完全な状態ではないという位置づけでした。TypeScriptを使用したプロジェクトのテンプレートはgenerator-easy-ui5では用意されていましたが、SAP Business Application Studioで利用可能なテンプレートはなく、プロジェクトで使用するのはためらわれる状態でした。
今後はこの状況が変わります。少なくともBTP用に開発するUI5アプリに関しては、TypeScriptが現実的な選択肢となってきます(S/4HANAで利用するUI5アプリは、バックエンドで利用可能なバージョンに依存)。
そこでこの記事では、2023年12月時点でのUI5&TypeScriptの開発をする手順について紹介します。
開発環境
IDE: SAP Business Application Studio
UI5バージョン:1.120.3
バックエンドサービス:OData V4(前回の記事で作成。※TypeScriptを使用する前提条件ではありません)
ステップ
- UI5プロジェクトを作成
- TypeScriptのソースを追加
- Managed Approuterの設定
- デプロイ
UI5アプリを作成する前のプロジェクトは以下の状態です。
ソースコード:https://github.com/miyasuta/cap-cds-typer
mta.yamlを右クリックし、"Create MTA Module from Template"を選択します。
"Approuter Configuration"をクリックします。
1. UI5プロジェクトを作成
Template Wizardから"SAP Fiori application"を選択します。
"List Report Page"のテンプレートを選択します。
データソースに"Use a Local CAP Project"を選択し、あらかじめ作成しておいたCAPのプロジェクト、サービスを指定します。
プロジェクトの属性を入力します。TypeScriptを使うために"Configure advanced options"を"Yes"にします。
Configure advanced optionsを有効にしたことで出てくるオプションのうち、最後の2つを有効化します。
- CAPプロジェクトにcds-plugin-ui5(※)を追加
- TypeScriptを有効化
※cds-plugin-ui5はCAPのサーバの中でUI5アプリを実行するためのプラグインです。CAPプロジェクトの中でTypeScriptのUI5アプリを開発するために必須となります。
生成されたプロジェクトにはTypeScriptの特徴があります。
- tsconfig.json:TypeScriptに関する設定
- .babelrc.json:TypeScriptのソースをJavaScriptにトランスパイル(変換)するための設定
- Component.ts:TypeScriptで作成されている
TypeScriptで必要となる型の定義は、node_modules/@sapui5/typesにインポートされています。
cds-ts watch
によりCAPアプリを起動すると、Web ApplicationsにUI5のプロジェクトのhtmlファイルが表示されます。一番上のリンクをクリックして開きます。動作はこちらの記事の通りになります。
2. TypeScriptのソースを追加
List Reportを拡張し、保存前にメッセージを出すようにしてみます。
Object Pageの"Show Controller Extensions"ボタンをクリックし、サイドパネルを表示させます。サイドパネルの"Add Controller Extension"ボタンをクリックします。
コントローラー拡張のファイルがTypeScriptで生成されます。
以下コードを追加します。削除ボタンが押されたら、確認のダイアログを表示するというものです。標準のダイアログが出た後に追加のダイアログが出ます。ソースコードはFlexible Programming Modelのサンプルソースを参考にしました。
import ControllerExtension from 'sap/ui/core/mvc/ControllerExtension';
import ExtensionAPI from 'sap/fe/templates/ObjectPage/ExtensionAPI';
import Dialog from 'sap/m/Dialog';
import Button from 'sap/m/Button';
import Text from 'sap/m/Text';
import { ButtonType, DialogType } from 'sap/m/library';
/**
* @namespace miyasuta.ordersui.ext.controller.ObjectPageExt
* @controller
*/
export default class ObjectPageExt extends ControllerExtension<ExtensionAPI> {
static overrides = {
/**
* Called when a controller is instantiated and its View controls (if available) are already created.
* Can be used to modify the View before it is displayed, to bind event handlers and do other one-time initialization.
* @memberOf miyasuta.ordersui.ext.controller.ObjectPageExt
*/
onInit(this: ObjectPageExt) {
// you can access the Fiori elements extensionAPI via this.base.getExtensionAPI
const model = this.base.getExtensionAPI().getModel();
},
editFlow: {
onBeforeDelete(mParameters: object) {
return createDialog("Do you wan to delete this object?")
}
}
}
}
async function createDialog(sText: string): Promise<void> {
return new Promise<void>((resolve, reject)=> {
const oApproveDialog = new Dialog({
type: DialogType.Message,
title: "Confirm",
content: new Text({text: sText}),
beginButton: new Button({
type: ButtonType.Emphasized,
text: "Continue",
press: () => {
oApproveDialog.close()
resolve()
}
}),
endButton: new Button({
text: "Cancel",
press: () => {
oApproveDialog.close()
reject()
}
}),
escapeHandler: (pCloseDialog) => {
pCloseDialog.resolve()
reject()
}
})
oApproveDialog.open()
})
}
TypeScriptのソースには、以下のような特徴があります。
-
sap.ui.define
の代わりにimport文により使用するクラスをインポートする - メソッド定義で引数、および返り値の型を指定する
詳しくは以下のブログをご参照ください。
開発時の気づき
- ソースの中でコントロールやクラスなどを使用すると、自動的にimport文が追加される
- 必ずしも正しいものがインポートされているとは限らないので、確認が必要
- Enum系(sap.m.DialogTypeやsap.m.ButtonType)のインポートは以下のように記述する
import { ButtonType, DialogType } from 'sap/m/library';
動作確認
"Delete"ボタンをクリックする
標準ダイアログが出る。"Delete"をクリックする
追加のダイアログが出る。"Continue"をクリックすると削除される
3. デプロイの設定
※以下にやや煩雑な設定がありますが、これはTypeScriptを使用しているためではなく、CAPとUI5を1つのMTAアプリケーションでデプロイしようとすることによるものです。UI5アプリを単体で開発する場合はテンプレートによって自動的に入ってくる設定なので、以下の手順は不要です。
3.1. Managed Approuterの設定
UIアプリをBTPコックピットのHTML Applications、またはSAP Build Work Zoneから実行できるようにするため、Managed Approuterを利用する設定が必要です。
まず、webapp/manifest.jsonに以下の設定を追加します。serviceの名前は任意ですが、ここではアプリIDと同じにしました。
"sap.cloud": {
"public": true,
"service": "miyasuta.ordersui"
}
続いてmta.yamlに以下のモジュールを追加します。(A)、(B)のサービス名はresourcesセクションに登録済みのものを利用します。
- name: capts_destination-content
type: com.sap.application.content
build-parameters:
no-source: true
requires:
- name: capts-auth # xsuaaサービス名 (A)
parameters:
service-key:
name: capts-auth-key # (A) + "-key"
- name: capts-repo-host # html5-apps-repoサービス名 (B)
parameters:
service-key:
name: capts-repo-host-key # (B) + "-key"
- name: capts-destination-service # destinationサービス名
parameters:
content-target: true
parameters:
content:
instance:
existing_destinations_policy: update
destinations:
- Name: capts-repo-host # (B)
ServiceInstanceName: capts-html5-srv # (B') html5-apps-repoサービス名(Bと異なる場合あり)
ServiceKeyName: capts-repo-host-key # (B) + "-key"
sap.cloud.service: miyasuta.ordersui # manifest.jsonのservice名
- Name: capts-auth # (A)
Authentication: OAuth2UserTokenExchange
ServiceInstanceName: capts-auth # (A)
ServiceKeyName: capts-auth-key # (A) + "-key"
sap.cloud.service: miyasuta.ordersui # manifest.jsonのservice名
3.2. ODataサービス呼び出しの設定
UI5アプリが使用するxs-app.jsonに、ODataサービスを呼ぶときのルートを追加します。このために、ODataサービスをDestinationとして登録する必要があります。Fiori toolsにより、これらの設定を自動で行うことができます。
"Add Deploy Config"のメニューを選択します。
Destination nameに"Local CAP Project API (Instance Based Destination)"を選択します。
これにより、UI5アプリプロジェクトのxs-app.jsonとmta.yamlに以下の設定が追加されます。
"routes": [
{
"source": "^/odata/(.*)$",
"target": "/odata/$1",
"destination": "capts-srv-api",
"authenticationType": "xsuaa",
"csrfProtection": false
},
- name: capts-destination-service
type: org.cloudfoundry.managed-service
parameters:
config:
HTML5Runtime_enabled: true
init_data:
instance:
destinations:
- Authentication: NoAuthentication
Name: ui5
ProxyType: Internet
Type: HTTP
URL: https://ui5.sap.com
# 追加されたDestination
- Authentication: NoAuthentication
HTML5.DynamicDestination: true
HTML5.ForwardAuthToken: true
Name: capts-srv-api
ProxyType: Internet
Type: HTTP
URL: ~{srv-api/srv-url}
existing_destinations_policy: update
4. デプロイ
以下のコマンドによりビルド、デプロイします。
mbt build --mtar archive
cf deploy mta_archives/archive.mtar
HTML5 Applicationsから利用可能になります。
TypeScriptで実装したロジック(JavaScriptに変換済み)が動いていることが確認できます。
おわりに
ここまでで見てきたように、UI5でのTypeScript開発はテンプレートによって必要な設定が自動で入ってくるので、開発者が特別な設定をする必要がありません。つまり、ほとんど労力をかけずにTypeScript開発を始めることができるのです。公式にTypeScriptがサポートされるようになった今が、TypeScriptの始め時だと私は思います。
UI5でTypeScriptを使うメリット
- 間違ったプロパティやメソッド名を指定するとエディタ上でエラーになるので、実行する前に気づくことができる → 生産性の向上
- ES6に対応した書き方ができる → UI5独特の構文を使わなくてよく、一般のWeb開発者が早く適応できる
UI5でTypeScriptを使うハードル
- UI5の学習リソースがほぼすべてJavaScriptベースなので、TypeScriptに切り替えるにあたり「差分」の学習が必要
- SDKのドキュメントやサンプルがJavaScriptベースなので、自分のソースで使う場合TypeScriptへの変換が必要
あえて「デメリット」ではなく「ハードル」と書きました。TypeScriptを使うデメリットは特になく、使用にあたって多少のハードルがあるととらえています。これは使うことで乗り越えていくしかないので、私もUI5(およびCAP)のコードを書く際はTypeScriptを積極的に使っていこうと思います。