1
0

params.wxml 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389
  1. <navigation-bar background="{{themeMode === 'dark' ? '#111827' : '#FFF'}}"></navigation-bar>
  2. <view wx:if="{{toastText}}" class="page-toast page-toast--{{toastType}} {{themeClass}}">
  3. {{toastText}}
  4. </view>
  5. <view class="subpage-fixed-header subpage-fixed-header--generic {{themeClass}}">
  6. <view class="subpage-page-header">
  7. <view wx:if="{{activeParamView == 'parameterGroups'}}" class="panel-actions subpage-actions generic-protocol-actions">
  8. <view wx:if="{{isStorageAccessProtocol}}" class="panel-action-button" bindtap="clearStorageAccessGroups">清除</view>
  9. <view class="panel-action-button {{connectedDevice ? '' : 'is-disabled'}}" bindtap="readAllParameterGroups">读取</view>
  10. <view class="panel-action-button" bindtap="saveParameterGroupsJson">保存</view>
  11. <view class="panel-action-button" bindtap="importParameterGroupsJson">加载</view>
  12. <view wx:if="{{isStorageAccessProtocol}}" class="panel-action-button" bindtap="completeParameterStructs">结构</view>
  13. <view wx:if="{{isModbusProtocol}}" class="panel-action-button panel-action-button--icon" bindtap="openParameterDraft">+</view>
  14. </view>
  15. <view wx:elif="{{activeParamView == 'parameterGroup'}}" class="panel-actions subpage-actions">
  16. <view
  17. class="panel-action-button {{connectedDevice && !activeParameterGroup.addressOverflow ? '' : 'is-disabled'}}"
  18. data-group-id="{{activeParameterGroup.id}}"
  19. bindtap="readParameterGroup"
  20. >读取</view>
  21. <view
  22. wx:if="{{activeParameterGroup.writable}}"
  23. class="panel-action-button {{connectedDevice && !activeParameterGroup.addressOverflow ? '' : 'is-disabled'}}"
  24. data-group-id="{{activeParameterGroup.id}}"
  25. bindtap="writeParameterGroup"
  26. >写入</view>
  27. <view class="panel-action-button" bindtap="backToParamsHome">返回</view>
  28. </view>
  29. </view>
  30. </view>
  31. <scroll-view class="scrollarea {{themeClass}} scrollarea--subpage scrollarea--generic" scroll-y type="list">
  32. <view class="page-shell">
  33. <block wx:if="{{activeParamView == 'parameterGroups'}}">
  34. <view
  35. wx:for="{{parameterGroups}}"
  36. wx:for-item="group"
  37. wx:key="id"
  38. class="generic-group-shell {{group.deleteVisible ? 'is-delete-visible' : ''}}"
  39. >
  40. <view
  41. wx:if="{{!group.expanded && group.deleteVisible}}"
  42. class="generic-delete-action"
  43. data-group-id="{{group.id}}"
  44. bindtap="deleteParameterGroup"
  45. >
  46. -
  47. </view>
  48. <view
  49. class="panel generic-group-panel {{group.expanded ? '' : 'panel--collapsed'}}"
  50. data-group-id="{{group.id}}"
  51. bindtouchstart="onParameterGroupTouchStart"
  52. bindtouchend="onParameterGroupTouchEnd"
  53. >
  54. <view class="panel-header panel-header--with-actions">
  55. <block wx:if="{{group.isStorageScalarGroup}}">
  56. <view class="panel-heading-toggle generic-scalar-heading">
  57. <view class="panel-icon icon-terminal">
  58. <image class="panel-icon-image" src="/assets/icons/terminal-white.png" mode="aspectFit" />
  59. </view>
  60. <view class="generic-group-title-wrap">
  61. <view
  62. class="panel-title"
  63. data-group-id="{{group.id}}"
  64. catchlongpress="openStorageScalarRegisterEdit"
  65. >{{group.displayName || group.name}}</view>
  66. <view class="param-meta generic-group-meta">{{group.listMetaText || group.addressRangeText}}</view>
  67. </view>
  68. </view>
  69. <view class="generic-scalar-value">{{group.scalarValueText || '--'}}</view>
  70. </block>
  71. <block wx:else>
  72. <view
  73. class="panel-heading-toggle"
  74. data-group-id="{{group.id}}"
  75. bindtap="openParameterGroup"
  76. >
  77. <view class="panel-icon icon-terminal">
  78. <image class="panel-icon-image" src="/assets/icons/terminal-white.png" mode="aspectFit" />
  79. </view>
  80. <view class="generic-group-title-wrap">
  81. <view class="panel-title" data-group-id="{{group.id}}" catchlongpress="openParameterGroupEdit">{{group.displayName || group.name}}</view>
  82. <view class="param-meta generic-group-meta">{{group.listMetaText || group.addressRangeText}}</view>
  83. </view>
  84. </view>
  85. <view wx:if="{{isModbusProtocol}}" class="panel-actions generic-group-actions">
  86. <view
  87. class="panel-action-button {{connectedDevice && !group.addressOverflow ? '' : 'is-disabled'}}"
  88. data-group-id="{{group.id}}"
  89. bindtap="readParameterGroup"
  90. >
  91. 读取
  92. </view>
  93. <view
  94. wx:if="{{group.writable}}"
  95. class="panel-action-button {{connectedDevice && !group.addressOverflow ? '' : 'is-disabled'}}"
  96. data-group-id="{{group.id}}"
  97. bindtap="writeParameterGroup"
  98. >
  99. 写入
  100. </view>
  101. <view wx:if="{{parameterCardControlEnabled}}" class="entry-chevron"></view>
  102. </view>
  103. </block>
  104. </view>
  105. <view wx:if="{{!parameterCardControlEnabled && group.expanded}}" class="generic-group-inline-registers">
  106. <view
  107. wx:for="{{group.registers}}"
  108. wx:for-item="register"
  109. wx:for-index="registerIndex"
  110. wx:key="id"
  111. class="generic-register-row generic-register-row--inline"
  112. >
  113. <view class="generic-register-layout-spacer"></view>
  114. <view class="generic-register-main">
  115. <view
  116. class="generic-register-name"
  117. data-group-id="{{group.id}}"
  118. data-index="{{registerIndex}}"
  119. bindtap="openParameterRegisterInfo"
  120. catchlongpress="openParameterRegisterEdit"
  121. >
  122. {{register.name}}
  123. </view>
  124. <view class="generic-register-meta">
  125. <text>{{register.metaText || (register.addressText + ' ' + register.rawValueText)}}</text>
  126. </view>
  127. </view>
  128. <view class="generic-register-input-wrap {{register.showUnit && register.unit ? 'generic-register-input-wrap--unit' : ''}}">
  129. <block wx:if="{{group.writable}}">
  130. <block wx:if="{{register.conversionFormula}}">
  131. <view class="param-value generic-readonly-value">{{register.displayValue || '--'}}{{register.showUnit && register.unit ? ' ' + register.unit : ''}}</view>
  132. </block>
  133. <block wx:else>
  134. <input
  135. class="value-input generic-register-value {{register.isDirty ? 'value-input--dirty' : ''}}"
  136. data-group-id="{{group.id}}"
  137. data-index="{{registerIndex}}"
  138. value="{{register.inputValue}}"
  139. bindinput="onParameterRegisterValueInput"
  140. bindblur="onParameterRegisterValueBlur"
  141. />
  142. <view wx:if="{{register.showUnit && register.unit}}" class="generic-register-unit">{{register.unit}}</view>
  143. </block>
  144. </block>
  145. <view wx:else class="param-value generic-readonly-value">{{register.displayValue || '--'}}{{register.showUnit && register.unit ? ' ' + register.unit : ''}}</view>
  146. <view wx:if="{{register.displayMetaText}}" class="generic-register-display-meta">{{register.displayMetaText}}</view>
  147. </view>
  148. </view>
  149. </view>
  150. </view>
  151. </view>
  152. </block>
  153. <block wx:elif="{{activeParamView == 'parameterGroup'}}">
  154. <view wx:if="{{activeParameterGroup}}" class="panel generic-group-detail-panel">
  155. <view class="generic-group-detail-header">
  156. <view class="panel-title">{{activeParameterGroup.detailTitleText || activeParameterGroup.displayName || activeParameterGroup.name}}</view>
  157. </view>
  158. <view class="generic-group-detail-meta">
  159. {{activeParameterGroup.detailMetaText || activeParameterGroup.addressRangeText}}
  160. </view>
  161. <view
  162. wx:for="{{activeParameterRegisterRows.length ? activeParameterRegisterRows : activeParameterGroup.registers}}"
  163. wx:for-item="register"
  164. wx:for-index="registerIndex"
  165. wx:key="id"
  166. class="generic-register-row {{register.dragClass}}"
  167. style="{{register.dragStyle}}"
  168. >
  169. <view
  170. wx:if="{{!activeParameterGroup.isStructLayout}}"
  171. class="generic-register-drag-handle {{register.dragHandleClass}}"
  172. data-group-id="{{activeParameterGroup.id}}"
  173. data-index="{{register.sourceIndex !== undefined ? register.sourceIndex : registerIndex}}"
  174. catchtouchstart="onParameterRegisterDragStart"
  175. catchtouchmove="onParameterRegisterDragMove"
  176. catchtouchend="onParameterRegisterDragEnd"
  177. catchtouchcancel="onParameterRegisterDragCancel"
  178. >
  179. <view class="generic-register-drag-bar"></view>
  180. <view class="generic-register-drag-bar"></view>
  181. <view class="generic-register-drag-bar"></view>
  182. </view>
  183. <view wx:else class="generic-register-layout-spacer"></view>
  184. <view class="generic-register-main">
  185. <view
  186. class="generic-register-name"
  187. data-group-id="{{activeParameterGroup.id}}"
  188. data-index="{{register.sourceIndex !== undefined ? register.sourceIndex : registerIndex}}"
  189. bindtap="openParameterRegisterInfo"
  190. catchlongpress="openParameterRegisterEdit"
  191. >
  192. {{register.name}}
  193. </view>
  194. <view class="generic-register-meta">
  195. <text>{{register.metaText || (register.addressText + ' ' + register.rawValueText)}}</text>
  196. </view>
  197. </view>
  198. <view class="generic-register-input-wrap {{register.showUnit && register.unit ? 'generic-register-input-wrap--unit' : ''}}">
  199. <block wx:if="{{activeParameterGroup.writable}}">
  200. <block wx:if="{{register.conversionFormula}}">
  201. <view class="param-value generic-readonly-value">{{register.displayValue || '--'}}{{register.showUnit && register.unit ? ' ' + register.unit : ''}}</view>
  202. </block>
  203. <block wx:else>
  204. <input
  205. class="value-input generic-register-value {{register.isDirty ? 'value-input--dirty' : ''}}"
  206. data-group-id="{{activeParameterGroup.id}}"
  207. data-index="{{register.sourceIndex !== undefined ? register.sourceIndex : registerIndex}}"
  208. value="{{register.inputValue}}"
  209. bindinput="onParameterRegisterValueInput"
  210. bindblur="onParameterRegisterValueBlur"
  211. />
  212. <view wx:if="{{register.showUnit && register.unit}}" class="generic-register-unit">{{register.unit}}</view>
  213. </block>
  214. </block>
  215. <view wx:else class="param-value generic-readonly-value">{{register.displayValue || '--'}}{{register.showUnit && register.unit ? ' ' + register.unit : ''}}</view>
  216. <view wx:if="{{register.displayMetaText}}" class="generic-register-display-meta">{{register.displayMetaText}}</view>
  217. </view>
  218. </view>
  219. </view>
  220. </block>
  221. </view>
  222. </scroll-view>
  223. <view wx:if="{{parameterDialog.visible}}" class="generic-dialog-mask {{themeClass}}" bindtap="closeParameterDraft">
  224. <view class="generic-dialog" catchtap="noop">
  225. <view class="generic-dialog-header">
  226. <view class="generic-dialog-title">{{parameterDialog.title}}</view>
  227. <view class="generic-dialog-close" bindtap="closeParameterDraft">×</view>
  228. </view>
  229. <block wx:if="{{parameterDialog.mode == 'createGroup' || parameterDialog.mode == 'editGroup'}}">
  230. <view class="generic-dialog-body">
  231. <view class="generic-config-row">
  232. <view class="param-main">
  233. <view class="param-name">{{isStorageAccessProtocol ? '结构体组名' : '寄存器组名'}}</view>
  234. <view class="param-meta">{{isStorageAccessProtocol ? '内存地址连续' : '每组寄存器地址连续'}}</view>
  235. </view>
  236. <input
  237. class="value-input generic-value-input"
  238. data-field="groupName"
  239. value="{{parameterDialog.groupName}}"
  240. bindinput="onParameterDraftInput"
  241. />
  242. </view>
  243. <view wx:if="{{isModbusProtocol}}" class="generic-config-row">
  244. <view class="param-main">
  245. <view class="param-name">寄存器类型</view>
  246. <view class="param-meta">决定读取功能码与是否可写</view>
  247. </view>
  248. <picker
  249. mode="selector"
  250. range="{{parameterRegisterTypeOptions}}"
  251. range-key="label"
  252. value="{{parameterDialog.registerTypeIndex}}"
  253. bindchange="onParameterDraftTypeChange"
  254. >
  255. <view class="generic-picker-value">{{parameterDialog.registerTypeText}}</view>
  256. </picker>
  257. </view>
  258. <view class="generic-config-row">
  259. <view class="param-main">
  260. <view class="param-name">{{isStorageAccessProtocol ? '内存起始地址' : '寄存器起始地址'}}</view>
  261. <view class="param-meta">{{isStorageAccessProtocol ? '16进制,最多 8 位,例如 000000A0' : '16进制,例如 00A0'}}</view>
  262. </view>
  263. <input
  264. class="value-input generic-value-input"
  265. data-field="startAddress"
  266. value="{{parameterDialog.startAddress}}"
  267. bindinput="onParameterDraftInput"
  268. />
  269. </view>
  270. <view class="generic-config-row">
  271. <view class="param-main">
  272. <view class="param-name">{{isStorageAccessProtocol ? '字节长度' : '寄存器数量'}}</view>
  273. <view class="param-meta">{{parameterDialog.structParsedSummary || '1 - 256'}}</view>
  274. </view>
  275. <input
  276. class="value-input generic-value-input"
  277. type="number"
  278. data-field="quantity"
  279. value="{{parameterDialog.quantity}}"
  280. bindinput="onParameterDraftInput"
  281. />
  282. </view>
  283. <view wx:if="{{parameterDialog.showPollEnabled}}" class="generic-config-row">
  284. <view class="param-main">
  285. <view class="param-name">参与轮询</view>
  286. <view class="param-meta">自动轮询</view>
  287. </view>
  288. <switch
  289. checked="{{parameterDialog.pollEnabled}}"
  290. color="#0f766e"
  291. data-field="pollEnabled"
  292. bindchange="onParameterDraftSwitchChange"
  293. />
  294. </view>
  295. <view wx:if="{{parameterDialog.mode == 'createGroup'}}" class="generic-struct-section">
  296. <view class="generic-struct-header">
  297. <view class="param-main">
  298. <view class="param-name">结构体/枚举定义</view>
  299. <view class="param-meta">支持 typedef struct、typedef enum、typedef 别名与数组</view>
  300. </view>
  301. <view class="panel-action-button" bindtap="parseParameterStructDefinition">解析</view>
  302. </view>
  303. <textarea
  304. class="generic-struct-input"
  305. maxlength="-1"
  306. placeholder="粘贴 C 结构体/枚举定义"
  307. data-field="structDefinition"
  308. value="{{parameterDialog.structDefinition}}"
  309. bindinput="onParameterDraftInput"
  310. />
  311. </view>
  312. </view>
  313. </block>
  314. <block wx:elif="{{parameterDialog.mode == 'editRegister' || parameterDialog.mode == 'viewRegister'}}">
  315. <view class="generic-dialog-body">
  316. <view class="generic-info-stack">
  317. <view class="generic-info-row">
  318. <view class="generic-info-label">名称</view>
  319. <view wx:if="{{parameterDialog.mode == 'viewRegister'}}" class="generic-info-value">{{parameterDialog.name}}</view>
  320. <input wx:else class="value-input generic-value-input" data-field="name" value="{{parameterDialog.name}}" bindinput="onParameterDraftInput" />
  321. </view>
  322. <view class="generic-info-row">
  323. <view class="generic-info-label">地址</view>
  324. <view class="generic-info-value">{{parameterDialog.addressText}}</view>
  325. </view>
  326. <view wx:if="{{parameterDialog.sourceMetaText}}" class="generic-info-row">
  327. <view class="generic-info-label">来源</view>
  328. <view class="generic-info-value">{{parameterDialog.sourceMetaText}}</view>
  329. </view>
  330. <view wx:if="{{parameterDialog.sourceByteLength}}" class="generic-info-row">
  331. <view class="generic-info-label">TLV长度</view>
  332. <view class="generic-info-value">{{parameterDialog.sourceByteLength}}B</view>
  333. </view>
  334. <view wx:if="{{parameterDialog.showDataType}}" class="generic-info-row">
  335. <view class="generic-info-label">类型</view>
  336. <view wx:if="{{parameterDialog.mode == 'viewRegister'}}" class="generic-info-value">{{parameterDialog.dataTypeText}}</view>
  337. <picker wx:else mode="selector" range="{{parameterDataTypeOptions}}" range-key="label" value="{{parameterDialog.dataTypeIndex}}" bindchange="onParameterDialogDataTypeChange">
  338. <view class="generic-picker-value">{{parameterDialog.dataTypeText}}</view>
  339. </picker>
  340. </view>
  341. <view wx:if="{{parameterDialog.showTextLength}}" class="generic-info-row">
  342. <view class="generic-info-label">长度</view>
  343. <view wx:if="{{parameterDialog.mode == 'viewRegister'}}" class="generic-info-value">{{parameterDialog.textByteLength || '--'}}B</view>
  344. <input wx:else class="value-input generic-value-input" type="number" data-field="textByteLength" value="{{parameterDialog.textByteLength}}" bindinput="onParameterDraftInput" />
  345. </view>
  346. <view class="generic-info-row">
  347. <view class="generic-info-label">备注</view>
  348. <view wx:if="{{parameterDialog.mode == 'viewRegister'}}" class="generic-info-value">{{parameterDialog.remark || '--'}}</view>
  349. <input wx:else class="value-input generic-value-input" data-field="remark" value="{{parameterDialog.remark}}" bindinput="onParameterDraftInput" />
  350. </view>
  351. <view wx:if="{{parameterDialog.showUnit}}" class="generic-info-row">
  352. <view class="generic-info-label">单位</view>
  353. <view wx:if="{{parameterDialog.mode == 'viewRegister'}}" class="generic-info-value">{{parameterDialog.unit || '--'}}</view>
  354. <input wx:else class="value-input generic-value-input" data-field="unit" value="{{parameterDialog.unit}}" bindinput="onParameterDraftInput" />
  355. </view>
  356. <view wx:if="{{parameterDialog.showDataType}}" class="generic-info-row">
  357. <view class="generic-info-label">公式</view>
  358. <view wx:if="{{parameterDialog.mode == 'viewRegister'}}" class="generic-info-value">{{parameterDialog.conversionFormula || '--'}}</view>
  359. <input wx:else class="value-input generic-value-input" data-field="conversionFormula" value="{{parameterDialog.conversionFormula}}" placeholder="x" bindinput="onParameterDraftInput" />
  360. </view>
  361. <view wx:if="{{parameterDialog.conversionFormulaErrorText}}" class="generic-info-row">
  362. <view class="generic-info-label">公式</view>
  363. <view class="generic-info-value">{{parameterDialog.conversionFormulaErrorText}}</view>
  364. </view>
  365. <view wx:if="{{parameterDialog.mode == 'viewRegister' || parameterDialog.showRange}}" class="generic-info-row">
  366. <view class="generic-info-label">最小值</view>
  367. <view wx:if="{{parameterDialog.mode == 'viewRegister'}}" class="generic-info-value">{{parameterDialog.minValue || '--'}}</view>
  368. <input wx:else class="value-input generic-value-input" data-field="minValue" value="{{parameterDialog.minValue}}" bindinput="onParameterDraftInput" />
  369. </view>
  370. <view wx:if="{{parameterDialog.mode == 'viewRegister' || parameterDialog.showRange}}" class="generic-info-row">
  371. <view class="generic-info-label">最大值</view>
  372. <view wx:if="{{parameterDialog.mode == 'viewRegister'}}" class="generic-info-value">{{parameterDialog.maxValue || '--'}}</view>
  373. <input wx:else class="value-input generic-value-input" data-field="maxValue" value="{{parameterDialog.maxValue}}" bindinput="onParameterDraftInput" />
  374. </view>
  375. </view>
  376. </view>
  377. </block>
  378. <view class="generic-draft-actions">
  379. <view class="panel-action-button" bindtap="closeParameterDraft">{{parameterDialog.cancelText}}</view>
  380. <view wx:if="{{parameterDialog.confirmText}}" class="panel-action-button is-active" bindtap="confirmParameterDialog">{{parameterDialog.confirmText}}</view>
  381. </view>
  382. </view>
  383. </view>