Compare commits

...

2 Commits

  1. 4
      win_text_editor/.vscode/settings.json
  2. 101
      win_text_editor/assets/config/uft_macro_list.yaml
  3. 17
      win_text_editor/lib/framework/services/macro_template_service.dart
  4. 37
      win_text_editor/lib/modules/call_function/controllers/call_function_controller.dart
  5. 35
      win_text_editor/lib/modules/call_function/models/call_function.dart
  6. 18
      win_text_editor/lib/modules/call_function/services/call_function_service.dart
  7. 44
      win_text_editor/lib/modules/code_creater/controllers/code_creater_controller.dart
  8. 290
      win_text_editor/lib/modules/code_creater/services/code_create_service.dart
  9. 14
      win_text_editor/lib/modules/code_creater/widgets/code_creater_view.dart
  10. 52
      win_text_editor/lib/modules/code_creater/widgets/node_table.dart
  11. 5
      win_text_editor/lib/modules/memory_table/controllers/memory_table_controller.dart
  12. 24
      win_text_editor/lib/modules/memory_table/models/memory_table.dart
  13. 10
      win_text_editor/lib/modules/memory_table/services/memory_table_service.dart
  14. 5
      win_text_editor/lib/modules/outline/models/outline_node.dart
  15. 8
      win_text_editor/lib/modules/outline/services/component_service.dart
  16. 22
      win_text_editor/lib/modules/outline/services/functions_service.dart
  17. 11
      win_text_editor/lib/modules/outline/services/uft_object_service.dart
  18. 9
      win_text_editor/lib/modules/outline/widgets/outline_explorer.dart
  19. 3
      win_text_editor/lib/modules/uft_component/services/uft_component_service.dart

4
win_text_editor/.vscode/settings.json vendored

@ -1 +1,3 @@
{} {
"cmake.sourceDirectory": "D:/aigc/manta/win_text_editor/linux"
}

101
win_text_editor/assets/config/uft_macro_list.yaml

@ -18,6 +18,15 @@ templates:
}else{ }else{
} }
获取记录_ALL:
body: |
[获取记录][{{tableName}}({{keyName}})][
{{#keyFields}}
{{name}} = @{{name}}{{^isLast}}, {{/isLast}}
{{/keyFields}}
][]
获取记录数: 获取记录数:
@ -36,7 +45,7 @@ templates:
[获取记录][{{tableName}}({{keyName}})][ [获取记录][{{tableName}}({{keyName}})][
{{#keyFields}} {{#keyFields}}
{{name}} = @{{name}} {{^isLast}}, {{/isLast}} {{name}} = @{{#partnerName}}{{partnerName}}.{{/partnerName}}{{name}} {{^isLast}}, {{/isLast}}
{{/keyFields}} {{/keyFields}}
] ]
@ -48,7 +57,7 @@ templates:
<F>[修改索引字段][{{tableName}}][ <F>[修改索引字段][{{tableName}}][
{{#keyFields}} {{#keyFields}}
{{name}} = @{{name}}{{^isLast}}, {{/isLast}} {{name}} = @{{#partnerName}}{{partnerName}}.{{/partnerName}}{{name}}{{^isLast}}, {{/isLast}}
{{/keyFields}} {{/keyFields}}
] ]
} }
@ -104,6 +113,23 @@ templates:
//balabala //balabala
} }
[遍历记录结束] [遍历记录结束]
遍历记录_ALL:
body: |
[遍历记录开始][{{tableName}}({{selectIndexOrKey.name}})][
{{#selectIndexOrKey.fields}}
{{name}} = @{{name}}{{^isLast}}, {{/isLast}}
{{/selectIndexOrKey.fields}}
][
{{#selectedFields}}
{{name}} = @{{name}}{{^isLast}}, {{/isLast}}
{{/selectedFields}}
]
遍历记录结束_ALL:
body: |
[遍历记录结束]
插入组件: 插入组件:
body: | body: |
@ -136,6 +162,17 @@ templates:
{{/selectedFields}} {{/selectedFields}}
] ]
获取组件_ALL:
body: |
{{#hasIndex}}[组件排序][{{name}}(index.name)]{{/hasIndex}}
[获取组件][{{name}}{{#hasIndex}}({{index.name}}){{/hasIndex}}][{{^hasIndex}}[@num][{{/hasIndex}}
{{#hasIndex}}
{{#index.fields}}
{{name}} = @{{name}}{{^isLast}}, {{/isLast}}
{{/index.fields}}
][
{{/hasIndex}}]
遍历组件: 遍历组件:
body: | body: |
{{#hasIndex}}[组件排序][{{name}}({{index.name}})]{{/hasIndex}} {{#hasIndex}}[组件排序][{{name}}({{index.name}})]{{/hasIndex}}
@ -153,7 +190,7 @@ templates:
{ {
} }
"[遍历组件结束]" [遍历组件结束]
组件大小: 组件大小:
body: "[组件大小][{{name}}][@num]" body: "[组件大小][{{name}}][@num]"
@ -166,7 +203,7 @@ templates:
{{/fields}} {{/fields}}
] ]
遍历组件所有: 遍历组件_ALL:
body: | body: |
{{#hasIndex}}[组件排序][{{name}}({{index.name}})]{{/hasIndex}} {{#hasIndex}}[组件排序][{{name}}({{index.name}})]{{/hasIndex}}
[遍历组件开始][{{name}}{{#hasIndex}}({{index.name}}){{/hasIndex}}][ [遍历组件开始][{{name}}{{#hasIndex}}({{index.name}}){{/hasIndex}}][
@ -175,19 +212,16 @@ templates:
{{name}} = @{{name}} {{^isLast}}, {{/isLast}} {{name}} = @{{name}} {{^isLast}}, {{/isLast}}
{{/index.fields}} {{/index.fields}}
{{/hasIndex}} {{/hasIndex}}
][
{{#fields}}
{{name}} = @{{name}}{{^isLast}}, {{/isLast}}
{{/fields}}
] ]
{
}
"[遍历组件结束]" 遍历组件结束_ALL:
body: |
[遍历组件结束]
普通调用: 普通调用_ALL:
body: | body: |
<Z>[{{chineseName}}][ [{{chineseName}}][
{{#input}} {{#input}}
{{name}} = @{{#partnerName}}{{partnerName}}.{{/partnerName}}{{name}}{{^isLast}},{{/isLast}} {{name}} = @{{#partnerName}}{{partnerName}}.{{/partnerName}}{{name}}{{^isLast}},{{/isLast}}
{{/input}} {{/input}}
@ -196,6 +230,19 @@ templates:
{{name}} = @{{name}}{{^isLast}},{{/isLast}} {{name}} = @{{name}}{{^isLast}},{{/isLast}}
{{/output}} {{/output}}
] ]
普通调用:
body: |
<Z>[{{chineseName}}][
{{#selectedInput}}
{{name}} = @{{#partnerName}}{{partnerName}}.{{/partnerName}}{{name}}{{^isLast}},{{/isLast}}
{{/selectedInput}}
][
{{#selectedOutput}}
{{name}} = @{{name}}{{^isLast}},{{/isLast}}
{{/selectedOutput}}
]
[处理失败] [处理失败]
{ {
[错误信息获取] [错误信息获取]
@ -206,9 +253,33 @@ templates:
else else
{ {
//balabala //balabala
} }
事务调用: 事务调用:
body: |
[事务处理开始]
<Z>[{{chineseName}}][
{{#selectedInput}}
{{name}} = @{{#partnerName}}{{partnerName}}.{{/partnerName}}{{name}}{{^isLast}},{{/isLast}}
{{/selectedInput}}
][
{{#selectedOutput}}
{{name}} = @{{name}}{{^isLast}},{{/isLast}}
{{/selectedOutput}}
]
[处理失败]
{
[错误信息获取]
//TODO:处理异常
[事务回滚]
[继续执行]
}
else
{
[事务处理结束]
}
事务调用_ALL:
body: | body: |
[事务处理开始] [事务处理开始]
<Z>[{{chineseName}}][ <Z>[{{chineseName}}][
@ -230,7 +301,7 @@ templates:
else else
{ {
[事务处理结束] [事务处理结束]
} }
因子服务: 因子服务:
body: | body: |

17
win_text_editor/lib/framework/services/macro_template_service.dart

@ -1,4 +1,5 @@
import 'package:flutter/services.dart' show rootBundle; import 'package:flutter/services.dart' show rootBundle;
import 'package:win_text_editor/framework/controllers/logger.dart';
import 'package:yaml/yaml.dart'; import 'package:yaml/yaml.dart';
import 'package:mustache_template/mustache_template.dart' as mustache; import 'package:mustache_template/mustache_template.dart' as mustache;
@ -41,11 +42,21 @@ class MacroTemplateService {
: null; : null;
} }
String renderTemplate(List<String> templateList, Map<String, dynamic> context) { String renderTemplate(
List<String> templateList,
Map<String, dynamic> context, {
bool selectAll = false,
}) {
StringBuffer buffer = StringBuffer(); StringBuffer buffer = StringBuffer();
for (var templateName in templateList) { for (var templateName in templateList) {
final template = getTemplate(templateName); Map<String, dynamic>? template = getTemplate('$templateName${selectAll ? '_ALL' : ''}');
if (template == null) throw Exception('Template $templateName not found'); if (template == null) {
Logger().warning("没有找到模板$templateName${selectAll ? '_ALL' : ''}");
template = getTemplate(templateName);
if (template == null) {
throw Exception('Template $templateName not found');
}
}
buffer.writeln(_render(template['body'], context)); buffer.writeln(_render(template['body'], context));

37
win_text_editor/lib/modules/call_function/controllers/call_function_controller.dart

@ -20,10 +20,9 @@ class CallFunctionController extends BaseContentController {
late DataGridSource inputSource; late DataGridSource inputSource;
late DataGridSource outputSource; late DataGridSource outputSource;
final CallFunctionService _service;
final MacroTemplateService templateService = MacroTemplateService(); final MacroTemplateService templateService = MacroTemplateService();
CallFunctionController() : _service = CallFunctionService(Logger()) { CallFunctionController() {
inputSource = FieldsDataSource( inputSource = FieldsDataSource(
[], [],
onSelectionChanged: (index, isSelected) { onSelectionChanged: (index, isSelected) {
@ -56,24 +55,24 @@ class CallFunctionController extends BaseContentController {
} }
// //
for (final input in modle.inputParameters) { // for (final input in modle.inputParameters) {
if (input.isSelected && input.type == CallFunction.componentType) { // if (input.isSelected && input.type == CallFunction.componentType) {
final component = modle.componentList?.firstWhereOrNull((c) => c.name == input.name); // final component = modle.componentList?.firstWhereOrNull((c) => c.name == input.name);
if (component != null) { // if (component != null) {
sb.write(templateService.renderTemplate(['插入组件'], component.toMap())); // sb.write(templateService.renderTemplate(['插入组件'], component.toMap()));
} // }
} // }
} // }
sb.write(templateService.renderTemplate(macroList, modle.toMap())); sb.write(templateService.renderTemplate(macroList, modle.toMap()));
// //
for (final output in modle.outputParameters) { // for (final output in modle.outputParameters) {
if (output.isSelected && output.type == CallFunction.componentType) { // if (output.isSelected && output.type == CallFunction.componentType) {
final component = modle.componentList?.firstWhereOrNull((c) => c.name == output.name); // final component = modle.componentList?.firstWhereOrNull((c) => c.name == output.name);
if (component != null) { // if (component != null) {
sb.write(templateService.renderTemplate(['遍历组件所有'], component.toMap())); // sb.write(templateService.renderTemplate(['遍历组件所有'], component.toMap()));
} // }
} // }
} // }
return sb.toString(); return sb.toString();
} }
@ -81,7 +80,7 @@ class CallFunctionController extends BaseContentController {
Future<void> onOpenFile(String filePath, {dynamic appendArg}) async { Future<void> onOpenFile(String filePath, {dynamic appendArg}) async {
Logger().info("Opening file: $filePath"); Logger().info("Opening file: $filePath");
try { try {
modle = await _service.parseXmlFile(filePath); modle = await CallFunctionService.parseXmlFile(filePath);
// datagrid显示的数据源 // datagrid显示的数据源
(inputSource as FieldsDataSource).updateData(modle.inputParameters); (inputSource as FieldsDataSource).updateData(modle.inputParameters);

35
win_text_editor/lib/modules/call_function/models/call_function.dart

@ -10,6 +10,7 @@ class CallFunction {
final List<Field> inputParameters; final List<Field> inputParameters;
final List<Field> outputParameters; final List<Field> outputParameters;
List<UftComponent>? componentList; List<UftComponent>? componentList;
List<UftComponent>? outComponentList;
String? factorParam; String? factorParam;
final List<CodePartner> codePartners = []; final List<CodePartner> codePartners = [];
@ -20,6 +21,7 @@ class CallFunction {
required this.inputParameters, required this.inputParameters,
required this.outputParameters, required this.outputParameters,
this.componentList, this.componentList,
this.outComponentList,
this.factorParam, this.factorParam,
}); });
@ -29,6 +31,25 @@ class CallFunction {
'chineseName': chineseName, 'chineseName': chineseName,
'factorParam': factorParam ?? '', 'factorParam': factorParam ?? '',
'input': 'input':
inputParameters
.map(
(field) => {
'id': field.id,
'name': field.name,
'chineseName': field.chineseName,
'type': field.type,
'isLast': inputParameters.indexOf(field) == inputParameters.length - 1,
'partnerName':
codePartners
.firstWhere(
(partner) => partner.fields.any((f) => f == field.name),
orElse: () => CodePartner(name: null, fields: []),
)
.name,
},
)
.toList(),
'selectedInput':
inputParameters inputParameters
.where((field) => field.isSelected) .where((field) => field.isSelected)
.map( .map(
@ -48,7 +69,7 @@ class CallFunction {
}, },
) )
.toList(), .toList(),
'output': 'selectedOutput':
outputParameters outputParameters
.where((field) => field.isSelected) .where((field) => field.isSelected)
.map( .map(
@ -61,6 +82,18 @@ class CallFunction {
}, },
) )
.toList(), .toList(),
'output':
outputParameters
.map(
(field) => {
'id': field.id,
'name': field.name,
'chineseName': field.chineseName,
'type': field.type,
'isLast': outputParameters.indexOf(field) == outputParameters.length - 1,
},
)
.toList(),
'inputComps': 'inputComps':
componentList componentList
?.map( ?.map(

18
win_text_editor/lib/modules/call_function/services/call_function_service.dart

@ -11,10 +11,6 @@ import 'package:xml/xml.dart' as xml;
import 'package:win_text_editor/framework/controllers/logger.dart'; import 'package:win_text_editor/framework/controllers/logger.dart';
class CallFunctionService { class CallFunctionService {
final Logger _logger;
CallFunctionService(this._logger);
static const Map<String, String> _functionRootTypeMap = { static const Map<String, String> _functionRootTypeMap = {
'uftfunction': 'business:Function', 'uftfunction': 'business:Function',
'uftservice': 'business:Service', 'uftservice': 'business:Service',
@ -24,12 +20,13 @@ class CallFunctionService {
'uftfactorservice': 'business:FactorService', 'uftfactorservice': 'business:FactorService',
}; };
Future<CallFunction> parseXmlFile(String filePath) async { static Future<CallFunction> parseXmlFile(String filePath) async {
try { try {
// 1. Check file extension // 1. Check file extension
final extendFileName = filePath.toLowerCase().split('.').last; final extendFileName = filePath.toLowerCase().split('.').last;
final String? rootNodeType = _functionRootTypeMap[extendFileName]; final String? rootNodeType = _functionRootTypeMap[extendFileName];
if (rootNodeType == null) { if (rootNodeType == null) {
Logger().error('文件扩展名不正确:$filePath');
throw const FormatException("文件扩展名不正确!"); throw const FormatException("文件扩展名不正确!");
} }
@ -53,6 +50,8 @@ class CallFunctionService {
List<UftComponent> componentList = []; List<UftComponent> componentList = [];
List<UftComponent> outComponentList = [];
// 5. Process inputParameters (fields) // 5. Process inputParameters (fields)
final inputParameters = document.findAllElements( final inputParameters = document.findAllElements(
extendFileName == "uftfactorservice" ? 'internalParams' : 'inputParameters', extendFileName == "uftfactorservice" ? 'internalParams' : 'inputParameters',
@ -68,7 +67,7 @@ class CallFunctionService {
// 6. Process outputParameters // 6. Process outputParameters
final outputParameters = document.findAllElements('outputParameters'); final outputParameters = document.findAllElements('outputParameters');
final outputFields = await parserFields(filePath, outputParameters, componentList); final outputFields = await parserFields(filePath, outputParameters, outComponentList);
return CallFunction( return CallFunction(
functionType: extendFileName, functionType: extendFileName,
@ -77,18 +76,19 @@ class CallFunctionService {
inputParameters: inputFields, inputParameters: inputFields,
outputParameters: outputFields, outputParameters: outputFields,
componentList: componentList, componentList: componentList,
outComponentList: outComponentList,
factorParam: factorParam, factorParam: factorParam,
); );
} on xml.XmlParserException catch (e) { } on xml.XmlParserException catch (e) {
_logger.error("XML解析错误: ${e.message}"); Logger().error("XML解析错误: ${e.message}");
rethrow; rethrow;
} catch (e) { } catch (e) {
_logger.error("处理文件时发生错误: $e"); Logger().error("处理文件时发生错误: $e");
rethrow; rethrow;
} }
} }
Future<List<Field>> parserFields( static Future<List<Field>> parserFields(
String filePath, String filePath,
Iterable<xml.XmlElement> parameters, Iterable<xml.XmlElement> parameters,
List<UftComponent> componentList, List<UftComponent> componentList,

44
win_text_editor/lib/modules/code_creater/controllers/code_creater_controller.dart

@ -1,5 +1,6 @@
import 'package:win_text_editor/framework/common/constants.dart'; import 'package:win_text_editor/framework/common/constants.dart';
import 'package:win_text_editor/framework/controllers/logger.dart'; import 'package:win_text_editor/framework/controllers/logger.dart';
import 'package:win_text_editor/modules/code_creater/services/code_create_service.dart';
import 'package:win_text_editor/modules/outline/models/outline_node.dart'; import 'package:win_text_editor/modules/outline/models/outline_node.dart';
import 'package:win_text_editor/shared/base/base_content_controller.dart'; import 'package:win_text_editor/shared/base/base_content_controller.dart';
@ -22,7 +23,6 @@ class CodeCreaterController extends BaseContentController {
_members.where((node) => !(unMatchMap[node.value]?.contains(value) ?? false)).toList(); _members.where((node) => !(unMatchMap[node.value]?.contains(value) ?? false)).toList();
_members = filtered; // _members = filtered; //
_selectedOperation = value; _selectedOperation = value;
notifyListeners(); //
} }
void updateMembers(List<OutlineNode> newMembers) { void updateMembers(List<OutlineNode> newMembers) {
@ -43,9 +43,49 @@ class CodeCreaterController extends BaseContentController {
return; return;
} }
if (members.contains(node)) {
Logger().error('不支持重复节点: ${node.title}');
return;
}
//
node.selectedAction = CodeCreateService.getNodeActions(node.name)[0];
members.add(node); members.add(node);
notifyListeners(); notifyListeners();
} }
genCodeString(List<String> list) {} Future<String> genCodeString() {
return CodeCreateService.genCodeString(members, selectedOperation);
}
void handleActionTypeChange(int index, String newAction) {
final memberName = members[index].name;
if (newAction == "遍历记录") {
members.add(
OutlineNode(
name: '${members[index].name}_END',
title: members[index].title,
value: '${members[index].value}_END',
selectedAction: "遍历记录结束",
),
);
} else if (newAction == "遍历组件") {
members.add(
OutlineNode(
name: '${members[index].name}_END',
title: members[index].title,
value: '${members[index].value}_END',
selectedAction: "遍历组件结束",
),
);
} else {
members.removeWhere((member) => member.name == '${memberName}_END');
}
// members[index].selectedAction = newAction;
notifyListeners();
}
} }

290
win_text_editor/lib/modules/code_creater/services/code_create_service.dart

@ -0,0 +1,290 @@
import 'package:flutter/material.dart';
import 'package:win_text_editor/framework/common/constants.dart';
import 'package:win_text_editor/framework/services/macro_template_service.dart';
import 'package:win_text_editor/modules/call_function/models/call_function.dart';
import 'package:win_text_editor/modules/call_function/services/call_function_service.dart';
import 'package:win_text_editor/modules/memory_table/models/memory_table.dart';
import 'package:win_text_editor/modules/memory_table/services/memory_table_service.dart';
import 'package:win_text_editor/modules/outline/models/code_partner.dart';
import 'package:win_text_editor/modules/outline/models/outline_node.dart';
import 'package:win_text_editor/modules/uft_component/models/uft_component.dart';
import 'package:win_text_editor/modules/uft_component/services/uft_component_service.dart';
class CodeCreateService {
static List<String> getNodeActions(String type) {
switch (type) {
case Constants.atomFunction:
case Constants.atomService:
case Constants.business:
return ['普通调用', '事务调用'];
case Constants.uftTable:
return ['获取记录', '遍历记录', '插入记录', '修改记录', '删除记录'];
case Constants.component:
return ['获取组件', '遍历组件', '插入组件', '修改组件', '尾部插入组件'];
case "${Constants.component}_END":
return ['遍历组件结束'];
case "${Constants.uftTable}_END":
return ['遍历记录结束'];
default:
return ['']; //
}
}
static final MacroTemplateService templateService = MacroTemplateService();
static void initTemplateService() {
if (!templateService.inited) {
templateService.init();
}
}
static Future<String> genCodeString(List<OutlineNode> members, String selectedOperation) async {
initTemplateService();
//members,value
StringBuffer codeBuffer = StringBuffer("//生成$selectedOperation代码\n\n");
List<String> inputFields = []; //
List<String> outputFields = []; //
List<String> macroCodeList = []; //
List<CodePartner> beforePartner = []; //
Map<String, List<String>> iterateOutputFields = {}; //-使
Map<String, List<CodePartner>> iteratePartnerMap = {}; //
List<String> iterateStack = []; //
//
Future<void> parseRelationShips(List<OutlineNode> nodes, String selectedOperation) async {
for (int i = 0; i < members.length; i++) {
final member = members[i];
final action =
member.selectedAction!.isEmpty
? getNodeActions(member.value)[0]
: member.selectedAction;
//
if (action!.startsWith("遍历") && action.endsWith("结束")) {
//
final memberName = iterateStack.removeLast();
debugPrint('删除遍历块:$memberName');
iterateOutputFields.remove(memberName);
iteratePartnerMap.remove(memberName);
final macroCode = templateService.renderTemplate([action], {}, selectAll: true);
macroCodeList.add(macroCode);
continue;
}
switch (member.value) {
case Constants.atomFunction:
case Constants.atomService:
case Constants.business:
CallFunction model = await CallFunctionService.parseXmlFile(member.path!);
model.codePartners.clear(); //
//
inputFields.addAll(
model.inputParameters
.where(
(field) =>
!outputFields.contains(field.name) &&
!inputFields.contains(field.name) &&
!iterateOutputFields.values.expand((list) => list).contains(field.name),
)
.map((field) => field.name),
);
//
model.codePartners.addAll(beforePartner.toSet());
//
model.codePartners.addAll(
iteratePartnerMap.values.expand((partners) => partners).toList(),
);
//
if (iterateStack.isNotEmpty) {
//
iterateOutputFields
.putIfAbsent(iterateStack.last, () => [])
.addAll(model.outputParameters.map((field) => field.name).toSet());
//:
for (UftComponent component in model.outComponentList!) {
List<String> fieldNames = [];
for (var field in component.fields) {
fieldNames.add(field.name);
}
iterateOutputFields
.putIfAbsent(iterateStack.last, () => [])
.addAll(component.fields.map((field) => field.name).toSet());
iteratePartnerMap
.putIfAbsent(component.name, () => [])
.add(CodePartner(name: component.name, fields: fieldNames));
}
} else {
//
outputFields.addAll(
model.outputParameters
.where((field) => !outputFields.contains(field.name))
.map((field) => field.name),
);
//
for (UftComponent component in model.outComponentList!) {
List<String> fieldNames = component.fields.map((col) => col.name).toList();
outputFields.addAll(fieldNames.toSet());
beforePartner.add(CodePartner(name: member.name, fields: fieldNames));
}
}
final macroCode = templateService.renderTemplate(
[action],
model.toMap(),
selectAll: true,
);
macroCodeList.add(macroCode);
break;
case Constants.uftTable:
TableData tableData = await MemoryTableService.parseStructureFile(member.path!);
final memoryTable = MemoryTable(
tableName: tableData.tableName,
columns: tableData.fields,
indexes: tableData.indexes,
);
memoryTable.codePartners.clear();
//
memoryTable.codePartners.addAll(beforePartner.toSet());
//
memoryTable.codePartners.addAll(
iteratePartnerMap.values.expand((partners) => partners).toList(),
);
List<String> fieldNames = memoryTable.columns.map((col) => col.name).toList();
switch (action) {
case '获取记录':
for (var fieldName in memoryTable.keyFields) {
if (!outputFields.contains(fieldName) &&
!inputFields.contains(fieldName) &&
!iterateOutputFields.values.expand((list) => list).contains(fieldName)) {
inputFields.add(fieldName);
}
}
if (iterateStack.isNotEmpty) {
//
iterateOutputFields.putIfAbsent(iterateStack.last, () => []).addAll(fieldNames);
iteratePartnerMap
.putIfAbsent(member.name, () => [])
.add(CodePartner(name: member.name, fields: fieldNames));
} else {
//
outputFields.addAll(fieldNames);
beforePartner.add(CodePartner(name: member.name, fields: fieldNames));
}
break;
case '遍历记录':
List<String> fieldNames = memoryTable.columns.map((col) => col.name).toList();
iterateStack.add(memoryTable.tableName); //
//
iterateOutputFields.putIfAbsent(iterateStack.last, () => []).addAll(fieldNames);
iteratePartnerMap
.putIfAbsent(member.name, () => [])
.add(CodePartner(name: member.name, fields: fieldNames));
break;
case '插入记录':
case '修改记录':
case '删除记录':
for (var fieldName in memoryTable.keyFields) {
if (!outputFields.contains(fieldName) &&
!inputFields.contains(fieldName) &&
!iterateOutputFields.values.expand((list) => list).contains(fieldName)) {
inputFields.add(fieldName);
}
}
break;
}
final macroCode = templateService.renderTemplate(
[action!],
memoryTable.toMap(),
selectAll: true,
);
macroCodeList.add(macroCode);
break;
case Constants.component:
final component = await UftComponentService.getUftComponent(member.path!, member.name);
component!.codePartners.clear();
component.codePartners.addAll(beforePartner);
component.codePartners.addAll(
iteratePartnerMap.values.expand((partners) => partners).toList(),
);
switch (action) {
case "获取组件":
case "遍历组件":
List<String> fieldNames = component.fields.map((col) => col.name).toList();
//
if (iterateStack.isNotEmpty) {
//
iterateOutputFields.putIfAbsent(iterateStack.last, () => []).addAll(fieldNames);
iteratePartnerMap
.putIfAbsent(member.name, () => [])
.add(CodePartner(name: member.name, fields: fieldNames));
} else {
//
outputFields.addAll(fieldNames);
beforePartner.add(CodePartner(name: member.name, fields: fieldNames));
}
iterateStack.add(component.name); //
break;
default:
for (var field in component.fields) {
if (!outputFields.contains(field.name) &&
!inputFields.contains(field.name) &&
!iterateOutputFields.values.expand((list) => list).contains(field.name)) {
inputFields.add(field.name);
}
}
break;
}
final macroCode = templateService.renderTemplate(
[action],
component.toMap(),
selectAll: true,
);
macroCodeList.add(macroCode);
break;
default:
macroCodeList.add('//未知类型: ${member.name}');
}
}
}
await parseRelationShips(members, selectedOperation);
codeBuffer.writeln("//可能需要传入的参数");
for (var field in inputFields) {
codeBuffer.writeln('@$field = ,');
}
codeBuffer.writeln('');
for (final mc in macroCodeList) {
codeBuffer.write(mc);
}
return codeBuffer.toString();
}
}

14
win_text_editor/lib/modules/code_creater/widgets/code_creater_view.dart

@ -39,6 +39,7 @@ class _CodeCreaterViewState extends State<CodeCreaterView> {
} }
_controller.addListener(_handleControllerUpdate); _controller.addListener(_handleControllerUpdate);
_controller.addListener(_updateDisplay);
} }
void _handleControllerUpdate() { void _handleControllerUpdate() {
@ -48,6 +49,7 @@ class _CodeCreaterViewState extends State<CodeCreaterView> {
@override @override
void dispose() { void dispose() {
_controller.removeListener(_handleControllerUpdate); _controller.removeListener(_handleControllerUpdate);
_controller.removeListener(_updateDisplay);
if (!_isControllerFromTabManager) { if (!_isControllerFromTabManager) {
_controller.dispose(); _controller.dispose();
} }
@ -71,6 +73,8 @@ class _CodeCreaterViewState extends State<CodeCreaterView> {
onDeleteMember: _deleteMember, onDeleteMember: _deleteMember,
onMoveToTop: _moveToTop, onMoveToTop: _moveToTop,
onMoveToBottom: _moveToBottom, onMoveToBottom: _moveToBottom,
onActionTypeChanged:
(index, newAction) => _controller.handleActionTypeChange(index, newAction),
), ),
), ),
), ),
@ -117,6 +121,7 @@ class _CodeCreaterViewState extends State<CodeCreaterView> {
setState(() { setState(() {
_controller.members.insert(to, member); // _controller.members.insert(to, member); //
_updateDisplay();
}); });
} }
@ -130,7 +135,12 @@ class _CodeCreaterViewState extends State<CodeCreaterView> {
void _deleteMember(int index) { void _deleteMember(int index) {
setState(() { setState(() {
final memberName = _controller.members[index].name;
//
_controller.members.removeWhere((member) => member.name == '${memberName}_END');
_controller.members.removeAt(index); _controller.members.removeAt(index);
_updateDisplay();
}); });
} }
@ -144,7 +154,7 @@ class _CodeCreaterViewState extends State<CodeCreaterView> {
}); });
} }
void _updateDisplay() { Future<void> _updateDisplay() async {
_codeController.text = _controller.genCodeString([_controller.selectedOperation!]) ?? ''; _codeController.text = await _controller.genCodeString();
} }
} }

52
win_text_editor/lib/modules/code_creater/widgets/node_table.dart

@ -3,6 +3,7 @@ import 'dart:ui';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:pluto_grid/pluto_grid.dart'; import 'package:pluto_grid/pluto_grid.dart';
import 'package:win_text_editor/framework/common/constants.dart'; import 'package:win_text_editor/framework/common/constants.dart';
import 'package:win_text_editor/modules/code_creater/services/code_create_service.dart';
import 'package:win_text_editor/modules/outline/models/outline_node.dart'; import 'package:win_text_editor/modules/outline/models/outline_node.dart';
import 'package:win_text_editor/shared/components/my_pluto_column.dart'; import 'package:win_text_editor/shared/components/my_pluto_column.dart';
import 'package:win_text_editor/shared/components/my_pluto_configuration.dart'; import 'package:win_text_editor/shared/components/my_pluto_configuration.dart';
@ -14,6 +15,7 @@ class NodeTable extends StatefulWidget {
final Function(int) onDeleteMember; final Function(int) onDeleteMember;
final Function(int) onMoveToTop; final Function(int) onMoveToTop;
final Function(int) onMoveToBottom; final Function(int) onMoveToBottom;
final Function(int, String)? onActionTypeChanged; //
const NodeTable({ const NodeTable({
super.key, super.key,
@ -22,6 +24,7 @@ class NodeTable extends StatefulWidget {
required this.onDeleteMember, required this.onDeleteMember,
required this.onMoveToTop, required this.onMoveToTop,
required this.onMoveToBottom, required this.onMoveToBottom,
this.onActionTypeChanged, //
}); });
@override @override
@ -79,6 +82,9 @@ class NodeTableState extends State<NodeTable> {
_stateManager?.setSelectingMode(PlutoGridSelectingMode.row); _stateManager?.setSelectingMode(PlutoGridSelectingMode.row);
_stateManager?.addListener(_handleRowSelection); _stateManager?.addListener(_handleRowSelection);
}, },
onChanged: (PlutoGridOnChangedEvent event) {
_handleCellChanged(event);
},
onRowDoubleTap: (event) { onRowDoubleTap: (event) {
_selectedRowIndex = event.rowIdx; _selectedRowIndex = event.rowIdx;
}, },
@ -166,6 +172,23 @@ class NodeTableState extends State<NodeTable> {
}); });
} }
void _handleCellChanged(PlutoGridOnChangedEvent event) {
if (event.column?.field == 'type' && event.rowIdx != null) {
final rowIndex = event.rowIdx!;
if (rowIndex >= 0 && rowIndex < widget.members.length) {
final newValue = event.value?.toString() ?? '';
setState(() {
widget.members[rowIndex].selectedAction = newValue;
});
// controller
if (widget.onActionTypeChanged != null) {
widget.onActionTypeChanged!(rowIndex, newValue);
}
}
}
}
final Map<Key, _NodeTypeData> _nodeTypeData = {}; final Map<Key, _NodeTypeData> _nodeTypeData = {};
List<PlutoColumn> _buildColumns() { List<PlutoColumn> _buildColumns() {
@ -179,7 +202,7 @@ class NodeTableState extends State<NodeTable> {
MyPlutoColumn( MyPlutoColumn(
title: '成员', title: '成员',
field: 'content', field: 'content',
width: 300, width: 200,
enableRowChecked: false, // enableRowChecked: false, //
), ),
MyPlutoDropdownColumn( MyPlutoDropdownColumn(
@ -201,7 +224,13 @@ class NodeTableState extends State<NodeTable> {
.map((index, member) { .map((index, member) {
final rowKey = ValueKey('${member.hashCode}_$index'); final rowKey = ValueKey('${member.hashCode}_$index');
final options = _getOptions(member.value); final options = _getOptions(member.value);
final initialValue = options.isNotEmpty ? options.first : ''; // Use selectedAction if it exists, otherwise use the first option
final initialValue =
member.selectedAction?.isNotEmpty == true
? member.selectedAction!
: options.isNotEmpty
? options.first
: '';
_nodeTypeData[rowKey] = _NodeTypeData(member.value, initialValue); _nodeTypeData[rowKey] = _NodeTypeData(member.value, initialValue);
@ -221,21 +250,6 @@ class NodeTableState extends State<NodeTable> {
.toList(); .toList();
} }
List<String> _getOptions(String type) {
switch (type) {
case Constants.atomFunction:
case Constants.atomService:
case Constants.business:
return ['普通调用', '事务调用'];
case Constants.uftTable:
return ['获取记录', '遍历记录', '插入记录', '修改记录', '删除记录'];
case Constants.component:
return ['获取组件', '遍历组件', '插入组件', '修改组件', '尾部插入组件'];
default:
return ['']; //
}
}
void refreshTable() { void refreshTable() {
if (_stateManager != null) { if (_stateManager != null) {
_stateManager!.refRows.clear(); _stateManager!.refRows.clear();
@ -250,4 +264,8 @@ class NodeTableState extends State<NodeTable> {
_scrollController.dispose(); _scrollController.dispose();
super.dispose(); super.dispose();
} }
List<String> _getOptions(String s) {
return CodeCreateService.getNodeActions(s);
}
} }

5
win_text_editor/lib/modules/memory_table/controllers/memory_table_controller.dart

@ -18,13 +18,12 @@ class MemoryTableController extends BaseContentController {
late DataGridSource fieldsSource; late DataGridSource fieldsSource;
late DataGridSource indexesSource; late DataGridSource indexesSource;
final MemoryTableService _service;
final MacroTemplateService templateService = MacroTemplateService(); final MacroTemplateService templateService = MacroTemplateService();
// MemoryTable对象 // MemoryTable对象
late MemoryTable _memoryTable; late MemoryTable _memoryTable;
MemoryTableController() : _service = MemoryTableService(Logger()) { MemoryTableController() {
// //
final initialFields = [Field('1', '', '', '', false), Field('2', '', '', '', false)]; final initialFields = [Field('1', '', '', '', false), Field('2', '', '', '', false)];
@ -68,7 +67,7 @@ class MemoryTableController extends BaseContentController {
Future<void> onOpenFile(String filePath, {dynamic appendArg}) async { Future<void> onOpenFile(String filePath, {dynamic appendArg}) async {
Logger().info("Opening file: $filePath"); Logger().info("Opening file: $filePath");
try { try {
final tableData = await _service.parseStructureFile(filePath); final tableData = await MemoryTableService.parseStructureFile(filePath);
// Update controller state // Update controller state
tableName = tableData.tableName; tableName = tableData.tableName;

24
win_text_editor/lib/modules/memory_table/models/memory_table.dart

@ -114,14 +114,34 @@ class MemoryTable {
'isPrimary': index.isPrimary, 'isPrimary': index.isPrimary,
'fields': 'fields':
index.fields.asMap().entries.map((entry) { index.fields.asMap().entries.map((entry) {
return {'name': entry.value, 'isLast': entry.key == keyFields.length - 1}; return {
'name': entry.value,
'partnerName':
codePartners
.firstWhere(
(partner) => partner.fields.any((f) => f == entry.value),
orElse: () => CodePartner(name: null, fields: []),
)
.name,
'isLast': entry.key == keyFields.length - 1,
};
}).toList(), }).toList(),
'rule': index.rule, 'rule': index.rule,
}, },
'keyName': keyName, 'keyName': keyName,
'keyFields': 'keyFields':
keyFields.asMap().entries.map((entry) { keyFields.asMap().entries.map((entry) {
return {'name': entry.value, 'isLast': entry.key == keyFields.length - 1}; return {
'name': entry.value,
'partnerName':
codePartners
.firstWhere(
(partner) => partner.fields.any((f) => f == entry.value),
orElse: () => CodePartner(name: null, fields: []),
)
.name,
'isLast': entry.key == keyFields.length - 1,
};
}).toList(), }).toList(),
}; };
} }

10
win_text_editor/lib/modules/memory_table/services/memory_table_service.dart

@ -9,11 +9,7 @@ import 'package:path/path.dart' as path;
import 'package:win_text_editor/framework/controllers/logger.dart'; import 'package:win_text_editor/framework/controllers/logger.dart';
class MemoryTableService { class MemoryTableService {
final Logger _logger; static Future<TableData> parseStructureFile(String filePath) async {
MemoryTableService(this._logger);
Future<TableData> parseStructureFile(String filePath) async {
try { try {
// 1. Check file extension // 1. Check file extension
if (!filePath.toLowerCase().endsWith('.uftstructure')) { if (!filePath.toLowerCase().endsWith('.uftstructure')) {
@ -95,10 +91,10 @@ class MemoryTableService {
indexes: indexList.isNotEmpty ? indexList : [], indexes: indexList.isNotEmpty ? indexList : [],
); );
} on xml.XmlParserException catch (e) { } on xml.XmlParserException catch (e) {
_logger.error("XML解析错误: ${e.message}"); Logger().error("XML解析错误: ${e.message}");
rethrow; rethrow;
} catch (e) { } catch (e) {
_logger.error("处理文件时发生错误: $e"); Logger().error("处理文件时发生错误: $e");
rethrow; rethrow;
} }
} }

5
win_text_editor/lib/modules/outline/models/outline_node.dart

@ -27,6 +27,9 @@ class OutlineNode implements TreeNode {
final String value; final String value;
String? wordClass; String? wordClass;
String? selectedAction;
String? path;
OutlineNode({ OutlineNode({
required this.name, required this.name,
required this.value, required this.value,
@ -39,6 +42,8 @@ class OutlineNode implements TreeNode {
this.wordClass, this.wordClass,
List<OutlineNode>? children, List<OutlineNode>? children,
this.title = "", this.title = "",
this.selectedAction = "",
this.path = "",
}) : _children = children ?? [] { }) : _children = children ?? [] {
id = DateTime.now().microsecondsSinceEpoch.toRadixString(36) + title.hashCode.toRadixString(36); id = DateTime.now().microsecondsSinceEpoch.toRadixString(36) + title.hashCode.toRadixString(36);
} }

8
win_text_editor/lib/modules/outline/services/component_service.dart

@ -1,6 +1,5 @@
import 'dart:io'; import 'dart:io';
import 'package:flutter/widgets.dart';
import 'package:win_text_editor/framework/common/constants.dart'; import 'package:win_text_editor/framework/common/constants.dart';
import 'package:win_text_editor/framework/controllers/logger.dart'; import 'package:win_text_editor/framework/controllers/logger.dart';
import 'package:win_text_editor/modules/outline/models/outline_node.dart'; import 'package:win_text_editor/modules/outline/models/outline_node.dart';
@ -9,10 +8,13 @@ import 'package:xml/xml.dart';
class ComponentService { class ComponentService {
// //
static Map<String, List<String>> componentFieldMap = {}; static Map<String, List<String>> componentFieldMap = {};
static String filePath = "";
// //
static Future<void> initComponentFieldMap(String rootPath) async { static Future<void> initComponentFieldMap(String rootPath) async {
final componentFile = File('$rootPath/metadata/component.xml'); filePath = '$rootPath\\metadata\\component.xml';
final componentFile = File(filePath);
if (!await componentFile.exists()) { if (!await componentFile.exists()) {
Logger().error('component.xml文件不存在'); Logger().error('component.xml文件不存在');
return; return;
@ -68,6 +70,7 @@ class ComponentService {
value: Constants.component, value: Constants.component,
isDirectory: false, isDirectory: false,
depth: 4, depth: 4,
path: filePath,
), ),
), ),
); );
@ -84,6 +87,7 @@ class ComponentService {
value: Constants.component, value: Constants.component,
isDirectory: false, isDirectory: false,
depth: 4, depth: 4,
path: filePath,
), ),
); );
} }

22
win_text_editor/lib/modules/outline/services/functions_service.dart

@ -1,5 +1,6 @@
import 'dart:io'; import 'dart:io';
import 'package:path/path.dart' as path;
import 'package:win_text_editor/framework/common/constants.dart'; import 'package:win_text_editor/framework/common/constants.dart';
import 'package:win_text_editor/framework/controllers/logger.dart'; import 'package:win_text_editor/framework/controllers/logger.dart';
import 'package:win_text_editor/modules/outline/models/outline_node.dart'; import 'package:win_text_editor/modules/outline/models/outline_node.dart';
@ -89,11 +90,12 @@ class FunctionsService {
if (value.contains(fieldName)) { if (value.contains(fieldName)) {
parentNode.children.add( parentNode.children.add(
OutlineNode( OutlineNode(
name: value[0], // name: path.basenameWithoutExtension(value[0]), //
title: key, title: key,
value: Constants.atomService, value: Constants.atomService,
isDirectory: false, // isDirectory: false, //
depth: 4, depth: 4,
path: value[0],
), ),
); );
} }
@ -107,11 +109,12 @@ class FunctionsService {
if (value.contains(fieldName)) { if (value.contains(fieldName)) {
parentNode.children.add( parentNode.children.add(
OutlineNode( OutlineNode(
name: value[0], // name: path.basenameWithoutExtension(value[0]), //
title: key, title: key,
value: Constants.atomFunction, value: Constants.atomFunction,
isDirectory: false, // isDirectory: false, //
depth: 4, depth: 4,
path: value[0],
), ),
); );
} }
@ -123,14 +126,16 @@ class FunctionsService {
if (fieldName == null || fieldName.isEmpty) return; if (fieldName == null || fieldName.isEmpty) return;
uftbusinessMap.forEach((key, value) { uftbusinessMap.forEach((key, value) {
final functionName = path.basenameWithoutExtension(value[0]);
if (value.contains(fieldName)) { if (value.contains(fieldName)) {
parentNode.children.add( parentNode.children.add(
OutlineNode( OutlineNode(
name: value[0], // name: functionName,
title: key, title: key,
value: Constants.business, value: Constants.business,
isDirectory: false, // isDirectory: false, //
depth: 4, depth: 4,
path: value[0], //
), ),
); );
} }
@ -140,14 +145,16 @@ class FunctionsService {
static List<OutlineNode> searchBusiness(String searchQuery) { static List<OutlineNode> searchBusiness(String searchQuery) {
final List<OutlineNode> nodes = []; final List<OutlineNode> nodes = [];
uftbusinessMap.forEach((key, value) { uftbusinessMap.forEach((key, value) {
final functionName = path.basenameWithoutExtension(value[0]);
if (key.contains(searchQuery)) { if (key.contains(searchQuery)) {
nodes.add( nodes.add(
OutlineNode( OutlineNode(
name: value[0], // name: functionName,
title: key, title: key,
value: Constants.business, value: Constants.business,
isDirectory: false, // isDirectory: false, //
depth: 4, depth: 4,
path: value[0], //
), ),
); );
} }
@ -159,14 +166,16 @@ class FunctionsService {
static List<OutlineNode> searchAtomServices(String searchQuery) { static List<OutlineNode> searchAtomServices(String searchQuery) {
final List<OutlineNode> nodes = []; final List<OutlineNode> nodes = [];
uftatomserviceMap.forEach((key, value) { uftatomserviceMap.forEach((key, value) {
final functionName = path.basenameWithoutExtension(value[0]);
if (key.contains(searchQuery)) { if (key.contains(searchQuery)) {
nodes.add( nodes.add(
OutlineNode( OutlineNode(
name: value[0], // name: functionName,
title: key, title: key,
value: Constants.atomService, value: Constants.atomService,
isDirectory: false, // isDirectory: false, //
depth: 4, depth: 4,
path: value[0], //
), ),
); );
} }
@ -181,11 +190,12 @@ class FunctionsService {
if (key.contains(searchQuery)) { if (key.contains(searchQuery)) {
nodes.add( nodes.add(
OutlineNode( OutlineNode(
name: value[0], // name: path.basenameWithoutExtension(value[0]),
title: key, title: key,
value: Constants.atomFunction, value: Constants.atomFunction,
isDirectory: false, // isDirectory: false, //
depth: 4, depth: 4,
path: value[0], //
), ),
); );
} }

11
win_text_editor/lib/modules/outline/services/uft_object_service.dart

@ -66,13 +66,15 @@ class UftObjectService {
uftObjectMap.forEach((key, value) { uftObjectMap.forEach((key, value) {
if (value.contains(fieldName)) { if (value.contains(fieldName)) {
final tableName = path.basenameWithoutExtension(value[0]);
parentNode.children.add( parentNode.children.add(
OutlineNode( OutlineNode(
name: value[0], name: tableName,
title: key, title: key,
value: Constants.uftTable, value: Constants.uftTable,
isDirectory: false, // isDirectory: false, //
depth: 4, depth: 4,
path: value[0],
), ),
); );
} }
@ -83,15 +85,16 @@ class UftObjectService {
final List<OutlineNode> searchNodes = []; final List<OutlineNode> searchNodes = [];
uftObjectMap.forEach((key, value) { uftObjectMap.forEach((key, value) {
if (key.contains(searchQuery) || final tableName = path.basenameWithoutExtension(value[0]);
path.basenameWithoutExtension(value[0]).contains(searchQuery)) { if (key.contains(searchQuery) || tableName.contains(searchQuery)) {
searchNodes.add( searchNodes.add(
OutlineNode( OutlineNode(
name: value[0], name: tableName,
title: key, title: key,
value: Constants.uftTable, value: Constants.uftTable,
isDirectory: false, isDirectory: false,
depth: 4, depth: 4,
path: value[0],
), ),
); );
} }

9
win_text_editor/lib/modules/outline/widgets/outline_explorer.dart

@ -178,15 +178,12 @@ class _OutlineExplorerState extends State<OutlineExplorer> {
case Constants.atomFunction: case Constants.atomFunction:
case Constants.atomService: case Constants.atomService:
if (widget.onFileDoubleTap != null) { if (widget.onFileDoubleTap != null) {
widget.onFileDoubleTap!(outlineNode.name, null); widget.onFileDoubleTap!(outlineNode.path!, null);
} }
break; break;
case Constants.component: case Constants.component:
if (widget.onFileDoubleTap != null && fileProvider.rootPath != null) { if (widget.onFileDoubleTap != null) {
widget.onFileDoubleTap!( widget.onFileDoubleTap!(outlineNode.path!, outlineNode.name);
'${fileProvider.rootPath}\\metadata\\component.xml',
outlineNode.name,
);
} }
break; break;
} }

3
win_text_editor/lib/modules/uft_component/services/uft_component_service.dart

@ -17,7 +17,8 @@ class UftComponentService {
static Future<UftComponent?> getUftComponent(String filePath, String componentName) async { static Future<UftComponent?> getUftComponent(String filePath, String componentName) async {
if (_components.isNotEmpty) { if (_components.isNotEmpty) {
return _components.firstWhereOrNull((c) => c.name == componentName); final com = _components.firstWhereOrNull((c) => c.name == componentName);
if (com != null) return com;
} }
return parseComponentFile( return parseComponentFile(
filePath, filePath,

Loading…
Cancel
Save