import { immutable, excludedInJSON, action } from '../decorators'; import { config, history, utils, LEVEL_ENUM, Vertex, Logic, Variable, View, Schema, ActionOptions } from '..'; import { paramService } from '../../service/logic'; import { viewService } from '../../service/page'; import { convert2RefType, convert2SchemaType } from '../data/dataTypeUtils'; import { getBasicTypeDefaultValue } from '../data/basicTypes'; import { schemaService } from '../../service/common'; /** * 页面局部变量 */ export class ViewVariable extends Variable { /** * 视图 */ @excludedInJSON() @immutable() public readonly view: View = undefined; public readonly viewId: string = undefined; /** * 默认值 * 按 JSON string 处理 * - string: 666 -> '666' * - string: true -> 'true' * - number: 666 -> 666 * - boolean: true -> true */ public defaultValue: string = undefined; /** * @param source 需要合并的部分参数 */ constructor(source?: Partial) { super(); source && this.assign(source); } /** * 添加页面局部变量 */ @action('添加页面局部变量') async create(none?: void, actionOptions?: ActionOptions) { config.defaultApp?.emit('saving'); const body = this.toJSON('', ['ideVersion', 'editable']); body.viewId = this.view.id; // convert2SchemaType(body.schema); utils.logger.debug('添加页面局部变量', body); const result: ViewVariable = await viewService.createViewVariable({ headers: { appId: config.defaultApp?.id, operationAction: actionOptions?.actionName || 'ViewVariable.create', operationDesc: actionOptions?.actionDesc || `添加页面"${this.view.name}"局部变量"${this.name}"`, }, body, }); // convert2RefType(result.schema); this.deepPick(result, ['id']); this.assign({ code: this.genCode() }); if (this.view) { this.view.page.service.emit('pageTreeChange'); this.view.emit('change'); } await config.defaultApp?.history.load(); config.defaultApp?.emit('saved'); return this; } /** * 删除页面局部变量 */ @action('删除页面局部变量') async delete(none?: void, actionOptions?: ActionOptions) { config.defaultApp?.emit('saving'); if (this.id) { try { await viewService.deleteViewVariable({ headers: { appId: config.defaultApp?.id, operationAction: actionOptions?.actionName || 'ViewVariable.delete', operationDesc: actionOptions?.actionDesc || `删除页面"${this.view.name}"局部变量"${this.name}"`, }, query: { id: this.id, }, }); } catch (err) { await config.defaultApp?.history.load(); throw err; } } const { variables } = this.view.$def; const index = variables.indexOf(this); ~index && variables.splice(index, 1); this.destroy(); if (this.view) { this.view.page.service.emit('pageTreeChange'); this.view.emit('change'); } await config.defaultApp?.history.load(); config.defaultApp?.emit('saved'); } /** * 修改页面局部变量 */ async update(none?: void, actionOptions?: ActionOptions, then?: () => Promise) { config.defaultApp?.emit('saving'); const body = this.toJSON(); delete body.ideVersion; delete body.editable; body.viewId = this.view.id; // convert2SchemaType(body.schema); utils.logger.debug('修改页面局部变量', body); const result = await viewService.updateViewVariable({ headers: { appId: config.defaultApp?.id, operationAction: actionOptions?.actionName || 'ViewParam.update', operationDesc: actionOptions?.actionDesc || `修改页面"${this.view.name}"局部变量"${this.name}"`, }, body, }); // convert2RefType(result.schema); // this.plainAssign(result); await config.defaultApp?.history.load(); config.defaultApp?.emit('saved'); return this; } /** * 设置页面局部变量名称 * @param name 名称 */ @action('设置页面局部变量名称') async setName(name: string) { const oldName = this.name; this.assign({ name }); await this.update(undefined, { actionDesc: `设置页面"${this.view.name}"的局部变量"${oldName}"的名称为"${name}"`, }); if (this.view) { this.view.page.service.emit('pageTreeChange'); this.view.emit('change'); } } /** * 设置页面局部变量描述 * @param description 描述 */ @action('设置页面局部变量描述') async setDescription(description: string) { this.assign({ description }); await this.update(undefined, { actionDesc: `设置页面"${this.view.name}"的局部变量"${this.name}"的描述为"${description}"`, }); } /** * 查找schema 顶点被引用的逻辑顶点列表 */ async getSchemaUsage() { if (this.schema) { const schemaId = this.schema.type === 'genericType' ? this.id : this.schema.id; const result = await schemaService.getSchemaUsage({ query: { schemaId, valSource: 'view_var', }, }); return result; } } /** * 设置页面局部变量的数据类型 */ @action('设置页面局部变量的数据类型') async setDataType(schema: Schema) { // 用于update失败后还原数据 const originalData = { schema: Object.assign({}, this.schema), defaultValue: this.defaultValue, }; Object.assign(this.schema, { $ref: undefined, //引用到Entity或者structure type: undefined, format: undefined, typeInstantiation: undefined, }); Object.assign(this.schema, schema); // if (this.defaultValue) this.assign({ defaultValue: getBasicTypeDefaultValue() }); try { await this.update(undefined, { actionDesc: `设置页面"${this.view.name}"的局部变量"${this.name}"的数据类型为"${schema.typeKey}"`, }); this.genSchemaChildren(); if (this.view) { this.view.page.service.emit('pageTreeChange'); this.view.emit('change'); } } catch (err) { this.assign(originalData); config.defaultApp?.emit('saved'); throw err; } } /** * 设置页面局部变量是否为列表 * @param isArray */ @action('设置页面局部变量是否为列表') async setAsList(isArray: boolean) { this.schema.isArray = isArray; this.assign({ defaultValue: getBasicTypeDefaultValue() }); await this.update(undefined, { actionDesc: `设置页面"${this.view.name}"的局部变量"${this.name}"${isArray ? '为' : '不为'}列表`, }); this.genSchemaChildren(); if (this.view) { this.view.page.service.emit('pageTreeChange'); this.view.emit('change'); } } /** * 设置页面局部变量的默认值 */ @action('设置页面局部变量的默认值') async setDefaultValue(defaultValue: string) { this.assign({ defaultValue }); await this.update(undefined, { actionDesc: `设置页面局部变量的默认值`, }); if (this.view) { this.view.page.service.emit('pageTreeChange'); this.view.emit('change'); } } /** * 设置页面局部变量是否必须 * @param required 必须 */ @action('设置页面局部变量是否必须') async setRequired(required: boolean) { this.assign({ required }); await this.update(undefined, { actionDesc: `设置页面"${this.view.name}"的局部变量"${this.name}"为"${required ? '必须' : '不必须'}"`, }); if (this.view) { this.view.page.service.emit('pageTreeChange'); this.view.emit('change'); } } /** * 从后端 JSON 生成规范的 ViewVariable 对象 */ public static from(source: any, view: View | Logic) { convert2RefType(source.schema); source.view = view; source.code = 'ID_' + source.id; const viewVariable = new ViewVariable(source); viewVariable.genSchemaChildren(); return viewVariable; } } export default ViewVariable;