vue2-client 1.15.87 → 1.15.88
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.editorconfig +9 -9
- package/.env +20 -20
- package/.env.development +1 -1
- package/.env.iot +19 -19
- package/.env.message +19 -19
- package/.env.revenue +19 -19
- package/.env.runtime +19 -19
- package/.eslintrc.json +5 -5
- package/CLAUDE.md +89 -89
- package/README.md +65 -65
- package/babel.config.js +21 -21
- package/docs/Logic/345/207/275/346/225/260/344/275/277/347/224/250/347/233/270/345/205/263.md +46 -46
- package/docs/notice.md +22 -22
- package/docs//345/207/275/346/225/260/344/275/277/347/224/250/347/233/270/345/205/263.md +179 -179
- package/jest.config.js +22 -22
- package/package.json +111 -111
- package/src/App.vue +196 -196
- package/src/ReportView.js +13 -13
- package/src/base-client/components/common/AddressSearchCombobox/AddressSearchCombobox.vue +532 -532
- package/src/base-client/components/common/AddressSearchCombobox/index.js +3 -3
- package/src/base-client/components/common/AmapMarker/index.js +3 -3
- package/src/base-client/components/common/CitySelect/CitySelect.vue +376 -376
- package/src/base-client/components/common/CitySelect/demo.vue +43 -43
- package/src/base-client/components/common/ColorPickerCombobox/ColorPickerCombobox.vue +99 -99
- package/src/base-client/components/common/ColorPickerCombobox/demo.vue +34 -34
- package/src/base-client/components/common/FormGroupEdit/FormGroupEdit.vue +146 -146
- package/src/base-client/components/common/JSONToTree/index.js +3 -3
- package/src/base-client/components/common/Upload/Upload.vue +323 -323
- package/src/base-client/components/common/XAddForm/index.js +3 -3
- package/src/base-client/components/common/XAddForm/index.md +61 -61
- package/src/base-client/components/common/XAddNativeForm/XAddNativeForm.vue +1169 -1169
- package/src/base-client/components/common/XAddNativeForm/demo.vue +54 -54
- package/src/base-client/components/common/XAddNativeForm/index.js +3 -3
- package/src/base-client/components/common/XAddReport/XAddReport.vue +212 -212
- package/src/base-client/components/common/XBadge/index.js +3 -3
- package/src/base-client/components/common/XBadge/index.md +39 -39
- package/src/base-client/components/common/XButtons/XButtons.vue +284 -284
- package/src/base-client/components/common/XCalendar/XCalendar.vue +369 -369
- package/src/base-client/components/common/XCalendar/index.md +284 -284
- package/src/base-client/components/common/XCard/index.js +3 -3
- package/src/base-client/components/common/XCard/index.md +43 -43
- package/src/base-client/components/common/XCardSet/XCardSet.vue +300 -300
- package/src/base-client/components/common/XCollapse/XCollapse.vue +354 -354
- package/src/base-client/components/common/XConversation/XConversation.vue +576 -576
- package/src/base-client/components/common/XConversation/XConversationDemo.vue +28 -28
- package/src/base-client/components/common/XDataCard/XDataCard.vue +629 -629
- package/src/base-client/components/common/XDatePicker/index.vue +276 -276
- package/src/base-client/components/common/XDescriptions/XDescriptions.vue +174 -174
- package/src/base-client/components/common/XDescriptions/XDescriptionsGroup.vue +314 -314
- package/src/base-client/components/common/XDescriptions/demo.vue +51 -51
- package/src/base-client/components/common/XForm/XForm.vue +420 -420
- package/src/base-client/components/common/XForm/XFormItem.vue +1474 -1474
- package/src/base-client/components/common/XForm/XTreeSelect.vue +276 -276
- package/src/base-client/components/common/XForm/demo.vue +105 -105
- package/src/base-client/components/common/XForm/index.js +3 -3
- package/src/base-client/components/common/XFormCol/XFormCol.vue +157 -157
- package/src/base-client/components/common/XFormCol/index.js +3 -3
- package/src/base-client/components/common/XFormCol/index.md +35 -35
- package/src/base-client/components/common/XFormGroup/XFormGroup.vue +301 -301
- package/src/base-client/components/common/XFormGroup/demo.vue +41 -41
- package/src/base-client/components/common/XFormTable/XFormTable.vue +938 -938
- package/src/base-client/components/common/XFormTable/demo.vue +87 -87
- package/src/base-client/components/common/XFormTable/index.js +3 -3
- package/src/base-client/components/common/XImportExcel/XImportExcel.vue +174 -174
- package/src/base-client/components/common/XImportExcel/index.js +3 -3
- package/src/base-client/components/common/XImportExcel/index.md +38 -38
- package/src/base-client/components/common/XInput/XInput.vue +128 -128
- package/src/base-client/components/common/XInput/index.js +3 -3
- package/src/base-client/components/common/XInput/index.md +97 -97
- package/src/base-client/components/common/XIntervalPicker/XIntervalPicker.vue +121 -121
- package/src/base-client/components/common/XLicensePlate/index.js +3 -3
- package/src/base-client/components/common/XLicensePlate/index.md +38 -38
- package/src/base-client/components/common/XPrint/Demo.vue +41 -41
- package/src/base-client/components/common/XPrint/PrintBill.vue +308 -308
- package/src/base-client/components/common/XRate/demo.vue +102 -102
- package/src/base-client/components/common/XRate/index.vue +149 -149
- package/src/base-client/components/common/XReport/XReport.vue +963 -963
- package/src/base-client/components/common/XReport/XReportDemo.vue +70 -70
- package/src/base-client/components/common/XReport/XReportTrGroup.vue +1005 -1005
- package/src/base-client/components/common/XReport/index.md +103 -103
- package/src/base-client/components/common/XReportDrawer/XReportDrawer.vue +201 -201
- package/src/base-client/components/common/XReportGrid/XReport.vue +1075 -1075
- package/src/base-client/components/common/XReportGrid/XReportDemo.vue +44 -44
- package/src/base-client/components/common/XReportGrid/XReportDesign.vue +620 -620
- package/src/base-client/components/common/XReportGrid/XReportTrGroup.vue +723 -723
- package/src/base-client/components/common/XReportGrid/print.js +184 -184
- package/src/base-client/components/common/XTab/XTab.vue +299 -299
- package/src/base-client/components/common/XTable/ExportExcel.vue +283 -283
- package/src/base-client/components/common/XTable/XTable.vue +1599 -1599
- package/src/base-client/components/common/XTable/XTableWrapper.vue +597 -597
- package/src/base-client/components/common/XTable/index.js +3 -3
- package/src/base-client/components/common/XTimeline/XTimeline.vue +358 -358
- package/src/base-client/components/common/XTimeline/index.md +191 -191
- package/src/base-client/components/common/XTree/XTreePro.vue +452 -450
- package/src/base-client/components/common/XTreeOne/index.js +3 -3
- package/src/base-client/components/common/XUploadFilesView/index.vue +485 -485
- package/src/base-client/components/his/XCharge/XCharge.vue +238 -238
- package/src/base-client/components/his/XCheckbox/XCheckbox.vue +105 -105
- package/src/base-client/components/his/XCheckbox/index.md +253 -253
- package/src/base-client/components/his/XHDescriptions/XHDescriptions.vue +430 -430
- package/src/base-client/components/his/XHDescriptions/index.md +217 -217
- package/src/base-client/components/his/XHisEditor/XHisEditor.vue +629 -629
- package/src/base-client/components/his/XHisEditor/diagnosisAutocomplete.js +263 -263
- package/src/base-client/components/his/XList/XList.vue +495 -495
- package/src/base-client/components/his/XQuestionnaire/XQuestionnaire.json +3 -3
- package/src/base-client/components/his/XQuestionnaire/XQuestionnaire.vue +106 -106
- package/src/base-client/components/his/XQuestionnaire/XQuestionnaireDemo.vue +51 -51
- package/src/base-client/components/his/XQuestionnaire/XQuestionnaireItem.vue +269 -269
- package/src/base-client/components/his/XRadio/XRadio.vue +125 -125
- package/src/base-client/components/his/XRadio/index.md +234 -234
- package/src/base-client/components/his/XSelect/XSelect.vue +72 -72
- package/src/base-client/components/his/XShiftSchedule/XShiftSchedule.vue +234 -234
- package/src/base-client/components/his/XShiftSchedule/dome.vue +29 -29
- package/src/base-client/components/his/XSidebar/XSidebar.vue +240 -240
- package/src/base-client/components/his/XTextCard/XTextCard.vue +207 -207
- package/src/base-client/components/his/XTimeSelect/XTimeSelect.vue +162 -162
- package/src/base-client/components/his/XTimeSelect/XTimeSelectDemo.vue +23 -23
- package/src/base-client/components/his/XTitle/README.md +113 -113
- package/src/base-client/components/his/XTitle/XTitle.vue +123 -123
- package/src/base-client/components/his/XTreeRows/TreeNode.vue +107 -107
- package/src/base-client/components/his/XTreeRows/XTreeRows.vue +307 -307
- package/src/base-client/components/his/threeTestOrders/dome.vue +68 -68
- package/src/base-client/components/his/threeTestOrders/editor.vue +111 -111
- package/src/base-client/components/his/threeTestOrders/textBox.vue +457 -457
- package/src/base-client/components/his/threeTestOrders/threeTestOrders.vue +475 -475
- package/src/base-client/components/layout/XPageView/RenderRow.vue +88 -88
- package/src/base-client/components/layout/XPageView/XErrorView.vue +22 -22
- package/src/base-client/components/layout/XPageView/XPageRowTemplate.vue +37 -37
- package/src/base-client/components/layout/XPageView/XPageView.vue +223 -223
- package/src/base-client/components/layout/XPageView/XTab/XTab.vue +96 -96
- package/src/base-client/components/layout/XPageView/XTab/index.js +3 -3
- package/src/base-client/components/layout/XPageView/componentTypes.js +22 -22
- package/src/base-client/components/layout/XPageView/index.js +2 -2
- package/src/base-client/components/layout/XPageView/index.md +96 -96
- package/src/base-client/components/system/DictionaryDetailsView/index.js +3 -3
- package/src/base-client/components/system/DictionaryDetailsView/index.md +41 -41
- package/src/base-client/components/system/LogDetailsView/LogDetailsView.vue +376 -376
- package/src/base-client/components/system/LogDetailsView/index.js +3 -3
- package/src/base-client/components/system/LogDetailsView/index.md +41 -41
- package/src/base-client/components/system/QueryParamsDetailsView/index.js +3 -3
- package/src/base-client/components/ticket/TicketDetailsView/TicketDetailsView.vue +807 -807
- package/src/base-client/components/ticket/TicketDetailsView/index.js +3 -3
- package/src/base-client/components/ticket/TicketDetailsView/index.md +29 -29
- package/src/base-client/components/ticket/TicketDetailsView/part/TicketDetailsFlow.vue +260 -260
- package/src/base-client/components/ticket/TicketDetailsView/part/index.js +3 -3
- package/src/base-client/components/ticket/TicketSubmitSuccessView/TicketSubmitSuccessView.vue +532 -532
- package/src/base-client/components/ticket/TicketSubmitSuccessView/index.js +3 -3
- package/src/base-client/components/ticket/TicketSubmitSuccessView/index.md +29 -29
- package/src/base-client/plugins/AppData.js +126 -126
- package/src/base-client/plugins/PagedList.js +177 -177
- package/src/base-client/plugins/__tests__/selectValueTypeHelper.test.js +154 -154
- package/src/base-client/plugins/authority-plugin.js +167 -167
- package/src/base-client/plugins/compatible/LoginServiceOA.js +20 -20
- package/src/base-client/plugins/i18n-extend.js +32 -32
- package/src/base-client/plugins/moment.js +8 -8
- package/src/base-client/plugins/selectValueTypeHelper.js +281 -281
- package/src/bootstrap.js +51 -51
- package/src/components/Ellipsis/Ellipsis.vue +65 -65
- package/src/components/Ellipsis/index.js +3 -3
- package/src/components/Ellipsis/index.md +38 -38
- package/src/components/FileImageItem/FileItem.vue +320 -320
- package/src/components/FileImageItem/FileItemGroup.vue +197 -197
- package/src/components/FileImageItem/ImageItem.vue +107 -107
- package/src/components/FileImageItem/index.js +4 -4
- package/src/components/FilePreview/FilePreview.vue +181 -181
- package/src/components/FilePreview/FilePreviewDemo.vue +30 -30
- package/src/components/FilePreview/index.js +3 -3
- package/src/components/HeightScanner/index.vue +615 -615
- package/src/components/STable/README.md +341 -341
- package/src/components/STable/index.js +558 -558
- package/src/components/TableSetting/TableSetting.vue +143 -143
- package/src/components/TableSetting/index.js +3 -3
- package/src/components/Trend/Trend.vue +41 -41
- package/src/components/Trend/index.js +3 -3
- package/src/components/Trend/index.less +41 -41
- package/src/components/Trend/index.md +45 -45
- package/src/components/_util/util.js +46 -46
- package/src/components/cache/AKeepAlive.js +179 -179
- package/src/components/exception/typeConfig.js +19 -19
- package/src/components/form/FormRow.vue +52 -52
- package/src/components/index.less +5 -5
- package/src/components/menu/Contextmenu.vue +84 -84
- package/src/components/menu/index.less +38 -38
- package/src/components/page/header/PageHeader.vue +64 -64
- package/src/components/page/header/index.less +40 -40
- package/src/components/result/Result.vue +77 -77
- package/src/components/setting/SettingItem.vue +26 -26
- package/src/components/setting/i18n.js +117 -117
- package/src/components/table/StandardTable.vue +141 -141
- package/src/components/table/advance/ActionColumns.vue +158 -158
- package/src/components/table/advance/ActionSize.vue +45 -45
- package/src/components/table/advance/AdvanceTable.vue +275 -275
- package/src/components/table/advance/SearchArea.vue +355 -355
- package/src/components/table/advance/index.js +2 -2
- package/src/components/table/api/ApiTable.vue +50 -50
- package/src/components/task/TaskGroup.vue +80 -80
- package/src/components/task/TaskItem.vue +26 -26
- package/src/components/tool/AvatarList.vue +68 -68
- package/src/components/tool/DetailList.vue +157 -157
- package/src/components/tool/Drawer.vue +142 -142
- package/src/components/tool/FooterToolBar.vue +30 -30
- package/src/components/tool/HeadInfo.vue +35 -35
- package/src/components/tool/TagSelect.vue +83 -83
- package/src/components/tool/TagSelectOption.vue +33 -33
- package/src/components/transition/PageToggleTransition.vue +97 -97
- package/src/config/default/admin.config.js +18 -18
- package/src/config/default/animate.config.js +21 -21
- package/src/config/default/index.js +6 -6
- package/src/config/index.js +3 -3
- package/src/config/replacer/index.js +10 -10
- package/src/config/replacer/resolve.config.js +67 -67
- package/src/expression/ExpressionRunner.js +26 -26
- package/src/expression/TestExpression.js +509 -509
- package/src/expression/core/Delegate.js +115 -115
- package/src/expression/core/Expression.js +1358 -1358
- package/src/expression/core/Program.js +932 -932
- package/src/expression/core/Token.js +27 -27
- package/src/expression/enums/ExpressionType.js +81 -81
- package/src/expression/enums/TokenType.js +11 -11
- package/src/expression/exception/BreakWayException.js +2 -2
- package/src/expression/exception/ContinueWayException.js +2 -2
- package/src/expression/exception/ExpressionException.js +28 -28
- package/src/expression/exception/ReturnWayException.js +14 -14
- package/src/expression/exception/ServiceException.js +22 -22
- package/src/expression/instances/LogicConsole.js +44 -44
- package/src/expression/ts/ExpressionRunner.ts +28 -28
- package/src/expression/ts/TestExpression.ts +509 -509
- package/src/expression/ts/core/Delegate.ts +114 -114
- package/src/expression/ts/core/Expression.ts +1309 -1309
- package/src/expression/ts/core/Program.ts +950 -950
- package/src/expression/ts/core/Token.ts +29 -29
- package/src/expression/ts/enums/ExpressionType.ts +81 -81
- package/src/expression/ts/enums/TokenType.ts +13 -13
- package/src/expression/ts/exception/BreakWayException.ts +2 -2
- package/src/expression/ts/exception/ContinueWayException.ts +2 -2
- package/src/expression/ts/exception/ExpressionException.ts +28 -28
- package/src/expression/ts/exception/ReturnWayException.ts +14 -14
- package/src/expression/ts/exception/ServiceException.ts +22 -22
- package/src/expression/ts/instances/JSONArray.ts +48 -48
- package/src/expression/ts/instances/JSONObject.ts +109 -109
- package/src/expression/ts/instances/LogicConsole.ts +32 -32
- package/src/layouts/AdminLayout.vue +176 -176
- package/src/layouts/BlankView.vue +79 -79
- package/src/layouts/ComponentLayoutOne.vue +47 -47
- package/src/layouts/GridView.vue +43 -43
- package/src/layouts/PageView.vue +55 -55
- package/src/layouts/footer/PageFooter.vue +49 -49
- package/src/layouts/header/HeaderAvatar.vue +64 -64
- package/src/layouts/header/HeaderSearch.vue +67 -67
- package/src/layouts/header/index.less +92 -92
- package/src/layouts/tabs/TabsView.vue +383 -383
- package/src/layouts/tabs/i18n.js +25 -25
- package/src/layouts/tabs/index.js +2 -2
- package/src/logic/LogicRunner.js +62 -62
- package/src/logic/TestLogic.js +13 -13
- package/src/logic/plugins/common/DateTools.js +35 -35
- package/src/logic/plugins/common/VueTools.js +30 -30
- package/src/logic/plugins/index.js +7 -7
- package/src/logic/ts/LogicRunner.ts +67 -67
- package/src/logic/ts/TestLogic.ts +13 -13
- package/src/main.js +33 -33
- package/src/mixins/formValidationMixin.js +46 -46
- package/src/mock/common/activityData.js +32 -32
- package/src/mock/common/index.js +89 -89
- package/src/mock/common/reportData.js +20 -20
- package/src/mock/common/tableData.js +118 -118
- package/src/mock/index.js +12 -12
- package/src/mock/project/index.js +17 -17
- package/src/mock/user/current.js +13 -13
- package/src/mock/user/login.js +39 -39
- package/src/mock/user/routes.js +61 -61
- package/src/mock/workplace/index.js +15 -15
- package/src/pages/LogicCallExample/index.vue +46 -46
- package/src/pages/ReportGrid/index.vue +76 -76
- package/src/pages/ReportView.vue +50 -50
- package/src/pages/WorkflowDetail/WorkFlowDemo.vue +47 -47
- package/src/pages/WorkflowDetail/WorkFlowDemo2.vue +204 -204
- package/src/pages/WorkflowDetail/WorkFlowDemo3.vue +203 -203
- package/src/pages/WorkflowDetail/WorkflowDetail.vue +391 -391
- package/src/pages/WorkflowDetail/WorkflowPageDetail/LeaveMessage.vue +388 -388
- package/src/pages/WorkflowDetail/WorkflowPageDetail/WorkFlowBaseInformation.vue +415 -415
- package/src/pages/WorkflowDetail/WorkflowPageDetail/WorkFlowHandle.vue +1766 -1766
- package/src/pages/WorkflowDetail/WorkflowPageDetail/WorkFlowHandleReso.vue +975 -975
- package/src/pages/WorkflowDetail/WorkflowPageDetail/WorkFlowPreview.vue +109 -109
- package/src/pages/WorkflowDetail/WorkflowPageDetail/WorkFlowTimeline.vue +929 -929
- package/src/pages/WorkflowDetail/WorkflowPageDetail/WorkOrderParentDetails.vue +222 -222
- package/src/pages/WorkflowDetail/WorkflowPageDetail/WorkflowDetailResso.vue +243 -243
- package/src/pages/WorkflowDetail/WorkflowPageDetail/WorkflowLog.vue +188 -188
- package/src/pages/WorkflowDetail/WorkflowPageDetail/components/WorkflowPersonSelector.vue +109 -109
- package/src/pages/WorkflowDetail/WorkflowPageDetail/worklog.vue +97 -97
- package/src/pages/XPageViewExample/index.vue +149 -149
- package/src/pages/addressSelect/addressDemo.vue +24 -24
- package/src/pages/addressSelect/index.vue +270 -270
- package/src/pages/dashboard/workplace/i18n.js +40 -40
- package/src/pages/dashboard/workplace/index.js +2 -2
- package/src/pages/dashboard/workplace/index.less +59 -59
- package/src/pages/exception/403.vue +21 -21
- package/src/pages/exception/404.vue +24 -24
- package/src/pages/exception/500.vue +21 -21
- package/src/pages/login/index.js +2 -2
- package/src/pages/report/ReportTableHome.vue +28 -28
- package/src/pages/resourceManage/depListManage.vue +23 -23
- package/src/pages/resourceManage/funListManage.vue +23 -23
- package/src/pages/resourceManage/index.js +15 -15
- package/src/pages/resourceManage/resourceManageMain.vue +57 -57
- package/src/pages/resourceManage/roleListManage.vue +23 -23
- package/src/pages/resourceManage/staffListManage.vue +23 -23
- package/src/pages/system/file/Info.vue +56 -56
- package/src/pages/system/file/index.vue +317 -317
- package/src/pages/system/settings/index.vue +126 -126
- package/src/pages/userInfoDetailManage/FillCardRecordQuery/index.vue +77 -77
- package/src/pages/userInfoDetailManage/FillGasRecordQuery/index.vue +75 -75
- package/src/pages/userInfoDetailManage/InsuranceDetailQuery/index.vue +64 -64
- package/src/pages/userInfoDetailManage/MachineRecordQuery/index.vue +75 -75
- package/src/pages/userInfoDetailManage/OtherChargeRecordQuery/index.vue +75 -75
- package/src/pages/userInfoDetailManage/PriceAdjustments/index.vue +64 -64
- package/src/pages/userInfoDetailManage/UserChargeRecordQuery/index.vue +94 -94
- package/src/pages/userInfoDetailManage/UserException/index.vue +64 -64
- package/src/pages/userInfoDetailManage/UserHandRecordQuery/index.vue +87 -87
- package/src/pages/userInfoDetailManage/UserRecordQuery/index.vue +74 -74
- package/src/pages/userInfoDetailManage/index.vue +290 -290
- package/src/pages/userInfoDetailManage/uploadFilesHistory/ImagePreview.vue +101 -101
- package/src/pages/userInfoDetailManage/uploadFilesHistory/index.vue +129 -129
- package/src/pages/userInfoDetailManage/userInfoDetailQueryTabs.vue +144 -144
- package/src/plugins/HiPrintPlugin.js +164 -164
- package/src/router/async/router.map.js +126 -126
- package/src/router/guards.js +262 -262
- package/src/router/i18n.js +57 -57
- package/src/router.js +17 -17
- package/src/services/api/DictionaryDetailsViewApi.js +6 -6
- package/src/services/api/LogDetailsViewApi.js +10 -10
- package/src/services/api/QueryParamsDetailsViewApi.js +6 -6
- package/src/services/api/logininfor/index.js +6 -6
- package/src/services/api/manage.js +8 -8
- package/src/services/api/restTools.js +215 -215
- package/src/services/api/workFlow.js +57 -57
- package/src/services/apiService.js +16 -16
- package/src/services/dataSource.js +12 -12
- package/src/services/index.js +7 -7
- package/src/services/user.js +92 -92
- package/src/services/v3Api.js +116 -116
- package/src/store/index.js +5 -5
- package/src/store/mutation-types.js +4 -4
- package/src/theme/antd/ant-menu.less +2 -2
- package/src/theme/antd/ant-message.less +3 -3
- package/src/theme/antd/ant-table.less +22 -22
- package/src/theme/antd/ant-time-picker.less +3 -3
- package/src/theme/antd/index.less +3 -3
- package/src/theme/default/color.less +43 -43
- package/src/theme/default/index.less +3 -3
- package/src/theme/default/nprogress.less +76 -76
- package/src/theme/global.less +279 -279
- package/src/theme/index.less +5 -5
- package/src/theme/reportTable.less +58 -58
- package/src/theme/theme.less +1 -1
- package/src/utils/EncryptUtil.js +162 -162
- package/src/utils/Objects.js +25 -25
- package/src/utils/axios-interceptors.js +100 -100
- package/src/utils/colors.js +107 -107
- package/src/utils/common.js +10 -10
- package/src/utils/excel/Blob.js +180 -180
- package/src/utils/excel/Export2Excel.js +141 -141
- package/src/utils/filter.js +21 -21
- package/src/utils/i18n.js +80 -80
- package/src/utils/indexedDB.js +549 -549
- package/src/utils/microAppUtils.js +49 -49
- package/src/utils/request.js +395 -395
- package/src/utils/routerUtil.js +553 -553
- package/src/utils/themeUtil.js +100 -100
- package/test/Tree.spec.js +168 -168
- package/test/myDialog.spec.js +47 -47
- package/test/request.test.js +17 -17
- package/test/util.test.js +53 -53
- package/test/v3Api.test.js +1984 -1984
- package/tests/unit/ReportTable.spec.js +16 -16
- package/vue.config.js +222 -222
- package/.claude/settings.local.json +0 -12
@@ -1,950 +1,950 @@
|
|
1
|
-
import Token from './Token'
|
2
|
-
import Expression from './Expression'
|
3
|
-
import Delegate from "./Delegate"
|
4
|
-
import ExpressionType from "../enums/ExpressionType"
|
5
|
-
import TokenType from "../enums/TokenType"
|
6
|
-
import JSONArray from "../instances/JSONArray"
|
7
|
-
|
8
|
-
/**
|
9
|
-
* 表达式入口
|
10
|
-
*/
|
11
|
-
export default class Program {
|
12
|
-
// 源程序
|
13
|
-
public readonly SOURCE : string;
|
14
|
-
public readonly SOURCE_LENGTH : number;
|
15
|
-
/**
|
16
|
-
* Token队列,用于回退
|
17
|
-
*/
|
18
|
-
private readonly tokens: Token[] = [];
|
19
|
-
/**
|
20
|
-
* 字符串处理环节堆栈,$进入字符串处理环节,“{}”中间部分脱离字符串处理环节
|
21
|
-
*/
|
22
|
-
private readonly inStrings: boolean[] = [];
|
23
|
-
|
24
|
-
/**
|
25
|
-
* 当前获取到的字符位置
|
26
|
-
*/
|
27
|
-
private pos: number = 0;
|
28
|
-
|
29
|
-
/**
|
30
|
-
* 当前是否在字符串处理环节
|
31
|
-
*/
|
32
|
-
private inString: boolean = false;
|
33
|
-
|
34
|
-
constructor(source: string) {
|
35
|
-
this.SOURCE = source;
|
36
|
-
this.SOURCE_LENGTH = this.SOURCE.length;
|
37
|
-
}
|
38
|
-
|
39
|
-
/**
|
40
|
-
* 调用解析过程
|
41
|
-
*/
|
42
|
-
public parse(): Delegate {
|
43
|
-
// 执行完以下方法后就已经诞生了一个完整的执行流程,也就是被翻译成了一套完整的指令集。
|
44
|
-
return this.createDelegate(this.expression());
|
45
|
-
}
|
46
|
-
|
47
|
-
public parseExpression(expression: Expression): Delegate {
|
48
|
-
// 执行完以下方法后就已经诞生了一个完整的执行流程,也就是被翻译成了一套完整的指令集。
|
49
|
-
return this.createDelegate(expression);
|
50
|
-
}
|
51
|
-
|
52
|
-
/**
|
53
|
-
* 编译,产生可执行单元Delegate
|
54
|
-
*/
|
55
|
-
public createDelegate(expression: Expression): Delegate {
|
56
|
-
return new Delegate(expression, this.SOURCE);
|
57
|
-
}
|
58
|
-
|
59
|
-
/**
|
60
|
-
* 获取完整表达式
|
61
|
-
*/
|
62
|
-
public expression(): Expression {
|
63
|
-
let expression = this.CommaExp();
|
64
|
-
if (this.pos < this.SOURCE_LENGTH) {
|
65
|
-
throw new Error(this.GetExceptionMessage("需要逗号结尾!"));
|
66
|
-
}
|
67
|
-
return expression;
|
68
|
-
}
|
69
|
-
|
70
|
-
/**
|
71
|
-
* 逗号表达式=赋值表达式(,赋值表达式)*
|
72
|
-
*/
|
73
|
-
private CommaExp(): Expression {
|
74
|
-
const exps: Expression[] = [];
|
75
|
-
|
76
|
-
const first: Expression = this.AssignExp();
|
77
|
-
exps.push(first);
|
78
|
-
|
79
|
-
let t: Token = this.GetToken();
|
80
|
-
|
81
|
-
// 没有结束
|
82
|
-
while (t.getType() === TokenType.Oper && (t.getValue() === "," || t.getValue() === ";" || t.getValue() === "\n")) {
|
83
|
-
const r: Expression = this.AssignExp();
|
84
|
-
exps.push(r);
|
85
|
-
t = this.GetToken();
|
86
|
-
}
|
87
|
-
|
88
|
-
this.tokens.push(t);
|
89
|
-
|
90
|
-
return Expression.Comma(exps, this.pos);
|
91
|
-
}
|
92
|
-
|
93
|
-
/**
|
94
|
-
* 赋值表达式=对象属性=一般表达式|一般表达式
|
95
|
-
*/
|
96
|
-
private AssignExp(): Expression {
|
97
|
-
let t: Token = this.GetToken();
|
98
|
-
|
99
|
-
// 把第一个Token的位置记录下来,方便后面回退
|
100
|
-
const firstPos: number = t.getStartPos();
|
101
|
-
|
102
|
-
if (t.getType() !== TokenType.Identy) {
|
103
|
-
this.pos = firstPos;
|
104
|
-
this.tokens.length = 0; // Clear the tokens array
|
105
|
-
this.inString = false;
|
106
|
-
return this.Exp();
|
107
|
-
}
|
108
|
-
|
109
|
-
let objExp: Expression = Expression.Identity(t.getValue(), this.pos);
|
110
|
-
objExp = this.ObjectPath(objExp);
|
111
|
-
|
112
|
-
// 只能给对象属性或者变量赋值
|
113
|
-
if (objExp.type !== ExpressionType.Property && objExp.type !== ExpressionType.Identity) {
|
114
|
-
this.pos = firstPos;
|
115
|
-
this.tokens.length = 0;
|
116
|
-
this.inString = false;
|
117
|
-
return this.Exp();
|
118
|
-
}
|
119
|
-
|
120
|
-
t = this.GetToken();
|
121
|
-
if (t.getType() !== TokenType.Oper || t.getValue() !== "=") {
|
122
|
-
this.pos = firstPos;
|
123
|
-
this.tokens.length = 0;
|
124
|
-
this.inString = false;
|
125
|
-
return this.Exp();
|
126
|
-
}
|
127
|
-
|
128
|
-
const exp: Expression = this.Exp();
|
129
|
-
|
130
|
-
// 如果是属性赋值
|
131
|
-
if (objExp.type === ExpressionType.Property) {
|
132
|
-
// 从属性Expression中获取对象及属性名
|
133
|
-
const name: string = <string>objExp.value;
|
134
|
-
objExp = objExp.children[0] as Expression;
|
135
|
-
return Expression.Assign(objExp, exp, name, this.pos);
|
136
|
-
} else {
|
137
|
-
// Identity 变量赋值
|
138
|
-
return Expression.Assign(null, exp, <string>objExp.value, this.pos);
|
139
|
-
}
|
140
|
-
}
|
141
|
-
|
142
|
-
private Exp(): Expression {
|
143
|
-
let v: Expression = this.Logic();
|
144
|
-
|
145
|
-
// 是':',表示条件,否则直接返回单结果
|
146
|
-
let t: Token = this.GetToken();
|
147
|
-
if (t.getType() === TokenType.Oper && t.getValue() === ":") {
|
148
|
-
// 第一项转换
|
149
|
-
let result: Expression = this.Logic();
|
150
|
-
|
151
|
-
// 下一个是",",继续读取下一个条件结果串,由于是右结合,只能采用递归
|
152
|
-
t = this.GetToken();
|
153
|
-
if (t.getType() === TokenType.Oper && t.getValue() === ",") {
|
154
|
-
// 第二项转换
|
155
|
-
let sExp: Expression = this.Exp();
|
156
|
-
// 返回
|
157
|
-
return Expression.Condition(v, result, sExp, this.SOURCE, this.pos);
|
158
|
-
} else {
|
159
|
-
throw new Error(this.GetExceptionMessage("必须有默认值!"));
|
160
|
-
}
|
161
|
-
} else {
|
162
|
-
this.tokens.push(t); // Put the token back
|
163
|
-
return v;
|
164
|
-
}
|
165
|
-
}
|
166
|
-
|
167
|
-
private Logic(): Expression {
|
168
|
-
let t: Token = this.GetToken();
|
169
|
-
|
170
|
-
// !表达式
|
171
|
-
const hasNot: boolean = t.getType() === TokenType.Oper && t.getValue() === "!";
|
172
|
-
if (!hasNot) {
|
173
|
-
this.tokens.push(t); // Put the token back if it's not '!'
|
174
|
-
}
|
175
|
-
|
176
|
-
let v: Expression = this.Compare();
|
177
|
-
if (hasNot) {
|
178
|
-
v = Expression.Not(v, this.SOURCE, this.pos);
|
179
|
-
}
|
180
|
-
|
181
|
-
t = this.GetToken();
|
182
|
-
while (t.getType() === TokenType.Oper && (t.getValue() === "&&" || t.getValue() === "||")) {
|
183
|
-
// 第二项转换
|
184
|
-
let exp: Expression = this.Logic();
|
185
|
-
|
186
|
-
// 执行
|
187
|
-
if (t.getValue() === "&&") {
|
188
|
-
v = Expression.And(v, exp, this.SOURCE, this.pos);
|
189
|
-
} else {
|
190
|
-
v = Expression.Or(v, exp, this.SOURCE, this.pos);
|
191
|
-
}
|
192
|
-
|
193
|
-
t = this.GetToken();
|
194
|
-
}
|
195
|
-
|
196
|
-
this.tokens.push(t); // Put the token back
|
197
|
-
return v;
|
198
|
-
}
|
199
|
-
|
200
|
-
private Compare(): Expression {
|
201
|
-
let left: Expression = this.Math();
|
202
|
-
let t: Token = this.GetToken();
|
203
|
-
|
204
|
-
if (t.getType() === TokenType.Oper && (
|
205
|
-
t.getValue() === ">" || t.getValue() === ">=" ||
|
206
|
-
t.getValue() === "<" || t.getValue() === "<=" || t.getValue() === "=>")) {
|
207
|
-
|
208
|
-
if (t.getValue() === "=>") {
|
209
|
-
this.tokens.push(t); // Push back the token for lambda handling
|
210
|
-
return this.Lambda();
|
211
|
-
}
|
212
|
-
|
213
|
-
let rExp: Expression = this.Math();
|
214
|
-
|
215
|
-
switch (t.getValue()) {
|
216
|
-
case ">":
|
217
|
-
return Expression.GreaterThan(left, rExp, this.SOURCE, this.pos);
|
218
|
-
case ">=":
|
219
|
-
return Expression.GreaterThanOrEqual(left, rExp, this.SOURCE, this.pos);
|
220
|
-
case "<":
|
221
|
-
return Expression.LessThan(left, rExp, this.SOURCE, this.pos);
|
222
|
-
case "<=":
|
223
|
-
return Expression.LessThanOrEqual(left, rExp, this.SOURCE, this.pos);
|
224
|
-
}
|
225
|
-
} else if (t.getType() === TokenType.Oper && (
|
226
|
-
t.getValue() === "==" || t.getValue() === "!=")) {
|
227
|
-
|
228
|
-
let rExp: Expression = this.Math();
|
229
|
-
|
230
|
-
// 相等比较
|
231
|
-
if (t.getValue() === "==") {
|
232
|
-
return Expression.Equal(left, rExp, this.SOURCE, this.pos);
|
233
|
-
} else {
|
234
|
-
return Expression.NotEqual(left, rExp, this.SOURCE, this.pos);
|
235
|
-
}
|
236
|
-
}
|
237
|
-
|
238
|
-
// 返回当前的表达式结果
|
239
|
-
this.tokens.push(t); // Push the token back
|
240
|
-
return left;
|
241
|
-
}
|
242
|
-
|
243
|
-
private Math(): Expression {
|
244
|
-
let v: Expression = this.Mul();
|
245
|
-
let t: Token = this.GetToken();
|
246
|
-
|
247
|
-
while (t.getType() === TokenType.Oper && (t.getValue() === "+" || t.getValue() === "-")) {
|
248
|
-
// 转换操作数2为数字
|
249
|
-
let r: Expression = this.Mul();
|
250
|
-
|
251
|
-
// 开始运算
|
252
|
-
if (t.getValue() === "+") {
|
253
|
-
v = Expression.Add(v, r, this.SOURCE, this.pos);
|
254
|
-
} else {
|
255
|
-
v = Expression.Subtract(v, r, this.SOURCE, this.pos);
|
256
|
-
}
|
257
|
-
|
258
|
-
t = this.GetToken();
|
259
|
-
}
|
260
|
-
|
261
|
-
this.tokens.push(t); // Push the token back
|
262
|
-
return v;
|
263
|
-
}
|
264
|
-
|
265
|
-
private Lambda(): Expression {
|
266
|
-
let t: Token = this.GetToken();
|
267
|
-
if (t.getValue() !== "=>" || t.getValue() == null) {
|
268
|
-
throw new Error(this.GetExceptionMessage("lambda必须以=>开始"));
|
269
|
-
}
|
270
|
-
|
271
|
-
t = this.GetToken();
|
272
|
-
if (t.getValue() !== "{" || t.getValue() == null) {
|
273
|
-
throw new Error(this.GetExceptionMessage("lambda必须以{开始"));
|
274
|
-
}
|
275
|
-
|
276
|
-
// 取lambda中的表达式
|
277
|
-
const exp: Expression = this.CommaExp();
|
278
|
-
|
279
|
-
t = this.GetToken();
|
280
|
-
if (t.getValue() !== "}" || t.getValue() == null) {
|
281
|
-
throw new Error(this.GetExceptionMessage("lambda必须以}结束"));
|
282
|
-
}
|
283
|
-
|
284
|
-
return Expression.Lambda(exp, this.pos);
|
285
|
-
}
|
286
|
-
|
287
|
-
private Mul(): Expression {
|
288
|
-
let v: Expression = this.UnarySub();
|
289
|
-
let t: Token = this.GetToken();
|
290
|
-
|
291
|
-
while (t.getType() === TokenType.Oper && (t.getValue() === "*" || t.getValue() === "/" || t.getValue() === "%")) {
|
292
|
-
// Convert the second operand to a number
|
293
|
-
const r: Expression = this.UnarySub();
|
294
|
-
|
295
|
-
// Perform the operation
|
296
|
-
if (t.getValue() === "*") {
|
297
|
-
v = Expression.Multiply(v, r, this.SOURCE, this.pos);
|
298
|
-
} else if (t.getValue() === "/") {
|
299
|
-
v = Expression.Divide(v, r, this.SOURCE, this.pos);
|
300
|
-
} else {
|
301
|
-
v = Expression.Modulo(v, r, this.SOURCE, this.pos);
|
302
|
-
}
|
303
|
-
|
304
|
-
t = this.GetToken();
|
305
|
-
}
|
306
|
-
|
307
|
-
this.tokens.push(t);
|
308
|
-
return v;
|
309
|
-
}
|
310
|
-
|
311
|
-
private UnarySub(): Expression {
|
312
|
-
let t: Token = this.GetToken();
|
313
|
-
if (t.getType() === TokenType.Oper && t.getValue() === "-") {
|
314
|
-
let r: Expression = this.Item();
|
315
|
-
return Expression.Subtract(Expression.Constant(0, this.pos), r, this.SOURCE, this.pos);
|
316
|
-
}
|
317
|
-
this.tokens.push(t); // Push the token back into the queue
|
318
|
-
return this.Item();
|
319
|
-
}
|
320
|
-
|
321
|
-
private Item(): Expression {
|
322
|
-
let objName: string = "";
|
323
|
-
// 获取对象表达式
|
324
|
-
let objExp: Expression = this.ItemHead(objName);
|
325
|
-
// 获取对象路径表达式
|
326
|
-
objExp = this.ObjectPath(objExp);
|
327
|
-
return objExp;
|
328
|
-
}
|
329
|
-
|
330
|
-
private ItemHead(name: string): Expression {
|
331
|
-
let t: Token = this.GetToken();
|
332
|
-
|
333
|
-
if (t.getType() === TokenType.Oper && t.getValue() === "(") {
|
334
|
-
let result: Expression = this.CommaExp();
|
335
|
-
t = this.GetToken();
|
336
|
-
if (t.getType() !== TokenType.Oper || t.getValue() !== ")") {
|
337
|
-
throw new Error(this.GetExceptionMessage("括号不匹配"));
|
338
|
-
}
|
339
|
-
return result;
|
340
|
-
} else if (t.getType() === TokenType.Oper && t.getValue() === "{") {
|
341
|
-
// JSON object
|
342
|
-
return this.Json();
|
343
|
-
} else if (t.getType() === TokenType.Oper && t.getValue() === "[") {
|
344
|
-
// JSON array
|
345
|
-
return this.JsonArray();
|
346
|
-
} else if (t.getType() === TokenType.Oper && (t.getValue() === "$" || t.getValue() === "\"")) {
|
347
|
-
// String concatenation sequence
|
348
|
-
return this.StringUnion();
|
349
|
-
} else if (t.getType() === TokenType.Int || t.getType() === TokenType.Double || t.getType() === TokenType.Bool) {
|
350
|
-
return Expression.Constant(t.getValue(), this.pos);
|
351
|
-
} else if (t.getType() === TokenType.Identy) {
|
352
|
-
let objName: string = <string>t.getValue();
|
353
|
-
|
354
|
-
switch (objName) {
|
355
|
-
case "return":
|
356
|
-
return Expression.Return(this.Exp(), this.pos);
|
357
|
-
case "try":
|
358
|
-
this.tokens.push(t); // Revert the token back to the queue
|
359
|
-
return this.Try();
|
360
|
-
case "throw":
|
361
|
-
return Expression.Throw(this.Exp(), this.pos);
|
362
|
-
case "validate":
|
363
|
-
return this.Validate();
|
364
|
-
case "assert":
|
365
|
-
return Expression.Assert(this.Exp(), this.pos);
|
366
|
-
case "break":
|
367
|
-
return Expression.Break(this.pos, undefined);
|
368
|
-
case "continue":
|
369
|
-
return Expression.Continue(this.pos, undefined);
|
370
|
-
}
|
371
|
-
|
372
|
-
// Return the object name
|
373
|
-
name += objName;
|
374
|
-
// Process the object name
|
375
|
-
return this.ObjectName(objName);
|
376
|
-
} else if (t.getType() === TokenType.Null) {
|
377
|
-
return Expression.Constant(null, this.pos);
|
378
|
-
}
|
379
|
-
|
380
|
-
throw new Error(this.GetExceptionMessage("单词类型错误:" + t.getType()));
|
381
|
-
}
|
382
|
-
|
383
|
-
/**
|
384
|
-
* TryCatch := try {expression} (catch(Exception类 名){表达式})* finally{表达式}?
|
385
|
-
*/
|
386
|
-
private Try(): Expression {
|
387
|
-
let t: Token = this.GetToken();
|
388
|
-
if (t.getValue() !== "try") {
|
389
|
-
throw new Error(this.GetExceptionMessage("try-catch必须以try开始"));
|
390
|
-
}
|
391
|
-
|
392
|
-
t = this.GetToken();
|
393
|
-
if (t.getValue() !== "{") {
|
394
|
-
throw new Error(this.GetExceptionMessage("try块必须以{开始"));
|
395
|
-
}
|
396
|
-
|
397
|
-
// Get the expression inside the try block
|
398
|
-
let tryExp: Expression = this.CommaExp();
|
399
|
-
|
400
|
-
t = this.GetToken();
|
401
|
-
if (t.getValue() !== "}") {
|
402
|
-
throw new Error(this.GetExceptionMessage("try块必须以}结束"));
|
403
|
-
}
|
404
|
-
|
405
|
-
// Handle the catch part
|
406
|
-
let catches: Expression[] = this.Catch();
|
407
|
-
|
408
|
-
// Handle finally
|
409
|
-
let finallyExp: Expression | undefined = undefined;
|
410
|
-
t = this.GetToken();
|
411
|
-
if (t.getValue()) {
|
412
|
-
if (t.getValue() === "finally") {
|
413
|
-
t = this.GetToken();
|
414
|
-
if (t.getValue() !== "{") {
|
415
|
-
throw new Error(this.GetExceptionMessage("finally块必须以{开始"));
|
416
|
-
}
|
417
|
-
|
418
|
-
// Get the expression inside the finally block
|
419
|
-
finallyExp = this.CommaExp();
|
420
|
-
t = this.GetToken();
|
421
|
-
if (t.getValue() !== "}") {
|
422
|
-
throw new Error(this.GetExceptionMessage("finally块必须以}结束"));
|
423
|
-
}
|
424
|
-
t = this.GetToken();
|
425
|
-
}
|
426
|
-
}
|
427
|
-
|
428
|
-
this.tokens.push(t);
|
429
|
-
|
430
|
-
// Return the exception handling expression
|
431
|
-
return Expression.Try(tryExp, catches, this.pos, finallyExp);
|
432
|
-
}
|
433
|
-
|
434
|
-
/**
|
435
|
-
* catch部分处理
|
436
|
-
*/
|
437
|
-
private Catch(): Expression[] {
|
438
|
-
let result: Expression[] = [];
|
439
|
-
let t: Token = this.GetToken();
|
440
|
-
|
441
|
-
while (t.getValue() === "catch") {
|
442
|
-
t = this.GetToken();
|
443
|
-
if (t.getValue() !== "(") {
|
444
|
-
throw new Error(this.GetExceptionMessage("catch块参数必须以(开始"));
|
445
|
-
}
|
446
|
-
|
447
|
-
// Get the class name
|
448
|
-
t = this.GetToken();
|
449
|
-
let className: string = <string>t.getValue();
|
450
|
-
|
451
|
-
// Get the variable name
|
452
|
-
t = this.GetToken();
|
453
|
-
let varName: string = <string>t.getValue();
|
454
|
-
|
455
|
-
t = this.GetToken();
|
456
|
-
if (t.getValue() !== ")") {
|
457
|
-
throw new Error(this.GetExceptionMessage("catch块参数必须以)结束"));
|
458
|
-
}
|
459
|
-
|
460
|
-
// Get the opening curly brace
|
461
|
-
t = this.GetToken();
|
462
|
-
if (t.getValue() !== "{") {
|
463
|
-
throw new Error(this.GetExceptionMessage("catch块必须以{开始"));
|
464
|
-
}
|
465
|
-
|
466
|
-
// Get the expression inside the catch block
|
467
|
-
let catchExp: Expression = this.CommaExp();
|
468
|
-
result.push(Expression.Catch(className, varName, catchExp, this.pos));
|
469
|
-
|
470
|
-
// Get the closing curly brace
|
471
|
-
t = this.GetToken();
|
472
|
-
if (t.getValue() !== "}") {
|
473
|
-
throw new Error(this.GetExceptionMessage("catch块必须以}结束"));
|
474
|
-
}
|
475
|
-
|
476
|
-
// Get the next token
|
477
|
-
t = this.GetToken();
|
478
|
-
}
|
479
|
-
|
480
|
-
this.tokens.push(t);
|
481
|
-
return result;
|
482
|
-
}
|
483
|
-
|
484
|
-
/**
|
485
|
-
* Validate block handling
|
486
|
-
*/
|
487
|
-
private Validate(): Expression {
|
488
|
-
let t: Token = this.GetToken();
|
489
|
-
if (t.getValue() !== "{") {
|
490
|
-
throw new Error(this.GetExceptionMessage("validate只接收Json对象"));
|
491
|
-
}
|
492
|
-
|
493
|
-
// Get the expression in the validate block
|
494
|
-
let exp: Expression = this.Json();
|
495
|
-
|
496
|
-
// Return Validate processing expression
|
497
|
-
return Expression.Validate(exp, this.pos);
|
498
|
-
}
|
499
|
-
|
500
|
-
/**
|
501
|
-
* JSON Object ::= {} | {propertyName: propertyValue, propertyName: propertyValue}
|
502
|
-
*/
|
503
|
-
private Json(): Expression {
|
504
|
-
let children: Expression[] = [];
|
505
|
-
|
506
|
-
let t: Token = this.GetToken();
|
507
|
-
// Empty object, return directly
|
508
|
-
if (t.getType() === TokenType.Oper && t.getValue() === "}") {
|
509
|
-
return Expression.Json(children, this.pos);
|
510
|
-
}
|
511
|
-
|
512
|
-
this.tokens.push(t);
|
513
|
-
children.push(this.Attr());
|
514
|
-
t = this.GetToken();
|
515
|
-
|
516
|
-
// If it's a comma, continue checking the next property
|
517
|
-
while (t.getType() === TokenType.Oper && t.getValue() === ",") {
|
518
|
-
let nextT: Token = this.GetToken();
|
519
|
-
if (nextT.getType() === TokenType.Oper && nextT.getValue() === "}") {
|
520
|
-
t = nextT;
|
521
|
-
break;
|
522
|
-
}
|
523
|
-
this.tokens.push(nextT);
|
524
|
-
children.push(this.Attr());
|
525
|
-
t = this.GetToken();
|
526
|
-
}
|
527
|
-
|
528
|
-
if (t.getType() !== TokenType.Oper || t.getValue() !== "}") {
|
529
|
-
throw new Error(this.GetExceptionMessage("JSON对象构建错误"));
|
530
|
-
}
|
531
|
-
|
532
|
-
return Expression.Json(children, this.pos);
|
533
|
-
}
|
534
|
-
|
535
|
-
/**
|
536
|
-
* JSON Array ::= []
|
537
|
-
*/
|
538
|
-
private JsonArray(): Expression {
|
539
|
-
let t: Token = this.GetToken();
|
540
|
-
// Empty array, return directly
|
541
|
-
if (t.getType() === TokenType.Oper && t.getValue() === "]") {
|
542
|
-
// Return JSON array constant
|
543
|
-
return Expression.Constant(new JSONArray(), this.pos);
|
544
|
-
}
|
545
|
-
|
546
|
-
// Loop to get array contents
|
547
|
-
this.tokens.push(t);
|
548
|
-
let ps: Expression[] = this.Params();
|
549
|
-
t = this.GetToken();
|
550
|
-
if (t.getType() !== TokenType.Oper || t.getValue() !== "]") {
|
551
|
-
throw new Error(this.GetExceptionMessage("JSON集合构建错误"));
|
552
|
-
}
|
553
|
-
|
554
|
-
return Expression.Array(ps, this.pos);
|
555
|
-
}
|
556
|
-
|
557
|
-
/**
|
558
|
-
* Attribute-Value Pair ::= propertyName: propertyValue
|
559
|
-
*/
|
560
|
-
private Attr(): Expression {
|
561
|
-
let name: Token = this.GetToken();
|
562
|
-
let value: string;
|
563
|
-
if (name.getType() === TokenType.Identy) {
|
564
|
-
value = <string>name.getValue();
|
565
|
-
} else if (name.getType() === TokenType.Oper && (name.getValue() === "$" || name.getValue() === "\"")) {
|
566
|
-
let expression: Expression = this.StringUnion().children[1] as Expression;
|
567
|
-
value = <string>expression.value;
|
568
|
-
} else {
|
569
|
-
throw new Error(this.GetExceptionMessage("JSON对象的key必须是属性名或字符串"));
|
570
|
-
}
|
571
|
-
|
572
|
-
let t: Token = this.GetToken();
|
573
|
-
if (t.getType() !== TokenType.Oper || t.getValue() !== ":") {
|
574
|
-
throw new Error(this.GetExceptionMessage("JSON构建错误"));
|
575
|
-
}
|
576
|
-
|
577
|
-
let exp: Expression = this.Exp();
|
578
|
-
return Expression.Attr(value, exp, this.pos);
|
579
|
-
}
|
580
|
-
|
581
|
-
/**
|
582
|
-
* Parse string concatenation sequence
|
583
|
-
*/
|
584
|
-
private StringUnion(): Expression {
|
585
|
-
let exp: Expression = Expression.Constant("", this.pos);
|
586
|
-
let t: Token = this.GetToken();
|
587
|
-
// Object sequence
|
588
|
-
while ((t.getType() === TokenType.Oper && t.getValue() === "{") || t.getType() === TokenType.String) {
|
589
|
-
// If it's a string, return the concatenated result
|
590
|
-
if (t.getType() === TokenType.String) {
|
591
|
-
exp = Expression.Concat(exp, Expression.Constant(t.getValue(), this.pos), this.SOURCE, this.pos);
|
592
|
-
} else {
|
593
|
-
// Handle content inside the {}
|
594
|
-
let objExp: Expression = this.Exp();
|
595
|
-
t = this.GetToken();
|
596
|
-
if (t.getType() !== TokenType.Oper || t.getValue() !== "}") {
|
597
|
-
throw new Error(this.GetExceptionMessage("缺少'}'"));
|
598
|
-
}
|
599
|
-
// String concatenation
|
600
|
-
exp = Expression.Concat(exp, objExp, this.SOURCE, this.pos);
|
601
|
-
}
|
602
|
-
t = this.GetToken();
|
603
|
-
}
|
604
|
-
this.tokens.push(t);
|
605
|
-
return exp;
|
606
|
-
}
|
607
|
-
|
608
|
-
/**
|
609
|
-
* Get object and object expression by name
|
610
|
-
*/
|
611
|
-
private ObjectName(objName: string): Expression {
|
612
|
-
return Expression.Identity(objName, this.pos);
|
613
|
-
}
|
614
|
-
|
615
|
-
/**
|
616
|
-
* Object path parsing, handles method calls, properties, and array indices
|
617
|
-
*/
|
618
|
-
private ObjectPath(inExp: Expression): Expression {
|
619
|
-
let objExp: Expression = this.ArrayIndex(inExp);
|
620
|
-
let t: Token = this.GetToken();
|
621
|
-
while (t.getType() === TokenType.Oper && t.getValue() === ".") {
|
622
|
-
let nameToken: Token = this.GetToken();
|
623
|
-
let n: Token = this.GetToken();
|
624
|
-
if (n.getType() === TokenType.Oper && n.getValue() === "(") {
|
625
|
-
let name: string = <string>nameToken.getValue();
|
626
|
-
if (name === "each") {
|
627
|
-
objExp = this.For(objExp);
|
628
|
-
} else {
|
629
|
-
objExp = this.MethodCall(name, objExp);
|
630
|
-
}
|
631
|
-
} else {
|
632
|
-
this.tokens.push(n);
|
633
|
-
let pi: string = <string>nameToken.getValue();
|
634
|
-
objExp = Expression.Property(objExp, pi, this.pos);
|
635
|
-
objExp = this.ArrayIndex(objExp);
|
636
|
-
}
|
637
|
-
t = this.GetToken();
|
638
|
-
}
|
639
|
-
this.tokens.push(t);
|
640
|
-
|
641
|
-
return objExp;
|
642
|
-
}
|
643
|
-
|
644
|
-
/**
|
645
|
-
* Array index parsing with optional condition
|
646
|
-
*/
|
647
|
-
private ArrayIndex(objExp: Expression): Expression {
|
648
|
-
let n: Token = this.GetToken();
|
649
|
-
if (n.getType() === TokenType.Oper && n.getValue() === "[") {
|
650
|
-
let exp: Expression = this.Exp();
|
651
|
-
n = this.GetToken();
|
652
|
-
if (n.getType() !== TokenType.Oper || n.getValue() !== "]") {
|
653
|
-
throw new Error(this.GetExceptionMessage("缺少']'"));
|
654
|
-
}
|
655
|
-
return Expression.ArrayIndex(objExp, exp, this.SOURCE, this.pos);
|
656
|
-
}
|
657
|
-
this.tokens.push(n);
|
658
|
-
return objExp;
|
659
|
-
}
|
660
|
-
|
661
|
-
/**
|
662
|
-
* For loop handling
|
663
|
-
*/
|
664
|
-
private For(obj: Expression): Expression {
|
665
|
-
let exp: Expression = this.CommaExp();
|
666
|
-
let t: Token = this.GetToken();
|
667
|
-
if (t.getType() !== TokenType.Oper || t.getValue() !== ")") {
|
668
|
-
throw new Error(this.GetExceptionMessage("函数调用括号不匹配"));
|
669
|
-
}
|
670
|
-
return Expression.For(obj, exp, this.SOURCE, this.pos);
|
671
|
-
}
|
672
|
-
|
673
|
-
/**
|
674
|
-
* Method call handling with parameters
|
675
|
-
*/
|
676
|
-
private MethodCall(name: string, obj: Expression): Expression {
|
677
|
-
let t: Token = this.GetToken();
|
678
|
-
if (t.getType() === TokenType.Oper && t.getValue() === ")") {
|
679
|
-
let ps: Expression[] = [];
|
680
|
-
return Expression.Call(obj, name, ps, this.pos);
|
681
|
-
}
|
682
|
-
|
683
|
-
this.tokens.push(t);
|
684
|
-
let ps: Expression[] = this.Params();
|
685
|
-
t = this.GetToken();
|
686
|
-
if (t.getType() !== TokenType.Oper || t.getValue() !== ")") {
|
687
|
-
throw new Error(this.GetExceptionMessage("函数调用括号不匹配"));
|
688
|
-
}
|
689
|
-
|
690
|
-
return Expression.Call(obj, name, ps, this.pos);
|
691
|
-
}
|
692
|
-
|
693
|
-
/**
|
694
|
-
* Function parameter list handling
|
695
|
-
*/
|
696
|
-
private Params(): Expression[] {
|
697
|
-
let ps: Expression[] = [];
|
698
|
-
let name: string = this.paramName();
|
699
|
-
let exp: Expression = this.Exp();
|
700
|
-
this.procParam(ps, exp, name);
|
701
|
-
let t: Token = this.GetToken();
|
702
|
-
while (t.getType() === TokenType.Oper && t.getValue() === ",") {
|
703
|
-
name = this.paramName();
|
704
|
-
exp = this.Exp();
|
705
|
-
this.procParam(ps, exp, name);
|
706
|
-
t = this.GetToken();
|
707
|
-
}
|
708
|
-
this.tokens.push(t);
|
709
|
-
return ps;
|
710
|
-
}
|
711
|
-
|
712
|
-
/**
|
713
|
-
* Parse parameter name
|
714
|
-
*/
|
715
|
-
private paramName(): string {
|
716
|
-
let t: Token = this.GetToken();
|
717
|
-
if (t.getType() === TokenType.Identy) {
|
718
|
-
let t1: Token = this.GetToken();
|
719
|
-
if (t1.getValue() === ":") {
|
720
|
-
return <string>t.getValue();
|
721
|
-
} else {
|
722
|
-
this.tokens.push(t);
|
723
|
-
this.tokens.push(t1);
|
724
|
-
return "";
|
725
|
-
}
|
726
|
-
} else {
|
727
|
-
this.tokens.push(t);
|
728
|
-
return "";
|
729
|
-
}
|
730
|
-
}
|
731
|
-
|
732
|
-
/**
|
733
|
-
* Process parameters, especially if it contains the "data" object
|
734
|
-
*/
|
735
|
-
private procParam(ps: Expression[], exp: Expression, name: string): void {
|
736
|
-
let delegate: Delegate = this.createDelegate(exp);
|
737
|
-
if (delegate.containsKey("data")) {
|
738
|
-
ps.push(Expression.Constant(delegate, this.pos));
|
739
|
-
} else {
|
740
|
-
ps.push(Expression.Param(exp, name, this.pos));
|
741
|
-
}
|
742
|
-
}
|
743
|
-
|
744
|
-
/**
|
745
|
-
* Get exception message with position information
|
746
|
-
*/
|
747
|
-
private GetExceptionMessage(msg: string): string {
|
748
|
-
let result: string = this.SOURCE.substring(0, this.pos) + " <- " + this.SOURCE.substring(this.pos);
|
749
|
-
return `${msg}:\n${result}`;
|
750
|
-
}
|
751
|
-
|
752
|
-
public GetToken(): Token {
|
753
|
-
// If there are tokens in the queue, return the last one saved
|
754
|
-
if (this.tokens.length > 0) {
|
755
|
-
return this.tokens.shift()!;
|
756
|
-
}
|
757
|
-
|
758
|
-
const sPos = this.pos;
|
759
|
-
let hasKeyword = this.pos < this.SOURCE_LENGTH;
|
760
|
-
let value: string | null = null;
|
761
|
-
|
762
|
-
if (hasKeyword) {
|
763
|
-
value = this.SOURCE.charAt(this.pos);
|
764
|
-
}
|
765
|
-
|
766
|
-
// If it's '{', save the previous state and change to non-string state
|
767
|
-
if (hasKeyword && value === '{') {
|
768
|
-
this.inStrings.push(this.inString);
|
769
|
-
this.inString = false;
|
770
|
-
const str = this.SOURCE.substring(this.pos, this.pos + 1);
|
771
|
-
this.pos += 1;
|
772
|
-
return new Token(TokenType.Oper, str, sPos);
|
773
|
-
}
|
774
|
-
|
775
|
-
// If in string state, process the string
|
776
|
-
if (this.inString) {
|
777
|
-
let startPos = this.pos;
|
778
|
-
while (hasKeyword && value !== '{' && value !== '}' && value !== '$' && value !== '\"') {
|
779
|
-
if (value === '\\') {
|
780
|
-
const nextStrValue = this.SOURCE.charAt(this.pos + 1);
|
781
|
-
if (this.pos + 1 < this.SOURCE_LENGTH && ['{', '}', '\"', '$', '\\'].includes(nextStrValue)) {
|
782
|
-
this.pos++;
|
783
|
-
} else {
|
784
|
-
break;
|
785
|
-
}
|
786
|
-
}
|
787
|
-
this.pos++;
|
788
|
-
if (this.pos < this.SOURCE_LENGTH) {
|
789
|
-
value = this.SOURCE.charAt(this.pos);
|
790
|
-
} else {
|
791
|
-
hasKeyword = false;
|
792
|
-
}
|
793
|
-
}
|
794
|
-
if (!(hasKeyword && value === '{')) {
|
795
|
-
this.inString = this.inStrings.pop()!;
|
796
|
-
}
|
797
|
-
const t = new Token(TokenType.String, this.SOURCE.substring(startPos, this.pos)
|
798
|
-
.replace("\\$", "$")
|
799
|
-
.replace("\\{", "{")
|
800
|
-
.replace("\\}", "}")
|
801
|
-
.replace(/\\"/g, '"')
|
802
|
-
.replace("\\\\", "\\"), sPos);
|
803
|
-
|
804
|
-
if (hasKeyword && (value === '$' || value === '\"')) {
|
805
|
-
this.pos++;
|
806
|
-
}
|
807
|
-
return t;
|
808
|
-
}
|
809
|
-
|
810
|
-
// Skip whitespace and comments
|
811
|
-
while (hasKeyword && (value === ' ' || value === '\n' || value === '\t' || (this.pos < this.SOURCE_LENGTH - 2 && value === '/' && this.SOURCE.charAt(this.pos + 1) === '/'))) {
|
812
|
-
if (value !== ' ' && value !== '\n' && value !== '\t') {
|
813
|
-
this.pos += 2;
|
814
|
-
if (this.pos < this.SOURCE_LENGTH) {
|
815
|
-
value = this.SOURCE.charAt(this.pos);
|
816
|
-
} else {
|
817
|
-
hasKeyword = false;
|
818
|
-
}
|
819
|
-
while (hasKeyword && value !== '\n') {
|
820
|
-
this.pos++;
|
821
|
-
if (this.pos < this.SOURCE_LENGTH) {
|
822
|
-
value = this.SOURCE.charAt(this.pos);
|
823
|
-
} else {
|
824
|
-
hasKeyword = false;
|
825
|
-
}
|
826
|
-
}
|
827
|
-
}
|
828
|
-
this.pos++;
|
829
|
-
if (this.pos < this.SOURCE_LENGTH) {
|
830
|
-
value = this.SOURCE.charAt(this.pos);
|
831
|
-
} else {
|
832
|
-
hasKeyword = false;
|
833
|
-
}
|
834
|
-
}
|
835
|
-
|
836
|
-
// If we are at the end of the source, return an "End" token
|
837
|
-
if (this.pos === this.SOURCE_LENGTH) {
|
838
|
-
return new Token(TokenType.End, null, sPos);
|
839
|
-
}
|
840
|
-
|
841
|
-
if (value !== null) {
|
842
|
-
// Handle number tokens
|
843
|
-
if (value >= '0' && value <= '9') {
|
844
|
-
const oldPos = this.pos;
|
845
|
-
while (hasKeyword && value >= '0' && value <= '9') {
|
846
|
-
this.pos++;
|
847
|
-
if (this.pos < this.SOURCE_LENGTH) {
|
848
|
-
value = this.SOURCE.charAt(this.pos);
|
849
|
-
} else {
|
850
|
-
hasKeyword = false;
|
851
|
-
}
|
852
|
-
}
|
853
|
-
|
854
|
-
if (hasKeyword && value === '.') {
|
855
|
-
do {
|
856
|
-
this.pos++;
|
857
|
-
if (this.pos < this.SOURCE_LENGTH) {
|
858
|
-
value = this.SOURCE.charAt(this.pos);
|
859
|
-
} else {
|
860
|
-
hasKeyword = false;
|
861
|
-
}
|
862
|
-
} while (hasKeyword && value >= '0' && value <= '9');
|
863
|
-
const str = this.SOURCE.substring(oldPos, this.pos);
|
864
|
-
return new Token(TokenType.Double, parseFloat(str), sPos);
|
865
|
-
} else {
|
866
|
-
const str = this.SOURCE.substring(oldPos, this.pos);
|
867
|
-
return new Token(TokenType.Int, parseInt(str, 10), sPos);
|
868
|
-
}
|
869
|
-
}
|
870
|
-
|
871
|
-
// Handle identifier tokens
|
872
|
-
else if ((value >= 'a' && value <= 'z') || (value >= 'A' && value <= 'Z') || value === '_' || /[\u4e00-\u9fa5]/.test(value)) {
|
873
|
-
const oldPos = this.pos;
|
874
|
-
while (hasKeyword && ((value >= 'a' && value <= 'z') || (value >= 'A' && value <= 'Z') || (value >= '0' && value <= '9') || value === '_' || /[\u4e00-\u9fa5]/.test(value))) {
|
875
|
-
this.pos++;
|
876
|
-
if (this.pos < this.SOURCE_LENGTH) {
|
877
|
-
value = this.SOURCE.charAt(this.pos);
|
878
|
-
} else {
|
879
|
-
hasKeyword = false;
|
880
|
-
}
|
881
|
-
}
|
882
|
-
const str = this.SOURCE.substring(oldPos, this.pos);
|
883
|
-
switch (str) {
|
884
|
-
case 'false':
|
885
|
-
case 'true':
|
886
|
-
return new Token(TokenType.Bool, Boolean(str), sPos);
|
887
|
-
case 'null':
|
888
|
-
return new Token(TokenType.Null, null, sPos);
|
889
|
-
default:
|
890
|
-
return new Token(TokenType.Identy, str, sPos);
|
891
|
-
}
|
892
|
-
}
|
893
|
-
|
894
|
-
// Handle operators like &&, ||, and single-character operators
|
895
|
-
else if ((value === '&' && this.SOURCE.charAt(this.pos + 1) === '&') || (value === '|' && this.SOURCE.charAt(this.pos + 1) === '|')) {
|
896
|
-
const str = this.SOURCE.substring(this.pos, this.pos + 2);
|
897
|
-
this.pos += 2;
|
898
|
-
return new Token(TokenType.Oper, str, sPos);
|
899
|
-
}
|
900
|
-
|
901
|
-
// Handle other operators (+, -, *, /, etc.)
|
902
|
-
else if (['+', '-', '*', '/', '%', '>', '<', '!'].includes(value)) {
|
903
|
-
let str = '';
|
904
|
-
if (this.SOURCE.charAt(this.pos + 1) === '=') {
|
905
|
-
str = this.SOURCE.substring(this.pos, this.pos + 2);
|
906
|
-
this.pos += 2;
|
907
|
-
} else {
|
908
|
-
str = this.SOURCE.substring(this.pos, this.pos + 1);
|
909
|
-
this.pos += 1;
|
910
|
-
}
|
911
|
-
return new Token(TokenType.Oper, str, sPos);
|
912
|
-
}
|
913
|
-
|
914
|
-
// Handle '=' with different cases
|
915
|
-
else if (value === '=') {
|
916
|
-
let str = '';
|
917
|
-
if (this.SOURCE.charAt(this.pos + 1) === '=' || this.SOURCE.charAt(this.pos + 1) === '>') {
|
918
|
-
str = this.SOURCE.substring(this.pos, this.pos + 2);
|
919
|
-
this.pos += 2;
|
920
|
-
} else {
|
921
|
-
str = this.SOURCE.substring(this.pos, this.pos + 1);
|
922
|
-
this.pos += 1;
|
923
|
-
}
|
924
|
-
return new Token(TokenType.Oper, str, sPos);
|
925
|
-
}
|
926
|
-
|
927
|
-
// Single character operators like parentheses, commas, etc.
|
928
|
-
else if ('()[],;.:@{}"$'.includes(value)) {
|
929
|
-
if (value === '$' || value === '\"') {
|
930
|
-
this.inStrings.push(false);
|
931
|
-
this.inString = true;
|
932
|
-
} else if ((value === '{' || value === '}') && this.inStrings.length) {
|
933
|
-
if (value === '{') {
|
934
|
-
this.inStrings.push(false);
|
935
|
-
this.inString = false;
|
936
|
-
} else {
|
937
|
-
this.inString = this.inStrings.pop()!;
|
938
|
-
}
|
939
|
-
}
|
940
|
-
const str = this.SOURCE.substring(this.pos, this.pos + 1);
|
941
|
-
this.pos += 1;
|
942
|
-
return new Token(TokenType.Oper, str, sPos);
|
943
|
-
} else {
|
944
|
-
throw new Error(`Invalid token: '${value}'`);
|
945
|
-
}
|
946
|
-
}
|
947
|
-
|
948
|
-
throw new Error('Unknown error while parsing.');
|
949
|
-
}
|
950
|
-
}
|
1
|
+
import Token from './Token'
|
2
|
+
import Expression from './Expression'
|
3
|
+
import Delegate from "./Delegate"
|
4
|
+
import ExpressionType from "../enums/ExpressionType"
|
5
|
+
import TokenType from "../enums/TokenType"
|
6
|
+
import JSONArray from "../instances/JSONArray"
|
7
|
+
|
8
|
+
/**
|
9
|
+
* 表达式入口
|
10
|
+
*/
|
11
|
+
export default class Program {
|
12
|
+
// 源程序
|
13
|
+
public readonly SOURCE : string;
|
14
|
+
public readonly SOURCE_LENGTH : number;
|
15
|
+
/**
|
16
|
+
* Token队列,用于回退
|
17
|
+
*/
|
18
|
+
private readonly tokens: Token[] = [];
|
19
|
+
/**
|
20
|
+
* 字符串处理环节堆栈,$进入字符串处理环节,“{}”中间部分脱离字符串处理环节
|
21
|
+
*/
|
22
|
+
private readonly inStrings: boolean[] = [];
|
23
|
+
|
24
|
+
/**
|
25
|
+
* 当前获取到的字符位置
|
26
|
+
*/
|
27
|
+
private pos: number = 0;
|
28
|
+
|
29
|
+
/**
|
30
|
+
* 当前是否在字符串处理环节
|
31
|
+
*/
|
32
|
+
private inString: boolean = false;
|
33
|
+
|
34
|
+
constructor(source: string) {
|
35
|
+
this.SOURCE = source;
|
36
|
+
this.SOURCE_LENGTH = this.SOURCE.length;
|
37
|
+
}
|
38
|
+
|
39
|
+
/**
|
40
|
+
* 调用解析过程
|
41
|
+
*/
|
42
|
+
public parse(): Delegate {
|
43
|
+
// 执行完以下方法后就已经诞生了一个完整的执行流程,也就是被翻译成了一套完整的指令集。
|
44
|
+
return this.createDelegate(this.expression());
|
45
|
+
}
|
46
|
+
|
47
|
+
public parseExpression(expression: Expression): Delegate {
|
48
|
+
// 执行完以下方法后就已经诞生了一个完整的执行流程,也就是被翻译成了一套完整的指令集。
|
49
|
+
return this.createDelegate(expression);
|
50
|
+
}
|
51
|
+
|
52
|
+
/**
|
53
|
+
* 编译,产生可执行单元Delegate
|
54
|
+
*/
|
55
|
+
public createDelegate(expression: Expression): Delegate {
|
56
|
+
return new Delegate(expression, this.SOURCE);
|
57
|
+
}
|
58
|
+
|
59
|
+
/**
|
60
|
+
* 获取完整表达式
|
61
|
+
*/
|
62
|
+
public expression(): Expression {
|
63
|
+
let expression = this.CommaExp();
|
64
|
+
if (this.pos < this.SOURCE_LENGTH) {
|
65
|
+
throw new Error(this.GetExceptionMessage("需要逗号结尾!"));
|
66
|
+
}
|
67
|
+
return expression;
|
68
|
+
}
|
69
|
+
|
70
|
+
/**
|
71
|
+
* 逗号表达式=赋值表达式(,赋值表达式)*
|
72
|
+
*/
|
73
|
+
private CommaExp(): Expression {
|
74
|
+
const exps: Expression[] = [];
|
75
|
+
|
76
|
+
const first: Expression = this.AssignExp();
|
77
|
+
exps.push(first);
|
78
|
+
|
79
|
+
let t: Token = this.GetToken();
|
80
|
+
|
81
|
+
// 没有结束
|
82
|
+
while (t.getType() === TokenType.Oper && (t.getValue() === "," || t.getValue() === ";" || t.getValue() === "\n")) {
|
83
|
+
const r: Expression = this.AssignExp();
|
84
|
+
exps.push(r);
|
85
|
+
t = this.GetToken();
|
86
|
+
}
|
87
|
+
|
88
|
+
this.tokens.push(t);
|
89
|
+
|
90
|
+
return Expression.Comma(exps, this.pos);
|
91
|
+
}
|
92
|
+
|
93
|
+
/**
|
94
|
+
* 赋值表达式=对象属性=一般表达式|一般表达式
|
95
|
+
*/
|
96
|
+
private AssignExp(): Expression {
|
97
|
+
let t: Token = this.GetToken();
|
98
|
+
|
99
|
+
// 把第一个Token的位置记录下来,方便后面回退
|
100
|
+
const firstPos: number = t.getStartPos();
|
101
|
+
|
102
|
+
if (t.getType() !== TokenType.Identy) {
|
103
|
+
this.pos = firstPos;
|
104
|
+
this.tokens.length = 0; // Clear the tokens array
|
105
|
+
this.inString = false;
|
106
|
+
return this.Exp();
|
107
|
+
}
|
108
|
+
|
109
|
+
let objExp: Expression = Expression.Identity(t.getValue(), this.pos);
|
110
|
+
objExp = this.ObjectPath(objExp);
|
111
|
+
|
112
|
+
// 只能给对象属性或者变量赋值
|
113
|
+
if (objExp.type !== ExpressionType.Property && objExp.type !== ExpressionType.Identity) {
|
114
|
+
this.pos = firstPos;
|
115
|
+
this.tokens.length = 0;
|
116
|
+
this.inString = false;
|
117
|
+
return this.Exp();
|
118
|
+
}
|
119
|
+
|
120
|
+
t = this.GetToken();
|
121
|
+
if (t.getType() !== TokenType.Oper || t.getValue() !== "=") {
|
122
|
+
this.pos = firstPos;
|
123
|
+
this.tokens.length = 0;
|
124
|
+
this.inString = false;
|
125
|
+
return this.Exp();
|
126
|
+
}
|
127
|
+
|
128
|
+
const exp: Expression = this.Exp();
|
129
|
+
|
130
|
+
// 如果是属性赋值
|
131
|
+
if (objExp.type === ExpressionType.Property) {
|
132
|
+
// 从属性Expression中获取对象及属性名
|
133
|
+
const name: string = <string>objExp.value;
|
134
|
+
objExp = objExp.children[0] as Expression;
|
135
|
+
return Expression.Assign(objExp, exp, name, this.pos);
|
136
|
+
} else {
|
137
|
+
// Identity 变量赋值
|
138
|
+
return Expression.Assign(null, exp, <string>objExp.value, this.pos);
|
139
|
+
}
|
140
|
+
}
|
141
|
+
|
142
|
+
private Exp(): Expression {
|
143
|
+
let v: Expression = this.Logic();
|
144
|
+
|
145
|
+
// 是':',表示条件,否则直接返回单结果
|
146
|
+
let t: Token = this.GetToken();
|
147
|
+
if (t.getType() === TokenType.Oper && t.getValue() === ":") {
|
148
|
+
// 第一项转换
|
149
|
+
let result: Expression = this.Logic();
|
150
|
+
|
151
|
+
// 下一个是",",继续读取下一个条件结果串,由于是右结合,只能采用递归
|
152
|
+
t = this.GetToken();
|
153
|
+
if (t.getType() === TokenType.Oper && t.getValue() === ",") {
|
154
|
+
// 第二项转换
|
155
|
+
let sExp: Expression = this.Exp();
|
156
|
+
// 返回
|
157
|
+
return Expression.Condition(v, result, sExp, this.SOURCE, this.pos);
|
158
|
+
} else {
|
159
|
+
throw new Error(this.GetExceptionMessage("必须有默认值!"));
|
160
|
+
}
|
161
|
+
} else {
|
162
|
+
this.tokens.push(t); // Put the token back
|
163
|
+
return v;
|
164
|
+
}
|
165
|
+
}
|
166
|
+
|
167
|
+
private Logic(): Expression {
|
168
|
+
let t: Token = this.GetToken();
|
169
|
+
|
170
|
+
// !表达式
|
171
|
+
const hasNot: boolean = t.getType() === TokenType.Oper && t.getValue() === "!";
|
172
|
+
if (!hasNot) {
|
173
|
+
this.tokens.push(t); // Put the token back if it's not '!'
|
174
|
+
}
|
175
|
+
|
176
|
+
let v: Expression = this.Compare();
|
177
|
+
if (hasNot) {
|
178
|
+
v = Expression.Not(v, this.SOURCE, this.pos);
|
179
|
+
}
|
180
|
+
|
181
|
+
t = this.GetToken();
|
182
|
+
while (t.getType() === TokenType.Oper && (t.getValue() === "&&" || t.getValue() === "||")) {
|
183
|
+
// 第二项转换
|
184
|
+
let exp: Expression = this.Logic();
|
185
|
+
|
186
|
+
// 执行
|
187
|
+
if (t.getValue() === "&&") {
|
188
|
+
v = Expression.And(v, exp, this.SOURCE, this.pos);
|
189
|
+
} else {
|
190
|
+
v = Expression.Or(v, exp, this.SOURCE, this.pos);
|
191
|
+
}
|
192
|
+
|
193
|
+
t = this.GetToken();
|
194
|
+
}
|
195
|
+
|
196
|
+
this.tokens.push(t); // Put the token back
|
197
|
+
return v;
|
198
|
+
}
|
199
|
+
|
200
|
+
private Compare(): Expression {
|
201
|
+
let left: Expression = this.Math();
|
202
|
+
let t: Token = this.GetToken();
|
203
|
+
|
204
|
+
if (t.getType() === TokenType.Oper && (
|
205
|
+
t.getValue() === ">" || t.getValue() === ">=" ||
|
206
|
+
t.getValue() === "<" || t.getValue() === "<=" || t.getValue() === "=>")) {
|
207
|
+
|
208
|
+
if (t.getValue() === "=>") {
|
209
|
+
this.tokens.push(t); // Push back the token for lambda handling
|
210
|
+
return this.Lambda();
|
211
|
+
}
|
212
|
+
|
213
|
+
let rExp: Expression = this.Math();
|
214
|
+
|
215
|
+
switch (t.getValue()) {
|
216
|
+
case ">":
|
217
|
+
return Expression.GreaterThan(left, rExp, this.SOURCE, this.pos);
|
218
|
+
case ">=":
|
219
|
+
return Expression.GreaterThanOrEqual(left, rExp, this.SOURCE, this.pos);
|
220
|
+
case "<":
|
221
|
+
return Expression.LessThan(left, rExp, this.SOURCE, this.pos);
|
222
|
+
case "<=":
|
223
|
+
return Expression.LessThanOrEqual(left, rExp, this.SOURCE, this.pos);
|
224
|
+
}
|
225
|
+
} else if (t.getType() === TokenType.Oper && (
|
226
|
+
t.getValue() === "==" || t.getValue() === "!=")) {
|
227
|
+
|
228
|
+
let rExp: Expression = this.Math();
|
229
|
+
|
230
|
+
// 相等比较
|
231
|
+
if (t.getValue() === "==") {
|
232
|
+
return Expression.Equal(left, rExp, this.SOURCE, this.pos);
|
233
|
+
} else {
|
234
|
+
return Expression.NotEqual(left, rExp, this.SOURCE, this.pos);
|
235
|
+
}
|
236
|
+
}
|
237
|
+
|
238
|
+
// 返回当前的表达式结果
|
239
|
+
this.tokens.push(t); // Push the token back
|
240
|
+
return left;
|
241
|
+
}
|
242
|
+
|
243
|
+
private Math(): Expression {
|
244
|
+
let v: Expression = this.Mul();
|
245
|
+
let t: Token = this.GetToken();
|
246
|
+
|
247
|
+
while (t.getType() === TokenType.Oper && (t.getValue() === "+" || t.getValue() === "-")) {
|
248
|
+
// 转换操作数2为数字
|
249
|
+
let r: Expression = this.Mul();
|
250
|
+
|
251
|
+
// 开始运算
|
252
|
+
if (t.getValue() === "+") {
|
253
|
+
v = Expression.Add(v, r, this.SOURCE, this.pos);
|
254
|
+
} else {
|
255
|
+
v = Expression.Subtract(v, r, this.SOURCE, this.pos);
|
256
|
+
}
|
257
|
+
|
258
|
+
t = this.GetToken();
|
259
|
+
}
|
260
|
+
|
261
|
+
this.tokens.push(t); // Push the token back
|
262
|
+
return v;
|
263
|
+
}
|
264
|
+
|
265
|
+
private Lambda(): Expression {
|
266
|
+
let t: Token = this.GetToken();
|
267
|
+
if (t.getValue() !== "=>" || t.getValue() == null) {
|
268
|
+
throw new Error(this.GetExceptionMessage("lambda必须以=>开始"));
|
269
|
+
}
|
270
|
+
|
271
|
+
t = this.GetToken();
|
272
|
+
if (t.getValue() !== "{" || t.getValue() == null) {
|
273
|
+
throw new Error(this.GetExceptionMessage("lambda必须以{开始"));
|
274
|
+
}
|
275
|
+
|
276
|
+
// 取lambda中的表达式
|
277
|
+
const exp: Expression = this.CommaExp();
|
278
|
+
|
279
|
+
t = this.GetToken();
|
280
|
+
if (t.getValue() !== "}" || t.getValue() == null) {
|
281
|
+
throw new Error(this.GetExceptionMessage("lambda必须以}结束"));
|
282
|
+
}
|
283
|
+
|
284
|
+
return Expression.Lambda(exp, this.pos);
|
285
|
+
}
|
286
|
+
|
287
|
+
private Mul(): Expression {
|
288
|
+
let v: Expression = this.UnarySub();
|
289
|
+
let t: Token = this.GetToken();
|
290
|
+
|
291
|
+
while (t.getType() === TokenType.Oper && (t.getValue() === "*" || t.getValue() === "/" || t.getValue() === "%")) {
|
292
|
+
// Convert the second operand to a number
|
293
|
+
const r: Expression = this.UnarySub();
|
294
|
+
|
295
|
+
// Perform the operation
|
296
|
+
if (t.getValue() === "*") {
|
297
|
+
v = Expression.Multiply(v, r, this.SOURCE, this.pos);
|
298
|
+
} else if (t.getValue() === "/") {
|
299
|
+
v = Expression.Divide(v, r, this.SOURCE, this.pos);
|
300
|
+
} else {
|
301
|
+
v = Expression.Modulo(v, r, this.SOURCE, this.pos);
|
302
|
+
}
|
303
|
+
|
304
|
+
t = this.GetToken();
|
305
|
+
}
|
306
|
+
|
307
|
+
this.tokens.push(t);
|
308
|
+
return v;
|
309
|
+
}
|
310
|
+
|
311
|
+
private UnarySub(): Expression {
|
312
|
+
let t: Token = this.GetToken();
|
313
|
+
if (t.getType() === TokenType.Oper && t.getValue() === "-") {
|
314
|
+
let r: Expression = this.Item();
|
315
|
+
return Expression.Subtract(Expression.Constant(0, this.pos), r, this.SOURCE, this.pos);
|
316
|
+
}
|
317
|
+
this.tokens.push(t); // Push the token back into the queue
|
318
|
+
return this.Item();
|
319
|
+
}
|
320
|
+
|
321
|
+
private Item(): Expression {
|
322
|
+
let objName: string = "";
|
323
|
+
// 获取对象表达式
|
324
|
+
let objExp: Expression = this.ItemHead(objName);
|
325
|
+
// 获取对象路径表达式
|
326
|
+
objExp = this.ObjectPath(objExp);
|
327
|
+
return objExp;
|
328
|
+
}
|
329
|
+
|
330
|
+
private ItemHead(name: string): Expression {
|
331
|
+
let t: Token = this.GetToken();
|
332
|
+
|
333
|
+
if (t.getType() === TokenType.Oper && t.getValue() === "(") {
|
334
|
+
let result: Expression = this.CommaExp();
|
335
|
+
t = this.GetToken();
|
336
|
+
if (t.getType() !== TokenType.Oper || t.getValue() !== ")") {
|
337
|
+
throw new Error(this.GetExceptionMessage("括号不匹配"));
|
338
|
+
}
|
339
|
+
return result;
|
340
|
+
} else if (t.getType() === TokenType.Oper && t.getValue() === "{") {
|
341
|
+
// JSON object
|
342
|
+
return this.Json();
|
343
|
+
} else if (t.getType() === TokenType.Oper && t.getValue() === "[") {
|
344
|
+
// JSON array
|
345
|
+
return this.JsonArray();
|
346
|
+
} else if (t.getType() === TokenType.Oper && (t.getValue() === "$" || t.getValue() === "\"")) {
|
347
|
+
// String concatenation sequence
|
348
|
+
return this.StringUnion();
|
349
|
+
} else if (t.getType() === TokenType.Int || t.getType() === TokenType.Double || t.getType() === TokenType.Bool) {
|
350
|
+
return Expression.Constant(t.getValue(), this.pos);
|
351
|
+
} else if (t.getType() === TokenType.Identy) {
|
352
|
+
let objName: string = <string>t.getValue();
|
353
|
+
|
354
|
+
switch (objName) {
|
355
|
+
case "return":
|
356
|
+
return Expression.Return(this.Exp(), this.pos);
|
357
|
+
case "try":
|
358
|
+
this.tokens.push(t); // Revert the token back to the queue
|
359
|
+
return this.Try();
|
360
|
+
case "throw":
|
361
|
+
return Expression.Throw(this.Exp(), this.pos);
|
362
|
+
case "validate":
|
363
|
+
return this.Validate();
|
364
|
+
case "assert":
|
365
|
+
return Expression.Assert(this.Exp(), this.pos);
|
366
|
+
case "break":
|
367
|
+
return Expression.Break(this.pos, undefined);
|
368
|
+
case "continue":
|
369
|
+
return Expression.Continue(this.pos, undefined);
|
370
|
+
}
|
371
|
+
|
372
|
+
// Return the object name
|
373
|
+
name += objName;
|
374
|
+
// Process the object name
|
375
|
+
return this.ObjectName(objName);
|
376
|
+
} else if (t.getType() === TokenType.Null) {
|
377
|
+
return Expression.Constant(null, this.pos);
|
378
|
+
}
|
379
|
+
|
380
|
+
throw new Error(this.GetExceptionMessage("单词类型错误:" + t.getType()));
|
381
|
+
}
|
382
|
+
|
383
|
+
/**
|
384
|
+
* TryCatch := try {expression} (catch(Exception类 名){表达式})* finally{表达式}?
|
385
|
+
*/
|
386
|
+
private Try(): Expression {
|
387
|
+
let t: Token = this.GetToken();
|
388
|
+
if (t.getValue() !== "try") {
|
389
|
+
throw new Error(this.GetExceptionMessage("try-catch必须以try开始"));
|
390
|
+
}
|
391
|
+
|
392
|
+
t = this.GetToken();
|
393
|
+
if (t.getValue() !== "{") {
|
394
|
+
throw new Error(this.GetExceptionMessage("try块必须以{开始"));
|
395
|
+
}
|
396
|
+
|
397
|
+
// Get the expression inside the try block
|
398
|
+
let tryExp: Expression = this.CommaExp();
|
399
|
+
|
400
|
+
t = this.GetToken();
|
401
|
+
if (t.getValue() !== "}") {
|
402
|
+
throw new Error(this.GetExceptionMessage("try块必须以}结束"));
|
403
|
+
}
|
404
|
+
|
405
|
+
// Handle the catch part
|
406
|
+
let catches: Expression[] = this.Catch();
|
407
|
+
|
408
|
+
// Handle finally
|
409
|
+
let finallyExp: Expression | undefined = undefined;
|
410
|
+
t = this.GetToken();
|
411
|
+
if (t.getValue()) {
|
412
|
+
if (t.getValue() === "finally") {
|
413
|
+
t = this.GetToken();
|
414
|
+
if (t.getValue() !== "{") {
|
415
|
+
throw new Error(this.GetExceptionMessage("finally块必须以{开始"));
|
416
|
+
}
|
417
|
+
|
418
|
+
// Get the expression inside the finally block
|
419
|
+
finallyExp = this.CommaExp();
|
420
|
+
t = this.GetToken();
|
421
|
+
if (t.getValue() !== "}") {
|
422
|
+
throw new Error(this.GetExceptionMessage("finally块必须以}结束"));
|
423
|
+
}
|
424
|
+
t = this.GetToken();
|
425
|
+
}
|
426
|
+
}
|
427
|
+
|
428
|
+
this.tokens.push(t);
|
429
|
+
|
430
|
+
// Return the exception handling expression
|
431
|
+
return Expression.Try(tryExp, catches, this.pos, finallyExp);
|
432
|
+
}
|
433
|
+
|
434
|
+
/**
|
435
|
+
* catch部分处理
|
436
|
+
*/
|
437
|
+
private Catch(): Expression[] {
|
438
|
+
let result: Expression[] = [];
|
439
|
+
let t: Token = this.GetToken();
|
440
|
+
|
441
|
+
while (t.getValue() === "catch") {
|
442
|
+
t = this.GetToken();
|
443
|
+
if (t.getValue() !== "(") {
|
444
|
+
throw new Error(this.GetExceptionMessage("catch块参数必须以(开始"));
|
445
|
+
}
|
446
|
+
|
447
|
+
// Get the class name
|
448
|
+
t = this.GetToken();
|
449
|
+
let className: string = <string>t.getValue();
|
450
|
+
|
451
|
+
// Get the variable name
|
452
|
+
t = this.GetToken();
|
453
|
+
let varName: string = <string>t.getValue();
|
454
|
+
|
455
|
+
t = this.GetToken();
|
456
|
+
if (t.getValue() !== ")") {
|
457
|
+
throw new Error(this.GetExceptionMessage("catch块参数必须以)结束"));
|
458
|
+
}
|
459
|
+
|
460
|
+
// Get the opening curly brace
|
461
|
+
t = this.GetToken();
|
462
|
+
if (t.getValue() !== "{") {
|
463
|
+
throw new Error(this.GetExceptionMessage("catch块必须以{开始"));
|
464
|
+
}
|
465
|
+
|
466
|
+
// Get the expression inside the catch block
|
467
|
+
let catchExp: Expression = this.CommaExp();
|
468
|
+
result.push(Expression.Catch(className, varName, catchExp, this.pos));
|
469
|
+
|
470
|
+
// Get the closing curly brace
|
471
|
+
t = this.GetToken();
|
472
|
+
if (t.getValue() !== "}") {
|
473
|
+
throw new Error(this.GetExceptionMessage("catch块必须以}结束"));
|
474
|
+
}
|
475
|
+
|
476
|
+
// Get the next token
|
477
|
+
t = this.GetToken();
|
478
|
+
}
|
479
|
+
|
480
|
+
this.tokens.push(t);
|
481
|
+
return result;
|
482
|
+
}
|
483
|
+
|
484
|
+
/**
|
485
|
+
* Validate block handling
|
486
|
+
*/
|
487
|
+
private Validate(): Expression {
|
488
|
+
let t: Token = this.GetToken();
|
489
|
+
if (t.getValue() !== "{") {
|
490
|
+
throw new Error(this.GetExceptionMessage("validate只接收Json对象"));
|
491
|
+
}
|
492
|
+
|
493
|
+
// Get the expression in the validate block
|
494
|
+
let exp: Expression = this.Json();
|
495
|
+
|
496
|
+
// Return Validate processing expression
|
497
|
+
return Expression.Validate(exp, this.pos);
|
498
|
+
}
|
499
|
+
|
500
|
+
/**
|
501
|
+
* JSON Object ::= {} | {propertyName: propertyValue, propertyName: propertyValue}
|
502
|
+
*/
|
503
|
+
private Json(): Expression {
|
504
|
+
let children: Expression[] = [];
|
505
|
+
|
506
|
+
let t: Token = this.GetToken();
|
507
|
+
// Empty object, return directly
|
508
|
+
if (t.getType() === TokenType.Oper && t.getValue() === "}") {
|
509
|
+
return Expression.Json(children, this.pos);
|
510
|
+
}
|
511
|
+
|
512
|
+
this.tokens.push(t);
|
513
|
+
children.push(this.Attr());
|
514
|
+
t = this.GetToken();
|
515
|
+
|
516
|
+
// If it's a comma, continue checking the next property
|
517
|
+
while (t.getType() === TokenType.Oper && t.getValue() === ",") {
|
518
|
+
let nextT: Token = this.GetToken();
|
519
|
+
if (nextT.getType() === TokenType.Oper && nextT.getValue() === "}") {
|
520
|
+
t = nextT;
|
521
|
+
break;
|
522
|
+
}
|
523
|
+
this.tokens.push(nextT);
|
524
|
+
children.push(this.Attr());
|
525
|
+
t = this.GetToken();
|
526
|
+
}
|
527
|
+
|
528
|
+
if (t.getType() !== TokenType.Oper || t.getValue() !== "}") {
|
529
|
+
throw new Error(this.GetExceptionMessage("JSON对象构建错误"));
|
530
|
+
}
|
531
|
+
|
532
|
+
return Expression.Json(children, this.pos);
|
533
|
+
}
|
534
|
+
|
535
|
+
/**
|
536
|
+
* JSON Array ::= []
|
537
|
+
*/
|
538
|
+
private JsonArray(): Expression {
|
539
|
+
let t: Token = this.GetToken();
|
540
|
+
// Empty array, return directly
|
541
|
+
if (t.getType() === TokenType.Oper && t.getValue() === "]") {
|
542
|
+
// Return JSON array constant
|
543
|
+
return Expression.Constant(new JSONArray(), this.pos);
|
544
|
+
}
|
545
|
+
|
546
|
+
// Loop to get array contents
|
547
|
+
this.tokens.push(t);
|
548
|
+
let ps: Expression[] = this.Params();
|
549
|
+
t = this.GetToken();
|
550
|
+
if (t.getType() !== TokenType.Oper || t.getValue() !== "]") {
|
551
|
+
throw new Error(this.GetExceptionMessage("JSON集合构建错误"));
|
552
|
+
}
|
553
|
+
|
554
|
+
return Expression.Array(ps, this.pos);
|
555
|
+
}
|
556
|
+
|
557
|
+
/**
|
558
|
+
* Attribute-Value Pair ::= propertyName: propertyValue
|
559
|
+
*/
|
560
|
+
private Attr(): Expression {
|
561
|
+
let name: Token = this.GetToken();
|
562
|
+
let value: string;
|
563
|
+
if (name.getType() === TokenType.Identy) {
|
564
|
+
value = <string>name.getValue();
|
565
|
+
} else if (name.getType() === TokenType.Oper && (name.getValue() === "$" || name.getValue() === "\"")) {
|
566
|
+
let expression: Expression = this.StringUnion().children[1] as Expression;
|
567
|
+
value = <string>expression.value;
|
568
|
+
} else {
|
569
|
+
throw new Error(this.GetExceptionMessage("JSON对象的key必须是属性名或字符串"));
|
570
|
+
}
|
571
|
+
|
572
|
+
let t: Token = this.GetToken();
|
573
|
+
if (t.getType() !== TokenType.Oper || t.getValue() !== ":") {
|
574
|
+
throw new Error(this.GetExceptionMessage("JSON构建错误"));
|
575
|
+
}
|
576
|
+
|
577
|
+
let exp: Expression = this.Exp();
|
578
|
+
return Expression.Attr(value, exp, this.pos);
|
579
|
+
}
|
580
|
+
|
581
|
+
/**
|
582
|
+
* Parse string concatenation sequence
|
583
|
+
*/
|
584
|
+
private StringUnion(): Expression {
|
585
|
+
let exp: Expression = Expression.Constant("", this.pos);
|
586
|
+
let t: Token = this.GetToken();
|
587
|
+
// Object sequence
|
588
|
+
while ((t.getType() === TokenType.Oper && t.getValue() === "{") || t.getType() === TokenType.String) {
|
589
|
+
// If it's a string, return the concatenated result
|
590
|
+
if (t.getType() === TokenType.String) {
|
591
|
+
exp = Expression.Concat(exp, Expression.Constant(t.getValue(), this.pos), this.SOURCE, this.pos);
|
592
|
+
} else {
|
593
|
+
// Handle content inside the {}
|
594
|
+
let objExp: Expression = this.Exp();
|
595
|
+
t = this.GetToken();
|
596
|
+
if (t.getType() !== TokenType.Oper || t.getValue() !== "}") {
|
597
|
+
throw new Error(this.GetExceptionMessage("缺少'}'"));
|
598
|
+
}
|
599
|
+
// String concatenation
|
600
|
+
exp = Expression.Concat(exp, objExp, this.SOURCE, this.pos);
|
601
|
+
}
|
602
|
+
t = this.GetToken();
|
603
|
+
}
|
604
|
+
this.tokens.push(t);
|
605
|
+
return exp;
|
606
|
+
}
|
607
|
+
|
608
|
+
/**
|
609
|
+
* Get object and object expression by name
|
610
|
+
*/
|
611
|
+
private ObjectName(objName: string): Expression {
|
612
|
+
return Expression.Identity(objName, this.pos);
|
613
|
+
}
|
614
|
+
|
615
|
+
/**
|
616
|
+
* Object path parsing, handles method calls, properties, and array indices
|
617
|
+
*/
|
618
|
+
private ObjectPath(inExp: Expression): Expression {
|
619
|
+
let objExp: Expression = this.ArrayIndex(inExp);
|
620
|
+
let t: Token = this.GetToken();
|
621
|
+
while (t.getType() === TokenType.Oper && t.getValue() === ".") {
|
622
|
+
let nameToken: Token = this.GetToken();
|
623
|
+
let n: Token = this.GetToken();
|
624
|
+
if (n.getType() === TokenType.Oper && n.getValue() === "(") {
|
625
|
+
let name: string = <string>nameToken.getValue();
|
626
|
+
if (name === "each") {
|
627
|
+
objExp = this.For(objExp);
|
628
|
+
} else {
|
629
|
+
objExp = this.MethodCall(name, objExp);
|
630
|
+
}
|
631
|
+
} else {
|
632
|
+
this.tokens.push(n);
|
633
|
+
let pi: string = <string>nameToken.getValue();
|
634
|
+
objExp = Expression.Property(objExp, pi, this.pos);
|
635
|
+
objExp = this.ArrayIndex(objExp);
|
636
|
+
}
|
637
|
+
t = this.GetToken();
|
638
|
+
}
|
639
|
+
this.tokens.push(t);
|
640
|
+
|
641
|
+
return objExp;
|
642
|
+
}
|
643
|
+
|
644
|
+
/**
|
645
|
+
* Array index parsing with optional condition
|
646
|
+
*/
|
647
|
+
private ArrayIndex(objExp: Expression): Expression {
|
648
|
+
let n: Token = this.GetToken();
|
649
|
+
if (n.getType() === TokenType.Oper && n.getValue() === "[") {
|
650
|
+
let exp: Expression = this.Exp();
|
651
|
+
n = this.GetToken();
|
652
|
+
if (n.getType() !== TokenType.Oper || n.getValue() !== "]") {
|
653
|
+
throw new Error(this.GetExceptionMessage("缺少']'"));
|
654
|
+
}
|
655
|
+
return Expression.ArrayIndex(objExp, exp, this.SOURCE, this.pos);
|
656
|
+
}
|
657
|
+
this.tokens.push(n);
|
658
|
+
return objExp;
|
659
|
+
}
|
660
|
+
|
661
|
+
/**
|
662
|
+
* For loop handling
|
663
|
+
*/
|
664
|
+
private For(obj: Expression): Expression {
|
665
|
+
let exp: Expression = this.CommaExp();
|
666
|
+
let t: Token = this.GetToken();
|
667
|
+
if (t.getType() !== TokenType.Oper || t.getValue() !== ")") {
|
668
|
+
throw new Error(this.GetExceptionMessage("函数调用括号不匹配"));
|
669
|
+
}
|
670
|
+
return Expression.For(obj, exp, this.SOURCE, this.pos);
|
671
|
+
}
|
672
|
+
|
673
|
+
/**
|
674
|
+
* Method call handling with parameters
|
675
|
+
*/
|
676
|
+
private MethodCall(name: string, obj: Expression): Expression {
|
677
|
+
let t: Token = this.GetToken();
|
678
|
+
if (t.getType() === TokenType.Oper && t.getValue() === ")") {
|
679
|
+
let ps: Expression[] = [];
|
680
|
+
return Expression.Call(obj, name, ps, this.pos);
|
681
|
+
}
|
682
|
+
|
683
|
+
this.tokens.push(t);
|
684
|
+
let ps: Expression[] = this.Params();
|
685
|
+
t = this.GetToken();
|
686
|
+
if (t.getType() !== TokenType.Oper || t.getValue() !== ")") {
|
687
|
+
throw new Error(this.GetExceptionMessage("函数调用括号不匹配"));
|
688
|
+
}
|
689
|
+
|
690
|
+
return Expression.Call(obj, name, ps, this.pos);
|
691
|
+
}
|
692
|
+
|
693
|
+
/**
|
694
|
+
* Function parameter list handling
|
695
|
+
*/
|
696
|
+
private Params(): Expression[] {
|
697
|
+
let ps: Expression[] = [];
|
698
|
+
let name: string = this.paramName();
|
699
|
+
let exp: Expression = this.Exp();
|
700
|
+
this.procParam(ps, exp, name);
|
701
|
+
let t: Token = this.GetToken();
|
702
|
+
while (t.getType() === TokenType.Oper && t.getValue() === ",") {
|
703
|
+
name = this.paramName();
|
704
|
+
exp = this.Exp();
|
705
|
+
this.procParam(ps, exp, name);
|
706
|
+
t = this.GetToken();
|
707
|
+
}
|
708
|
+
this.tokens.push(t);
|
709
|
+
return ps;
|
710
|
+
}
|
711
|
+
|
712
|
+
/**
|
713
|
+
* Parse parameter name
|
714
|
+
*/
|
715
|
+
private paramName(): string {
|
716
|
+
let t: Token = this.GetToken();
|
717
|
+
if (t.getType() === TokenType.Identy) {
|
718
|
+
let t1: Token = this.GetToken();
|
719
|
+
if (t1.getValue() === ":") {
|
720
|
+
return <string>t.getValue();
|
721
|
+
} else {
|
722
|
+
this.tokens.push(t);
|
723
|
+
this.tokens.push(t1);
|
724
|
+
return "";
|
725
|
+
}
|
726
|
+
} else {
|
727
|
+
this.tokens.push(t);
|
728
|
+
return "";
|
729
|
+
}
|
730
|
+
}
|
731
|
+
|
732
|
+
/**
|
733
|
+
* Process parameters, especially if it contains the "data" object
|
734
|
+
*/
|
735
|
+
private procParam(ps: Expression[], exp: Expression, name: string): void {
|
736
|
+
let delegate: Delegate = this.createDelegate(exp);
|
737
|
+
if (delegate.containsKey("data")) {
|
738
|
+
ps.push(Expression.Constant(delegate, this.pos));
|
739
|
+
} else {
|
740
|
+
ps.push(Expression.Param(exp, name, this.pos));
|
741
|
+
}
|
742
|
+
}
|
743
|
+
|
744
|
+
/**
|
745
|
+
* Get exception message with position information
|
746
|
+
*/
|
747
|
+
private GetExceptionMessage(msg: string): string {
|
748
|
+
let result: string = this.SOURCE.substring(0, this.pos) + " <- " + this.SOURCE.substring(this.pos);
|
749
|
+
return `${msg}:\n${result}`;
|
750
|
+
}
|
751
|
+
|
752
|
+
public GetToken(): Token {
|
753
|
+
// If there are tokens in the queue, return the last one saved
|
754
|
+
if (this.tokens.length > 0) {
|
755
|
+
return this.tokens.shift()!;
|
756
|
+
}
|
757
|
+
|
758
|
+
const sPos = this.pos;
|
759
|
+
let hasKeyword = this.pos < this.SOURCE_LENGTH;
|
760
|
+
let value: string | null = null;
|
761
|
+
|
762
|
+
if (hasKeyword) {
|
763
|
+
value = this.SOURCE.charAt(this.pos);
|
764
|
+
}
|
765
|
+
|
766
|
+
// If it's '{', save the previous state and change to non-string state
|
767
|
+
if (hasKeyword && value === '{') {
|
768
|
+
this.inStrings.push(this.inString);
|
769
|
+
this.inString = false;
|
770
|
+
const str = this.SOURCE.substring(this.pos, this.pos + 1);
|
771
|
+
this.pos += 1;
|
772
|
+
return new Token(TokenType.Oper, str, sPos);
|
773
|
+
}
|
774
|
+
|
775
|
+
// If in string state, process the string
|
776
|
+
if (this.inString) {
|
777
|
+
let startPos = this.pos;
|
778
|
+
while (hasKeyword && value !== '{' && value !== '}' && value !== '$' && value !== '\"') {
|
779
|
+
if (value === '\\') {
|
780
|
+
const nextStrValue = this.SOURCE.charAt(this.pos + 1);
|
781
|
+
if (this.pos + 1 < this.SOURCE_LENGTH && ['{', '}', '\"', '$', '\\'].includes(nextStrValue)) {
|
782
|
+
this.pos++;
|
783
|
+
} else {
|
784
|
+
break;
|
785
|
+
}
|
786
|
+
}
|
787
|
+
this.pos++;
|
788
|
+
if (this.pos < this.SOURCE_LENGTH) {
|
789
|
+
value = this.SOURCE.charAt(this.pos);
|
790
|
+
} else {
|
791
|
+
hasKeyword = false;
|
792
|
+
}
|
793
|
+
}
|
794
|
+
if (!(hasKeyword && value === '{')) {
|
795
|
+
this.inString = this.inStrings.pop()!;
|
796
|
+
}
|
797
|
+
const t = new Token(TokenType.String, this.SOURCE.substring(startPos, this.pos)
|
798
|
+
.replace("\\$", "$")
|
799
|
+
.replace("\\{", "{")
|
800
|
+
.replace("\\}", "}")
|
801
|
+
.replace(/\\"/g, '"')
|
802
|
+
.replace("\\\\", "\\"), sPos);
|
803
|
+
|
804
|
+
if (hasKeyword && (value === '$' || value === '\"')) {
|
805
|
+
this.pos++;
|
806
|
+
}
|
807
|
+
return t;
|
808
|
+
}
|
809
|
+
|
810
|
+
// Skip whitespace and comments
|
811
|
+
while (hasKeyword && (value === ' ' || value === '\n' || value === '\t' || (this.pos < this.SOURCE_LENGTH - 2 && value === '/' && this.SOURCE.charAt(this.pos + 1) === '/'))) {
|
812
|
+
if (value !== ' ' && value !== '\n' && value !== '\t') {
|
813
|
+
this.pos += 2;
|
814
|
+
if (this.pos < this.SOURCE_LENGTH) {
|
815
|
+
value = this.SOURCE.charAt(this.pos);
|
816
|
+
} else {
|
817
|
+
hasKeyword = false;
|
818
|
+
}
|
819
|
+
while (hasKeyword && value !== '\n') {
|
820
|
+
this.pos++;
|
821
|
+
if (this.pos < this.SOURCE_LENGTH) {
|
822
|
+
value = this.SOURCE.charAt(this.pos);
|
823
|
+
} else {
|
824
|
+
hasKeyword = false;
|
825
|
+
}
|
826
|
+
}
|
827
|
+
}
|
828
|
+
this.pos++;
|
829
|
+
if (this.pos < this.SOURCE_LENGTH) {
|
830
|
+
value = this.SOURCE.charAt(this.pos);
|
831
|
+
} else {
|
832
|
+
hasKeyword = false;
|
833
|
+
}
|
834
|
+
}
|
835
|
+
|
836
|
+
// If we are at the end of the source, return an "End" token
|
837
|
+
if (this.pos === this.SOURCE_LENGTH) {
|
838
|
+
return new Token(TokenType.End, null, sPos);
|
839
|
+
}
|
840
|
+
|
841
|
+
if (value !== null) {
|
842
|
+
// Handle number tokens
|
843
|
+
if (value >= '0' && value <= '9') {
|
844
|
+
const oldPos = this.pos;
|
845
|
+
while (hasKeyword && value >= '0' && value <= '9') {
|
846
|
+
this.pos++;
|
847
|
+
if (this.pos < this.SOURCE_LENGTH) {
|
848
|
+
value = this.SOURCE.charAt(this.pos);
|
849
|
+
} else {
|
850
|
+
hasKeyword = false;
|
851
|
+
}
|
852
|
+
}
|
853
|
+
|
854
|
+
if (hasKeyword && value === '.') {
|
855
|
+
do {
|
856
|
+
this.pos++;
|
857
|
+
if (this.pos < this.SOURCE_LENGTH) {
|
858
|
+
value = this.SOURCE.charAt(this.pos);
|
859
|
+
} else {
|
860
|
+
hasKeyword = false;
|
861
|
+
}
|
862
|
+
} while (hasKeyword && value >= '0' && value <= '9');
|
863
|
+
const str = this.SOURCE.substring(oldPos, this.pos);
|
864
|
+
return new Token(TokenType.Double, parseFloat(str), sPos);
|
865
|
+
} else {
|
866
|
+
const str = this.SOURCE.substring(oldPos, this.pos);
|
867
|
+
return new Token(TokenType.Int, parseInt(str, 10), sPos);
|
868
|
+
}
|
869
|
+
}
|
870
|
+
|
871
|
+
// Handle identifier tokens
|
872
|
+
else if ((value >= 'a' && value <= 'z') || (value >= 'A' && value <= 'Z') || value === '_' || /[\u4e00-\u9fa5]/.test(value)) {
|
873
|
+
const oldPos = this.pos;
|
874
|
+
while (hasKeyword && ((value >= 'a' && value <= 'z') || (value >= 'A' && value <= 'Z') || (value >= '0' && value <= '9') || value === '_' || /[\u4e00-\u9fa5]/.test(value))) {
|
875
|
+
this.pos++;
|
876
|
+
if (this.pos < this.SOURCE_LENGTH) {
|
877
|
+
value = this.SOURCE.charAt(this.pos);
|
878
|
+
} else {
|
879
|
+
hasKeyword = false;
|
880
|
+
}
|
881
|
+
}
|
882
|
+
const str = this.SOURCE.substring(oldPos, this.pos);
|
883
|
+
switch (str) {
|
884
|
+
case 'false':
|
885
|
+
case 'true':
|
886
|
+
return new Token(TokenType.Bool, Boolean(str), sPos);
|
887
|
+
case 'null':
|
888
|
+
return new Token(TokenType.Null, null, sPos);
|
889
|
+
default:
|
890
|
+
return new Token(TokenType.Identy, str, sPos);
|
891
|
+
}
|
892
|
+
}
|
893
|
+
|
894
|
+
// Handle operators like &&, ||, and single-character operators
|
895
|
+
else if ((value === '&' && this.SOURCE.charAt(this.pos + 1) === '&') || (value === '|' && this.SOURCE.charAt(this.pos + 1) === '|')) {
|
896
|
+
const str = this.SOURCE.substring(this.pos, this.pos + 2);
|
897
|
+
this.pos += 2;
|
898
|
+
return new Token(TokenType.Oper, str, sPos);
|
899
|
+
}
|
900
|
+
|
901
|
+
// Handle other operators (+, -, *, /, etc.)
|
902
|
+
else if (['+', '-', '*', '/', '%', '>', '<', '!'].includes(value)) {
|
903
|
+
let str = '';
|
904
|
+
if (this.SOURCE.charAt(this.pos + 1) === '=') {
|
905
|
+
str = this.SOURCE.substring(this.pos, this.pos + 2);
|
906
|
+
this.pos += 2;
|
907
|
+
} else {
|
908
|
+
str = this.SOURCE.substring(this.pos, this.pos + 1);
|
909
|
+
this.pos += 1;
|
910
|
+
}
|
911
|
+
return new Token(TokenType.Oper, str, sPos);
|
912
|
+
}
|
913
|
+
|
914
|
+
// Handle '=' with different cases
|
915
|
+
else if (value === '=') {
|
916
|
+
let str = '';
|
917
|
+
if (this.SOURCE.charAt(this.pos + 1) === '=' || this.SOURCE.charAt(this.pos + 1) === '>') {
|
918
|
+
str = this.SOURCE.substring(this.pos, this.pos + 2);
|
919
|
+
this.pos += 2;
|
920
|
+
} else {
|
921
|
+
str = this.SOURCE.substring(this.pos, this.pos + 1);
|
922
|
+
this.pos += 1;
|
923
|
+
}
|
924
|
+
return new Token(TokenType.Oper, str, sPos);
|
925
|
+
}
|
926
|
+
|
927
|
+
// Single character operators like parentheses, commas, etc.
|
928
|
+
else if ('()[],;.:@{}"$'.includes(value)) {
|
929
|
+
if (value === '$' || value === '\"') {
|
930
|
+
this.inStrings.push(false);
|
931
|
+
this.inString = true;
|
932
|
+
} else if ((value === '{' || value === '}') && this.inStrings.length) {
|
933
|
+
if (value === '{') {
|
934
|
+
this.inStrings.push(false);
|
935
|
+
this.inString = false;
|
936
|
+
} else {
|
937
|
+
this.inString = this.inStrings.pop()!;
|
938
|
+
}
|
939
|
+
}
|
940
|
+
const str = this.SOURCE.substring(this.pos, this.pos + 1);
|
941
|
+
this.pos += 1;
|
942
|
+
return new Token(TokenType.Oper, str, sPos);
|
943
|
+
} else {
|
944
|
+
throw new Error(`Invalid token: '${value}'`);
|
945
|
+
}
|
946
|
+
}
|
947
|
+
|
948
|
+
throw new Error('Unknown error while parsing.');
|
949
|
+
}
|
950
|
+
}
|