1
0

model.js 33 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868
  1. const {
  2. clampInteger,
  3. createId,
  4. formatFixedValue,
  5. normalizeTextValue,
  6. padHex,
  7. pickFields
  8. } = require('../../utils/base-utils.js')
  9. const {
  10. BYTE_ADDRESS_MEMORY_AREAS,
  11. DATA_TYPE_OPTIONS,
  12. DEFAULT_DATA_TYPE,
  13. DEFAULT_REGISTER_TYPE,
  14. DEFAULT_TEXT_BYTE_LENGTH,
  15. GROUP_LAYOUT_REGISTER,
  16. GROUP_LAYOUT_STRUCT,
  17. MAX_MODBUS_ADDRESS,
  18. MAX_PARAMETER_GROUP_ITEMS,
  19. MAX_STORAGE_ADDRESS,
  20. REGISTER_TYPE_OPTIONS,
  21. SOURCE_GROUP_FIELDS,
  22. SOURCE_REGISTER_FIELDS,
  23. STRUCT_REGISTER_FIELDS
  24. } = require('./constants.js')
  25. const {
  26. evaluateValueFormula
  27. } = require('./value-formula.js')
  28. const {
  29. alignEvenByteLength,
  30. decodeRegisterValue,
  31. formatCoilDisplayValue,
  32. formatRawByteText,
  33. formatRawByteTextWithDefault,
  34. formatRawWordText,
  35. formatRegisterValue,
  36. getDataType,
  37. getDataTypeIndex,
  38. getRegisterByteLength,
  39. getRegisterValueTypeLabel,
  40. getRegisterWordCount,
  41. getRegisterWordCountAtOffset,
  42. isBitFieldRegister,
  43. isBitRegisterType,
  44. isByteRegister,
  45. isTextRegister,
  46. normalizeBitOffset,
  47. normalizeBitWidth,
  48. normalizeTextByteLength,
  49. parseCoilValue,
  50. registerTypeIsBit,
  51. supportsRange,
  52. supportsUnit
  53. } = require('./value-codec.js')
  54. const {
  55. decodeRegisterFromByteCache,
  56. decodeRegisterFromWordCache,
  57. getGroupEncodedBytes,
  58. getGroupEncodedWords,
  59. getRegisterBytesFromByteCache,
  60. getRegisterEncodedBytes,
  61. getRegisterEncodedWords,
  62. getRegisterWordsFromByteCache,
  63. getRegisterWordsFromWordCache,
  64. getRegisterWriteValueText,
  65. splitWordSpans,
  66. validateRegisterValue
  67. } = require('./register-io.js')
  68. function normalizeAddress(value, fallback = 0) {
  69. if (typeof value === 'number') {
  70. return Number.isFinite(value) ? clampInteger(value, 0, MAX_MODBUS_ADDRESS, fallback) : fallback
  71. }
  72. const text = String(value === undefined || value === null ? '' : value).trim()
  73. if (!text) return fallback
  74. const hexText = text.toUpperCase().startsWith('0X') ? text.slice(2) : text
  75. if (/^[0-9A-F]+$/i.test(hexText)) {
  76. const parsedHex = parseInt(hexText, 16)
  77. return Number.isFinite(parsedHex) ? clampInteger(parsedHex, 0, MAX_MODBUS_ADDRESS, fallback) : fallback
  78. }
  79. const numberValue = Number(text)
  80. return Number.isFinite(numberValue) ? clampInteger(numberValue, 0, MAX_MODBUS_ADDRESS, fallback) : fallback
  81. }
  82. function normalizeStorageAddress(value, fallback = 0) {
  83. if (typeof value === 'number') {
  84. return Number.isFinite(value) ? clampInteger(value, 0, MAX_STORAGE_ADDRESS, fallback) : fallback
  85. }
  86. const text = String(value === undefined || value === null ? '' : value).trim()
  87. if (!text) return fallback
  88. const hexText = text.toUpperCase().startsWith('0X') ? text.slice(2) : text
  89. if (/^[0-9A-F]+$/i.test(hexText)) {
  90. const parsedHex = parseInt(hexText, 16)
  91. return Number.isFinite(parsedHex) ? clampInteger(parsedHex, 0, MAX_STORAGE_ADDRESS, fallback) : fallback
  92. }
  93. const numberValue = Number(text)
  94. return Number.isFinite(numberValue) ? clampInteger(numberValue, 0, MAX_STORAGE_ADDRESS, fallback) : fallback
  95. }
  96. function parseConfigAddress(value, options = {}) {
  97. const maxAddress = Number.isFinite(Number(options.maxAddress)) ? Number(options.maxAddress) : MAX_MODBUS_ADDRESS
  98. const maxHexLength = maxAddress > MAX_MODBUS_ADDRESS ? 8 : 4
  99. const label = options.label || '寄存器起始地址'
  100. if (typeof value === 'number') {
  101. return clampInteger(value, 0, maxAddress, 0)
  102. }
  103. const text = String(value === undefined || value === null ? '' : value).trim()
  104. const hexText = text.toUpperCase().startsWith('0X') ? text.slice(2) : text
  105. if (!new RegExp(`^[0-9A-F]{1,${maxHexLength}}$`, 'i').test(hexText)) {
  106. throw new Error(`${label}无效`)
  107. }
  108. return parseInt(hexText, 16)
  109. }
  110. function parseConfigQuantity(value, maxQuantity) {
  111. const text = String(value === undefined || value === null ? '' : value).trim()
  112. const quantity = Number(text)
  113. if (!Number.isInteger(quantity) || quantity < 1 || quantity > maxQuantity) {
  114. throw new Error(`寄存器数量需为 1 - ${maxQuantity}`)
  115. }
  116. return quantity
  117. }
  118. function getRegisterType(typeKey) {
  119. return REGISTER_TYPE_OPTIONS.find((item) => item.key === typeKey) || REGISTER_TYPE_OPTIONS[0]
  120. }
  121. function getRegisterTypeIndex(typeKey) {
  122. return Math.max(0, REGISTER_TYPE_OPTIONS.findIndex((item) => item.key === getRegisterType(typeKey).key))
  123. }
  124. function isStructLayout(layout) {
  125. return layout === GROUP_LAYOUT_STRUCT
  126. }
  127. function padWordHex(value) {
  128. const numberValue = Number(value || 0)
  129. const length = numberValue > MAX_MODBUS_ADDRESS ? 8 : 4
  130. return numberValue.toString(16).toUpperCase().padStart(length, '0')
  131. }
  132. function normalizeStorageAddressWidth(value, address = 0) {
  133. const numberValue = Number(value)
  134. if (numberValue === 16 || numberValue === 2) return 16
  135. if (numberValue === 32 || numberValue === 4) return 32
  136. return Number(address) > MAX_MODBUS_ADDRESS ? 32 : 16
  137. }
  138. function resolveStorageAddressWidth(source = {}, address = 0) {
  139. const codeInfoContext = source.codeInfoContext || {}
  140. const explicitWidth = source.sourceAddressWidth
  141. || source.storageAddressWidth
  142. || source.addressWidth
  143. || codeInfoContext.storageAddressWidth
  144. || codeInfoContext.addressWidth
  145. || codeInfoContext.codeInfoAddressWidth
  146. const explicitByteLength = source.sourceAddressByteLength
  147. || source.addressByteLength
  148. || codeInfoContext.sourceAddressByteLength
  149. || codeInfoContext.addressByteLength
  150. if (explicitWidth) return normalizeStorageAddressWidth(explicitWidth, address)
  151. if (explicitByteLength) return normalizeStorageAddressWidth(explicitByteLength, address)
  152. const memoryArea = String(source.sourceMemoryArea || '').trim().toUpperCase()
  153. if (memoryArea === 'ADDR32' || memoryArea === 'ADDRESS32') return 32
  154. return normalizeStorageAddressWidth(0, address)
  155. }
  156. function getStorageHexWidth(addressWidth, address = 0) {
  157. return normalizeStorageAddressWidth(addressWidth, address) === 16 ? 4 : 8
  158. }
  159. function getStorageAddressMax(addressWidth, address = 0) {
  160. return normalizeStorageAddressWidth(addressWidth, address) === 16
  161. ? MAX_MODBUS_ADDRESS
  162. : MAX_STORAGE_ADDRESS
  163. }
  164. function padStorageHex(value, addressWidth = 0) {
  165. const numberValue = Math.max(0, Math.floor(Number(value) || 0))
  166. return numberValue.toString(16).toUpperCase().padStart(getStorageHexWidth(addressWidth, numberValue), '0')
  167. }
  168. function formatAddressRange(startAddress, wordCount) {
  169. const address = normalizeAddress(startAddress, 0)
  170. const count = Math.max(1, Number(wordCount) || 1)
  171. const endAddress = address + count - 1
  172. const safeEndAddress = Math.min(endAddress, MAX_MODBUS_ADDRESS)
  173. const overflowText = endAddress > MAX_MODBUS_ADDRESS ? '+' : ''
  174. if (count <= 1) return `0x${padWordHex(address)}`
  175. return `0x${padWordHex(address)}-0x${padWordHex(safeEndAddress)}${overflowText}`
  176. }
  177. function formatStorageAddressRange(startAddress, byteCount, addressWidth = 0) {
  178. const address = normalizeStorageAddress(startAddress, 0)
  179. const count = Math.max(1, Number(byteCount) || 1)
  180. const addressMax = getStorageAddressMax(addressWidth, address)
  181. const endAddress = address + count - 1
  182. const safeEndAddress = Math.min(endAddress, addressMax)
  183. const overflowText = endAddress > addressMax ? '+' : ''
  184. if (count <= 1) return `0x${padStorageHex(address, addressWidth)}${overflowText}`
  185. return `0x${padStorageHex(address, addressWidth)}-0x${padStorageHex(safeEndAddress, addressWidth)}${overflowText}`
  186. }
  187. function isByteAddressedGroup(group = {}) {
  188. const memoryArea = String(group.sourceMemoryArea || '').trim().toUpperCase()
  189. return group.addressUnit === 'byte'
  190. || group.addressUnit === 'bytes'
  191. || BYTE_ADDRESS_MEMORY_AREAS.indexOf(memoryArea) >= 0
  192. }
  193. function formatRegisterAddressText(address, byteOffset, byteLength, registerType) {
  194. if (isBitRegisterType(registerType)) return `0x${padHex(address)}`
  195. if (byteLength === 1) return `0x${padHex(address)}${byteOffset === 0 ? 'H' : 'L'}`
  196. return `0x${padHex(address)}`
  197. }
  198. function formatBitFieldAddressText(address, byteOffset, bitOffset, bitWidth) {
  199. const byteText = formatRegisterAddressText(address, byteOffset, 1, DEFAULT_REGISTER_TYPE)
  200. const startBit = normalizeBitOffset(bitOffset)
  201. const endBit = startBit + normalizeBitWidth(bitWidth) - 1
  202. return endBit === startBit
  203. ? `${byteText}.b${startBit}`
  204. : `${byteText}.b${startBit}..${endBit}`
  205. }
  206. function isAddressRangeOverflow(startAddress, wordCount) {
  207. const address = normalizeAddress(startAddress, 0)
  208. const count = Math.max(1, Number(wordCount) || 1)
  209. return address + count - 1 > MAX_MODBUS_ADDRESS
  210. }
  211. function isStorageAddressRangeOverflow(startAddress, byteCount, addressWidth = 0) {
  212. const address = normalizeStorageAddress(startAddress, 0)
  213. const count = Math.max(1, Number(byteCount) || 1)
  214. const addressMax = getStorageAddressMax(addressWidth, address)
  215. return address + count - 1 > addressMax
  216. }
  217. function getMaxQuantity() {
  218. return MAX_PARAMETER_GROUP_ITEMS
  219. }
  220. function getRegisterSavedValue(register) {
  221. if (register.inputValue !== undefined && register.inputValue !== null) return normalizeTextValue(register.inputValue)
  222. if (register.value !== undefined && register.value !== null) return normalizeTextValue(register.value)
  223. return null
  224. }
  225. function normalizeRegisterDataType(register, registerType) {
  226. if (isBitRegisterType(registerType)) return DEFAULT_DATA_TYPE
  227. return getDataType(register.dataType || register.type || DEFAULT_DATA_TYPE).key
  228. }
  229. function createRegisterSourceMetaText(register) {
  230. const bitText = isBitFieldRegister(register)
  231. ? `bit${normalizeBitOffset(register.bitOffset)}:${normalizeBitWidth(register.bitWidth)}`
  232. : ''
  233. const sourceByteLength = Number(register.sourceByteLength)
  234. const sourceByteLengthText = Number.isFinite(sourceByteLength) && sourceByteLength > 0
  235. ? `${Math.floor(sourceByteLength)}B`
  236. : ''
  237. const parts = [
  238. register.sourceMemoryArea,
  239. register.sourceAddressText,
  240. sourceByteLengthText,
  241. bitText,
  242. register.sourceSymbolType && register.sourceSymbolType !== '---' ? register.sourceSymbolType : ''
  243. ].filter(Boolean)
  244. return parts.join(' · ')
  245. }
  246. function createGroupSourceMetaText(group) {
  247. const parts = [
  248. group.sourceMemoryArea,
  249. group.sourceAddressText,
  250. group.sourceSymbolName,
  251. group.sourceSegmentModule
  252. ].filter(Boolean)
  253. return parts.join(' · ')
  254. }
  255. function formatCompactNumber(value, precision = 4) {
  256. const text = formatFixedValue(value, precision)
  257. return text === '--' ? '--' : text.replace(/(\.\d*?)0+$/, '$1').replace(/\.$/, '')
  258. }
  259. function formatCardMetricValue(value) {
  260. if (value === '' || value === null || value === undefined) return ''
  261. return formatCompactNumber(value)
  262. }
  263. function readOptionalNumber(value) {
  264. if (value === '' || value === null || value === undefined) return null
  265. const numberValue = Number(value)
  266. return Number.isFinite(numberValue) ? numberValue : null
  267. }
  268. function addMetricItem(items, value, unit = '') {
  269. const text = formatCardMetricValue(value)
  270. if (!text) return
  271. items.push({
  272. text: `${text}${unit}`
  273. })
  274. }
  275. function assignOptionalNumber(target, key, value) {
  276. if (value === null || value === undefined) return
  277. target[key] = value
  278. }
  279. function normalizeMemoryEndian(value) {
  280. const text = String(value || '').trim().toLowerCase()
  281. if (text === 'little' || text === 'le' || text === '1') return 'little'
  282. return 'big'
  283. }
  284. function normalizeStorageCodeInfoCard(codeInfo = null) {
  285. const hasCodeInfo = !!codeInfo && codeInfo.hasCodeInfo !== false
  286. const refVolt = hasCodeInfo ? readOptionalNumber(codeInfo.refVolt) : null
  287. const refVoltRaw = hasCodeInfo ? readOptionalNumber(codeInfo.refVoltRaw) : null
  288. const card = {
  289. alongDiv: hasCodeInfo ? readOptionalNumber(codeInfo.alongDiv) : null,
  290. ampGain: hasCodeInfo ? readOptionalNumber(codeInfo.ampGain) : null,
  291. busDiv: hasCodeInfo ? readOptionalNumber(codeInfo.busDiv) : null,
  292. caveFreq: hasCodeInfo ? readOptionalNumber(codeInfo.caveFreq) : null,
  293. chipModel: hasCodeInfo ? String(codeInfo.chipModel || '').trim() : '',
  294. maxPacketLength: hasCodeInfo ? (Number(codeInfo.maxPacketLength) || 0) : 0,
  295. memoryEndian: hasCodeInfo ? normalizeMemoryEndian(codeInfo.memoryEndian) : 'big',
  296. memoryEndianRaw: hasCodeInfo ? (Number(codeInfo.memoryEndianRaw) || 0) : 0,
  297. model: hasCodeInfo ? String(codeInfo.model || '').trim() : '',
  298. refVolt: hasCodeInfo
  299. ? (refVolt !== null ? refVolt : (refVoltRaw !== null ? refVoltRaw / 10 : null))
  300. : null,
  301. refVoltRaw: refVoltRaw === null ? 0 : refVoltRaw,
  302. rsShunt: hasCodeInfo ? readOptionalNumber(codeInfo.rsShunt) : null,
  303. addressWidth: hasCodeInfo
  304. ? (Number(codeInfo.addressWidth || codeInfo.codeInfoAddressWidth || codeInfo.storageAddressWidth) || 0)
  305. : 0
  306. }
  307. const codeInfoContext = hasCodeInfo
  308. ? {
  309. addressWidth: card.addressWidth,
  310. codeInfoAddressWidth: card.addressWidth,
  311. maxPacketLength: card.maxPacketLength,
  312. memoryEndian: card.memoryEndian,
  313. storageAddressWidth: card.addressWidth
  314. }
  315. : {}
  316. if (hasCodeInfo) {
  317. assignOptionalNumber(codeInfoContext, 'alongDiv', card.alongDiv)
  318. assignOptionalNumber(codeInfoContext, 'ampGain', card.ampGain)
  319. assignOptionalNumber(codeInfoContext, 'busDiv', card.busDiv)
  320. assignOptionalNumber(codeInfoContext, 'caveFreq', card.caveFreq)
  321. assignOptionalNumber(codeInfoContext, 'refVolt', card.refVolt)
  322. assignOptionalNumber(codeInfoContext, 'rsShunt', card.rsShunt)
  323. }
  324. const metricItems = []
  325. addMetricItem(metricItems, card.caveFreq, 'KHz')
  326. addMetricItem(metricItems, card.refVolt, 'V')
  327. addMetricItem(metricItems, card.ampGain)
  328. addMetricItem(metricItems, card.rsShunt, 'mΩ')
  329. return {
  330. ...card,
  331. codeInfoContext,
  332. hasCodeInfo,
  333. metricItems
  334. }
  335. }
  336. function isStorageStructGroup(group = {}) {
  337. return isStructLayout(group.layout)
  338. && isByteAddressedGroup(group)
  339. && !!String(group.sourceMemoryArea || '').trim()
  340. }
  341. function isStorageMemoryGroup(group = {}) {
  342. return isByteAddressedGroup(group)
  343. && !!String(group.sourceMemoryArea || '').trim()
  344. }
  345. function isParameterGroupPollEnabled(group = {}) {
  346. return group.pollEnabled !== false
  347. }
  348. function getStorageAreaText(group = {}, lowercase = false) {
  349. const area = String(group.sourceMemoryArea || '').trim()
  350. return lowercase ? area.toLowerCase() : area.toUpperCase()
  351. }
  352. function stripStorageAreaPrefix(text) {
  353. const source = String(text || '').trim()
  354. const cleaned = source.replace(/^(?:IDATA|XDATA|DATA|CODE)[\s:_-]+/i, '').trim()
  355. return cleaned || source
  356. }
  357. function formatRegisterStartAddressText(address) {
  358. return `0x${padWordHex(address)}`
  359. }
  360. function formatStorageStartAddressText(address, addressWidth = 0) {
  361. return `0x${padStorageHex(address, addressWidth)}`
  362. }
  363. function formatStartAddressText(address, maxAddress = MAX_MODBUS_ADDRESS) {
  364. return `0x${padHex(address, maxAddress > MAX_MODBUS_ADDRESS ? 8 : 4)}`
  365. }
  366. function formatStorageHexBytes(bytes = [], byteLength = 1) {
  367. const safeLength = Math.max(1, Math.floor(Number(byteLength) || 1))
  368. const source = Array.isArray(bytes) ? bytes : []
  369. return `0x${Array.from({ length: safeLength }, (_, index) => (
  370. (Number(source[index] || 0) & 0xFF).toString(16).toUpperCase().padStart(2, '0')
  371. )).join('')}`
  372. }
  373. function formatStorageRegisterHexValue(register, rawBytes = []) {
  374. if (isTextRegister(register.dataType)) {
  375. return formatRawByteTextWithDefault(rawBytes, register.byteLength)
  376. }
  377. return formatStorageHexBytes(rawBytes, register.byteLength)
  378. }
  379. function normalizeRegister(register, group, index, address, byteOffset = 0) {
  380. const registerType = getRegisterType(group.registerType).key
  381. const layout = isStructLayout(group.layout) ? GROUP_LAYOUT_STRUCT : GROUP_LAYOUT_REGISTER
  382. const isStorageMemory = isStorageMemoryGroup(group)
  383. const byteAddressed = isByteAddressedGroup(group)
  384. const dataType = normalizeRegisterDataType(register, registerType)
  385. const bitOffset = normalizeBitOffset(register.bitOffset)
  386. const bitWidth = normalizeBitWidth(register.bitWidth)
  387. const isBitField = !isBitRegisterType(registerType) && isBitFieldRegister(register)
  388. const isPlaceholderByteField = !!register.isPlaceholderByteField
  389. const textByteLength = isTextRegister(dataType)
  390. ? normalizeTextByteLength(register.textByteLength, DEFAULT_TEXT_BYTE_LENGTH)
  391. : ''
  392. const defaultValue = normalizeTextValue(register.defaultValue)
  393. const savedValue = getRegisterSavedValue(register)
  394. const inputValue = savedValue === null ? defaultValue : savedValue
  395. const rawValue = register.rawValue === undefined ? null : register.rawValue
  396. const memoryEndian = isStorageMemory ? normalizeMemoryEndian(register.memoryEndian || group.codeInfoContext && group.codeInfoContext.memoryEndian) : 'big'
  397. const storageAddressWidth = isStorageMemory
  398. ? resolveStorageAddressWidth({
  399. ...group,
  400. sourceAddressByteLength: register.sourceAddressByteLength || group.sourceAddressByteLength,
  401. sourceAddressWidth: register.sourceAddressWidth || group.sourceAddressWidth,
  402. sourceMemoryArea: register.sourceMemoryArea || group.sourceMemoryArea
  403. }, address)
  404. : 0
  405. const byteLength = isBitRegisterType(registerType)
  406. ? 1
  407. : getRegisterByteLength(dataType, { ...register, bitOffset, bitWidth, isBitField, layout, textByteLength })
  408. const registerCount = isBitRegisterType(registerType)
  409. ? 1
  410. : getRegisterWordCountAtOffset(dataType, byteOffset, { ...register, bitOffset, bitWidth, isBitField, layout, textByteLength })
  411. const canShowUnit = !isBitRegisterType(registerType) && !isBitField && supportsUnit(dataType)
  412. const rawWords = Array.isArray(register.rawWords)
  413. ? register.rawWords.slice(0, registerCount).map((word) => Number(word) & 0xFFFF)
  414. : []
  415. const rawBytes = Array.isArray(register.rawBytes)
  416. ? register.rawBytes.slice(0, byteLength).map((byte) => Number(byte) & 0xFF)
  417. : []
  418. const defaultRawValueText = rawValue === null
  419. ? '--'
  420. : (isBitRegisterType(registerType)
  421. ? formatCoilDisplayValue(rawValue)
  422. : (byteAddressed ? formatRawByteText(rawBytes) : formatRawWordText(rawWords)))
  423. const rawValueText = isStorageMemory
  424. ? formatStorageRegisterHexValue({
  425. ...register,
  426. byteLength,
  427. dataType
  428. }, rawBytes)
  429. : defaultRawValueText
  430. const displayValue = rawValue === null
  431. ? (inputValue.trim() ? inputValue : '--')
  432. : formatRegisterValue({ ...register, dataType, byteOffset, memoryEndian }, rawValue)
  433. const conversionFormula = normalizeTextValue(register.conversionFormula).trim()
  434. const conversionResult = conversionFormula && rawValue !== null
  435. ? evaluateValueFormula(conversionFormula, rawValue, group.codeInfoContext || {})
  436. : null
  437. const convertedDisplayValue = conversionResult && conversionResult.ok
  438. ? conversionResult.text
  439. : displayValue
  440. const displayMetaText = conversionResult && conversionResult.ok
  441. ? `raw ${displayValue}`
  442. : ''
  443. const addressRangeText = isBitRegisterType(registerType)
  444. ? `0x${padHex(address)}`
  445. : (byteAddressed
  446. ? (isStorageMemory
  447. ? formatStorageAddressRange(address, Math.max(1, byteLength), storageAddressWidth)
  448. : formatAddressRange(address, Math.max(1, byteLength)))
  449. : formatAddressRange(address, registerCount))
  450. const addressText = isBitField
  451. ? (byteAddressed
  452. ? `${isStorageMemory
  453. ? formatStorageAddressRange(address, Math.max(1, byteLength), storageAddressWidth)
  454. : formatAddressRange(address, Math.max(1, byteLength))}.b${bitOffset}${bitWidth > 1 ? `..b${bitOffset + bitWidth - 1}` : ''}`
  455. : formatBitFieldAddressText(address, byteOffset, bitOffset, bitWidth))
  456. : (byteAddressed
  457. ? (isStorageMemory
  458. ? formatStorageAddressRange(address, Math.max(1, byteLength), storageAddressWidth)
  459. : formatAddressRange(address, Math.max(1, byteLength)))
  460. : formatRegisterAddressText(address, byteOffset, byteLength, registerType))
  461. const registerStartAddressText = isStorageMemory
  462. ? formatStorageStartAddressText(address, storageAddressWidth)
  463. : formatRegisterStartAddressText(address)
  464. const metaText = isStorageMemory
  465. ? `${registerStartAddressText} ${rawValueText}`
  466. : `${addressText} ${rawValueText}`.trim()
  467. return {
  468. address,
  469. addressRangeText,
  470. addressText,
  471. bitOffset: isBitField ? bitOffset : '',
  472. bitWidth: isBitField ? bitWidth : '',
  473. byteLength,
  474. byteLengthText: isBitRegisterType(registerType)
  475. ? '1bit'
  476. : (isBitField
  477. ? (bitWidth === 1 ? `1bit/占${byteLength}B` : `${bitWidth}bit/占${byteLength}B`)
  478. : (textByteLength && textByteLength !== byteLength ? `${textByteLength}B/占${byteLength}B` : `${byteLength}B`)),
  479. byteStart: Math.max(0, Math.floor(Number(register.byteStart) || 0)),
  480. dataType,
  481. dataTypeIndex: getDataTypeIndex(dataType),
  482. dataTypeText: getRegisterValueTypeLabel(dataType),
  483. defaultValue,
  484. displayMetaText,
  485. displayValue: convertedDisplayValue,
  486. id: register.id || createId('gm-reg'),
  487. inputType: isTextRegister(dataType) ? 'text' : 'text',
  488. inputValue,
  489. isStructField: layout === GROUP_LAYOUT_STRUCT || !!register.isStructField,
  490. layout,
  491. isBitField,
  492. isPlaceholderByteField,
  493. isDirty: !!register.isDirty,
  494. maxValue: normalizeTextValue(register.maxValue),
  495. metaText,
  496. minValue: normalizeTextValue(register.minValue),
  497. memoryEndian,
  498. name: register.name || `寄存器 ${index + 1}`,
  499. conversionFormula,
  500. conversionFormulaErrorText: conversionResult && conversionResult.errorText ? conversionResult.errorText : '',
  501. rawValue,
  502. rawValueText,
  503. rawBytes,
  504. rawWords,
  505. registerCount,
  506. byteOffset,
  507. registerType,
  508. showDataType: !isBitRegisterType(registerType),
  509. showRange: !isBitRegisterType(registerType) && !isBitField && supportsRange(dataType),
  510. showTextLength: !isBitRegisterType(registerType) && isTextRegister(dataType),
  511. showUnit: canShowUnit,
  512. textByteLength,
  513. unit: canShowUnit ? normalizeTextValue(register.unit).trim() : '',
  514. structByteLength: register.structByteLength,
  515. remark: register.remark || '',
  516. ...pickFields(register, SOURCE_REGISTER_FIELDS),
  517. sourceMetaText: createRegisterSourceMetaText(register)
  518. }
  519. }
  520. function normalizeGroup(group) {
  521. const registerType = getRegisterType(group.registerType || group.type || DEFAULT_REGISTER_TYPE)
  522. const sourceMemoryArea = String(group.sourceMemoryArea || '').trim()
  523. const isStorageGroupSource = !!sourceMemoryArea || group.addressUnit === 'byte' || group.addressUnit === 'bytes'
  524. const startAddress = isStorageGroupSource
  525. ? normalizeStorageAddress(
  526. group.sourceAddress !== undefined && group.sourceAddress !== null && group.sourceAddress !== ''
  527. ? group.sourceAddress
  528. : group.startAddress,
  529. 0
  530. )
  531. : normalizeAddress(group.startAddress, 0)
  532. const maxQuantity = getMaxQuantity(registerType.key)
  533. const sourceRegisters = Array.isArray(group.registers) ? group.registers : []
  534. const codeInfoContext = group.codeInfoContext || {}
  535. const layout = isStructLayout(group.layout) ? GROUP_LAYOUT_STRUCT : GROUP_LAYOUT_REGISTER
  536. const byteAddressed = isByteAddressedGroup(group)
  537. const hasExplicitQuantity = group.quantity !== undefined && group.quantity !== null && group.quantity !== ''
  538. const quantity = hasExplicitQuantity
  539. ? clampInteger(group.quantity, 1, maxQuantity, 1)
  540. : clampInteger(sourceRegisters.length || 1, 1, maxQuantity, 1)
  541. const baseGroup = {
  542. deleteVisible: !!group.deleteVisible,
  543. expanded: group.expanded === true,
  544. id: group.id || createId('gm-group'),
  545. isStructLayout: layout === GROUP_LAYOUT_STRUCT,
  546. layout,
  547. name: String(group.name || group.groupName || '寄存器组').trim() || '寄存器组',
  548. pollEnabled: group.pollEnabled === false ? false : true,
  549. quantity,
  550. registerType: registerType.key,
  551. startAddress,
  552. touchStartX: 0,
  553. codeInfoContext,
  554. ...pickFields(group, SOURCE_GROUP_FIELDS)
  555. }
  556. const registers = []
  557. let nextAddress = startAddress
  558. let nextByteOffset = 0
  559. for (let index = 0; index < quantity; index += 1) {
  560. const sourceRegister = sourceRegisters[index] || {}
  561. let normalizedSourceRegister = sourceRegister
  562. const dataType = normalizeRegisterDataType(sourceRegister, baseGroup.registerType)
  563. const textByteLength = isTextRegister(dataType)
  564. ? normalizeTextByteLength(sourceRegister.textByteLength, DEFAULT_TEXT_BYTE_LENGTH)
  565. : ''
  566. const isBitRegister = isBitRegisterType(baseGroup.registerType)
  567. let address = startAddress + index
  568. let byteOffset = 0
  569. if (!isBitRegister) {
  570. const explicitByteStart = Number(sourceRegister.byteStart)
  571. const hasExplicitByteStart = layout === GROUP_LAYOUT_STRUCT && Number.isFinite(explicitByteStart)
  572. const byteLength = getRegisterByteLength(dataType, { ...sourceRegister, layout, textByteLength })
  573. if (layout !== GROUP_LAYOUT_STRUCT && !isByteRegister(dataType) && nextByteOffset % 2 !== 0) {
  574. nextByteOffset += 1
  575. }
  576. const currentByteStart = hasExplicitByteStart
  577. ? Math.max(0, Math.floor(explicitByteStart))
  578. : nextByteOffset
  579. address = byteAddressed ? startAddress + currentByteStart : startAddress + Math.floor(currentByteStart / 2)
  580. byteOffset = byteAddressed ? 0 : currentByteStart % 2
  581. normalizedSourceRegister = {
  582. ...sourceRegister,
  583. byteStart: currentByteStart
  584. }
  585. nextByteOffset = Math.max(nextByteOffset, currentByteStart + byteLength)
  586. }
  587. const register = normalizeRegister(normalizedSourceRegister, baseGroup, index, address, byteOffset)
  588. registers.push(register)
  589. if (isBitRegister) nextAddress += register.registerCount
  590. }
  591. const byteLength = isBitRegisterType(baseGroup.registerType)
  592. ? Math.max(1, nextAddress - startAddress)
  593. : Math.max(1, nextByteOffset)
  594. const paddedByteLength = isBitRegisterType(baseGroup.registerType)
  595. ? byteLength
  596. : (byteAddressed ? byteLength : alignEvenByteLength(byteLength))
  597. const wordQuantity = isBitRegisterType(baseGroup.registerType)
  598. ? Math.max(1, nextAddress - startAddress)
  599. : (byteAddressed ? Math.max(1, byteLength) : Math.max(1, paddedByteLength / 2))
  600. const addressSpan = byteAddressed ? Math.max(1, byteLength) : wordQuantity
  601. const storageMemory = isStorageMemoryGroup(baseGroup)
  602. const storageAddressWidth = storageMemory ? resolveStorageAddressWidth(baseGroup, startAddress) : 0
  603. const storageAddressMax = getStorageAddressMax(storageAddressWidth, startAddress)
  604. const addressOverflow = storageMemory
  605. ? isStorageAddressRangeOverflow(startAddress, addressSpan, storageAddressWidth)
  606. : isAddressRangeOverflow(startAddress, addressSpan)
  607. const endAddress = startAddress + addressSpan - 1
  608. const addressMax = storageMemory ? storageAddressMax : MAX_MODBUS_ADDRESS
  609. const startAddressText = storageMemory
  610. ? formatStorageStartAddressText(startAddress, storageAddressWidth)
  611. : formatStartAddressText(startAddress, addressMax)
  612. const endAddressText = storageMemory
  613. ? (addressOverflow ? `0x${padStorageHex(addressMax, storageAddressWidth)}+` : `0x${padStorageHex(endAddress, storageAddressWidth)}`)
  614. : (addressOverflow ? `0x${padWordHex(addressMax)}+` : `0x${padWordHex(endAddress)}`)
  615. const displayName = storageMemory
  616. ? stripStorageAreaPrefix(baseGroup.name)
  617. : baseGroup.name
  618. const listMetaText = storageMemory
  619. ? `${getStorageAreaText(baseGroup)} ${startAddressText}-${endAddressText} ${byteLength}B`
  620. : `${formatAddressRange(startAddress, addressSpan)} · ${quantity}/${wordQuantity}${createGroupSourceMetaText(baseGroup) ? ` · ${createGroupSourceMetaText(baseGroup)}` : ''}${addressOverflow ? ' · 地址超出 0xFFFF' : ''}`
  621. const detailMetaText = storageMemory
  622. ? `${getStorageAreaText(baseGroup, true)} ${startAddressText} ${quantity}/${byteLength}B`
  623. : `${formatAddressRange(startAddress, addressSpan)} · ${quantity}/${wordQuantity}${createGroupSourceMetaText(baseGroup) ? ` · ${createGroupSourceMetaText(baseGroup)}` : ''}${addressOverflow ? ' · 地址超出 0xFFFF' : ''}`
  624. const sourceMetaText = createGroupSourceMetaText(baseGroup)
  625. return {
  626. ...baseGroup,
  627. addressRangeText: storageMemory
  628. ? formatStorageAddressRange(startAddress, addressSpan, storageAddressWidth)
  629. : formatAddressRange(startAddress, addressSpan),
  630. addressOverflow,
  631. addressWarningText: addressOverflow
  632. ? `地址超出 0x${storageMemory ? padStorageHex(addressMax, storageAddressWidth) : padWordHex(addressMax)}`
  633. : '',
  634. detailMetaText,
  635. detailTitleText: displayName,
  636. displayName,
  637. endAddressText,
  638. functionCode: registerType.functionCode,
  639. isReadOnly: !registerType.writable,
  640. byteLength,
  641. listMetaText,
  642. maxQuantity,
  643. registerTypeIndex: getRegisterTypeIndex(registerType.key),
  644. registerTypeText: registerType.label,
  645. registers,
  646. paddedByteLength,
  647. sourceMetaText,
  648. startAddressText,
  649. wordQuantity,
  650. writable: registerType.writable
  651. }
  652. }
  653. function normalizeGroupConfig(config = {}) {
  654. const registerType = config.registerTypeIndex !== undefined && config.registerTypeIndex !== null
  655. ? (REGISTER_TYPE_OPTIONS[Number(config.registerTypeIndex)] || REGISTER_TYPE_OPTIONS[0])
  656. : getRegisterType(config.registerType || config.type || DEFAULT_REGISTER_TYPE)
  657. const maxQuantity = getMaxQuantity(registerType.key)
  658. const isStorageConfig = !!String(config.sourceMemoryArea || '').trim()
  659. || config.addressUnit === 'byte'
  660. || config.addressUnit === 'bytes'
  661. const storageAddressWidth = isStorageConfig ? resolveStorageAddressWidth(config, 0) : 0
  662. const storageAddressByteLength = storageAddressWidth === 16 ? 2 : 4
  663. return {
  664. ...(isStorageConfig ? {
  665. addressUnit: config.addressUnit || 'byte',
  666. sourceAddressByteLength: config.sourceAddressByteLength || storageAddressByteLength,
  667. sourceAddressWidth: storageAddressWidth,
  668. sourceMemoryArea: config.sourceMemoryArea || 'XDATA'
  669. } : {}),
  670. layout: isStructLayout(config.layout) ? GROUP_LAYOUT_STRUCT : GROUP_LAYOUT_REGISTER,
  671. name: String(config.name || config.groupName || '寄存器组').trim() || '寄存器组',
  672. pollEnabled: config.pollEnabled === false ? false : true,
  673. quantity: parseConfigQuantity(config.quantity, maxQuantity),
  674. registerType: registerType.key,
  675. startAddress: parseConfigAddress(config.startAddress, {
  676. label: isStorageConfig ? '内存起始地址' : '寄存器起始地址',
  677. maxAddress: isStorageConfig ? getStorageAddressMax(storageAddressWidth) : MAX_MODBUS_ADDRESS
  678. })
  679. }
  680. }
  681. function getRegisterJsonValue(register) {
  682. if (register.inputValue !== undefined && register.inputValue !== null) {
  683. return normalizeTextValue(register.inputValue)
  684. }
  685. if (register.defaultValue !== undefined && register.defaultValue !== null) {
  686. return normalizeTextValue(register.defaultValue)
  687. }
  688. if (register.displayValue !== undefined && register.displayValue !== null && register.displayValue !== '--') {
  689. return normalizeTextValue(register.displayValue)
  690. }
  691. return ''
  692. }
  693. function normalizeImportedRegisterDataType(register) {
  694. const dataType = register.dataType || register.type || DEFAULT_DATA_TYPE
  695. return getDataType(dataType).key
  696. }
  697. function cloneImportedGroup(group) {
  698. return {
  699. layout: isStructLayout(group.layout) ? GROUP_LAYOUT_STRUCT : GROUP_LAYOUT_REGISTER,
  700. name: group.name,
  701. pollEnabled: group.pollEnabled === false ? false : true,
  702. quantity: group.quantity,
  703. registerType: group.registerType || group.type || DEFAULT_REGISTER_TYPE,
  704. registers: (Array.isArray(group.registers) ? group.registers : []).map((register) => ({
  705. conversionFormula: register.conversionFormula,
  706. dataType: normalizeImportedRegisterDataType(register),
  707. defaultValue: register.defaultValue,
  708. inputValue: register.inputValue,
  709. maxValue: register.maxValue,
  710. minValue: register.minValue,
  711. name: register.name,
  712. isStructField: !!register.isStructField,
  713. textByteLength: register.textByteLength,
  714. remark: register.remark,
  715. unit: register.unit,
  716. value: register.value,
  717. ...pickFields(register, STRUCT_REGISTER_FIELDS),
  718. ...pickFields(register, SOURCE_REGISTER_FIELDS)
  719. })),
  720. startAddress: group.startAddress,
  721. ...pickFields(group, SOURCE_GROUP_FIELDS)
  722. }
  723. }
  724. module.exports = {
  725. DATA_TYPE_OPTIONS,
  726. DEFAULT_DATA_TYPE,
  727. DEFAULT_REGISTER_TYPE,
  728. GROUP_LAYOUT_REGISTER,
  729. GROUP_LAYOUT_STRUCT,
  730. MAX_MODBUS_ADDRESS,
  731. MAX_STORAGE_ADDRESS,
  732. REGISTER_TYPE_OPTIONS,
  733. cloneImportedGroup,
  734. decodeRegisterFromByteCache,
  735. decodeRegisterFromWordCache,
  736. decodeRegisterValue,
  737. formatCoilDisplayValue,
  738. formatRegisterValue,
  739. getDataType,
  740. getRegisterEncodedBytes,
  741. getRegisterEncodedWords,
  742. getGroupEncodedBytes,
  743. getGroupEncodedWords,
  744. getRegisterWordCount,
  745. getRegisterJsonValue,
  746. getRegisterBytesFromByteCache,
  747. getRegisterWordsFromByteCache,
  748. getRegisterWordsFromWordCache,
  749. getRegisterWriteValueText,
  750. isAddressRangeOverflow,
  751. isBitRegisterType,
  752. isByteRegister,
  753. isParameterGroupPollEnabled,
  754. isStorageMemoryGroup,
  755. isStorageStructGroup,
  756. isTextRegister,
  757. normalizeGroup,
  758. normalizeGroupConfig,
  759. normalizeStorageCodeInfoCard,
  760. normalizeRegister,
  761. parseCoilValue,
  762. registerTypeIsBit,
  763. splitWordSpans,
  764. validateRegisterValue
  765. }