1
0

model.js 33 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872
  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, fallback = 'little') {
  280. const text = String(value || '').trim().toLowerCase()
  281. if (text === 'little' || text === 'le' || text === '1') return 'little'
  282. if (text === 'big' || text === 'be' || text === '0') return 'big'
  283. return fallback
  284. }
  285. function normalizeStorageCodeInfoCard(codeInfo = null) {
  286. const hasCodeInfo = !!codeInfo && codeInfo.hasCodeInfo !== false
  287. const refVolt = hasCodeInfo ? readOptionalNumber(codeInfo.refVolt) : null
  288. const refVoltRaw = hasCodeInfo ? readOptionalNumber(codeInfo.refVoltRaw) : null
  289. const card = {
  290. alongDiv: hasCodeInfo ? readOptionalNumber(codeInfo.alongDiv) : null,
  291. ampGain: hasCodeInfo ? readOptionalNumber(codeInfo.ampGain) : null,
  292. busDiv: hasCodeInfo ? readOptionalNumber(codeInfo.busDiv) : null,
  293. caveFreq: hasCodeInfo ? readOptionalNumber(codeInfo.caveFreq) : null,
  294. chipModel: hasCodeInfo ? String(codeInfo.chipModel || '').trim() : '',
  295. maxPacketLength: hasCodeInfo ? (Number(codeInfo.maxPacketLength) || 0) : 0,
  296. memoryEndian: hasCodeInfo ? normalizeMemoryEndian(codeInfo.memoryEndian) : 'little',
  297. memoryEndianRaw: hasCodeInfo ? (Number(codeInfo.memoryEndianRaw) || 0) : 0,
  298. model: hasCodeInfo ? String(codeInfo.model || '').trim() : '',
  299. refVolt: hasCodeInfo
  300. ? (refVolt !== null ? refVolt : (refVoltRaw !== null ? refVoltRaw / 10 : null))
  301. : null,
  302. refVoltRaw: refVoltRaw === null ? 0 : refVoltRaw,
  303. rsShunt: hasCodeInfo ? readOptionalNumber(codeInfo.rsShunt) : null,
  304. addressWidth: hasCodeInfo
  305. ? (Number(codeInfo.addressWidth || codeInfo.codeInfoAddressWidth || codeInfo.storageAddressWidth) || 0)
  306. : 0
  307. }
  308. const codeInfoContext = hasCodeInfo
  309. ? {
  310. addressWidth: card.addressWidth,
  311. codeInfoAddressWidth: card.addressWidth,
  312. maxPacketLength: card.maxPacketLength,
  313. memoryEndian: card.memoryEndian,
  314. memoryEndianRaw: card.memoryEndianRaw,
  315. storageAddressWidth: card.addressWidth
  316. }
  317. : {}
  318. if (hasCodeInfo) {
  319. assignOptionalNumber(codeInfoContext, 'alongDiv', card.alongDiv)
  320. assignOptionalNumber(codeInfoContext, 'ampGain', card.ampGain)
  321. assignOptionalNumber(codeInfoContext, 'busDiv', card.busDiv)
  322. assignOptionalNumber(codeInfoContext, 'caveFreq', card.caveFreq)
  323. assignOptionalNumber(codeInfoContext, 'refVolt', card.refVolt)
  324. assignOptionalNumber(codeInfoContext, 'rsShunt', card.rsShunt)
  325. }
  326. const metricItems = []
  327. addMetricItem(metricItems, card.caveFreq, 'KHz')
  328. addMetricItem(metricItems, card.refVolt, 'V')
  329. addMetricItem(metricItems, card.ampGain)
  330. addMetricItem(metricItems, card.rsShunt, 'mΩ')
  331. return {
  332. ...card,
  333. codeInfoContext,
  334. hasCodeInfo,
  335. metricItems
  336. }
  337. }
  338. function isStorageStructGroup(group = {}) {
  339. return isStructLayout(group.layout)
  340. && isByteAddressedGroup(group)
  341. && !!String(group.sourceMemoryArea || '').trim()
  342. }
  343. function isStorageMemoryGroup(group = {}) {
  344. return isByteAddressedGroup(group)
  345. && !!String(group.sourceMemoryArea || '').trim()
  346. }
  347. function isParameterGroupPollEnabled(group = {}) {
  348. return group.pollEnabled !== false
  349. }
  350. function getStorageAreaText(group = {}, lowercase = false) {
  351. const area = String(group.sourceMemoryArea || '').trim()
  352. return lowercase ? area.toLowerCase() : area.toUpperCase()
  353. }
  354. function stripStorageAreaPrefix(text) {
  355. const source = String(text || '').trim()
  356. const cleaned = source.replace(/^(?:IDATA|XDATA|DATA|CODE)[\s:_-]+/i, '').trim()
  357. return cleaned || source
  358. }
  359. function formatRegisterStartAddressText(address) {
  360. return `0x${padWordHex(address)}`
  361. }
  362. function formatStorageStartAddressText(address, addressWidth = 0) {
  363. return `0x${padStorageHex(address, addressWidth)}`
  364. }
  365. function formatStartAddressText(address, maxAddress = MAX_MODBUS_ADDRESS) {
  366. return `0x${padHex(address, maxAddress > MAX_MODBUS_ADDRESS ? 8 : 4)}`
  367. }
  368. function formatStorageHexBytes(bytes = [], byteLength = 1) {
  369. const safeLength = Math.max(1, Math.floor(Number(byteLength) || 1))
  370. const source = Array.isArray(bytes) ? bytes : []
  371. return `0x${Array.from({ length: safeLength }, (_, index) => (
  372. (Number(source[index] || 0) & 0xFF).toString(16).toUpperCase().padStart(2, '0')
  373. )).join('')}`
  374. }
  375. function formatStorageRegisterHexValue(register, rawBytes = []) {
  376. if (isTextRegister(register.dataType)) {
  377. return formatRawByteTextWithDefault(rawBytes, register.byteLength)
  378. }
  379. return formatStorageHexBytes(rawBytes, register.byteLength)
  380. }
  381. function normalizeRegister(register, group, index, address, byteOffset = 0) {
  382. const registerType = getRegisterType(group.registerType).key
  383. const layout = isStructLayout(group.layout) ? GROUP_LAYOUT_STRUCT : GROUP_LAYOUT_REGISTER
  384. const isStorageMemory = isStorageMemoryGroup(group)
  385. const byteAddressed = isByteAddressedGroup(group)
  386. const dataType = normalizeRegisterDataType(register, registerType)
  387. const bitOffset = normalizeBitOffset(register.bitOffset)
  388. const bitWidth = normalizeBitWidth(register.bitWidth)
  389. const isBitField = !isBitRegisterType(registerType) && isBitFieldRegister(register)
  390. const isPlaceholderByteField = !!register.isPlaceholderByteField
  391. const textByteLength = isTextRegister(dataType)
  392. ? normalizeTextByteLength(register.textByteLength, DEFAULT_TEXT_BYTE_LENGTH)
  393. : ''
  394. const defaultValue = normalizeTextValue(register.defaultValue)
  395. const savedValue = getRegisterSavedValue(register)
  396. const inputValue = savedValue === null ? defaultValue : savedValue
  397. const rawValue = register.rawValue === undefined ? null : register.rawValue
  398. const memoryEndian = isStorageMemory
  399. ? normalizeMemoryEndian(register.memoryEndian || group.codeInfoContext && group.codeInfoContext.memoryEndian)
  400. : 'big'
  401. const storageAddressWidth = isStorageMemory
  402. ? resolveStorageAddressWidth({
  403. ...group,
  404. sourceAddressByteLength: register.sourceAddressByteLength || group.sourceAddressByteLength,
  405. sourceAddressWidth: register.sourceAddressWidth || group.sourceAddressWidth,
  406. sourceMemoryArea: register.sourceMemoryArea || group.sourceMemoryArea
  407. }, address)
  408. : 0
  409. const byteLength = isBitRegisterType(registerType)
  410. ? 1
  411. : getRegisterByteLength(dataType, { ...register, bitOffset, bitWidth, isBitField, layout, textByteLength })
  412. const registerCount = isBitRegisterType(registerType)
  413. ? 1
  414. : getRegisterWordCountAtOffset(dataType, byteOffset, { ...register, bitOffset, bitWidth, isBitField, layout, textByteLength })
  415. const canShowUnit = !isBitRegisterType(registerType) && !isBitField && supportsUnit(dataType)
  416. const rawWords = Array.isArray(register.rawWords)
  417. ? register.rawWords.slice(0, registerCount).map((word) => Number(word) & 0xFFFF)
  418. : []
  419. const rawBytes = Array.isArray(register.rawBytes)
  420. ? register.rawBytes.slice(0, byteLength).map((byte) => Number(byte) & 0xFF)
  421. : []
  422. const defaultRawValueText = rawValue === null
  423. ? '--'
  424. : (isBitRegisterType(registerType)
  425. ? formatCoilDisplayValue(rawValue)
  426. : (byteAddressed ? formatRawByteText(rawBytes) : formatRawWordText(rawWords)))
  427. const rawValueText = isStorageMemory
  428. ? formatStorageRegisterHexValue({
  429. ...register,
  430. byteLength,
  431. dataType
  432. }, rawBytes)
  433. : defaultRawValueText
  434. const displayValue = rawValue === null
  435. ? (inputValue.trim() ? inputValue : '--')
  436. : formatRegisterValue({ ...register, dataType, byteOffset, memoryEndian }, rawValue)
  437. const conversionFormula = normalizeTextValue(register.conversionFormula).trim()
  438. const conversionResult = conversionFormula && rawValue !== null
  439. ? evaluateValueFormula(conversionFormula, rawValue, group.codeInfoContext || {})
  440. : null
  441. const convertedDisplayValue = conversionResult && conversionResult.ok
  442. ? conversionResult.text
  443. : displayValue
  444. const displayMetaText = conversionResult && conversionResult.ok
  445. ? `raw ${displayValue}`
  446. : ''
  447. const addressRangeText = isBitRegisterType(registerType)
  448. ? `0x${padHex(address)}`
  449. : (byteAddressed
  450. ? (isStorageMemory
  451. ? formatStorageAddressRange(address, Math.max(1, byteLength), storageAddressWidth)
  452. : formatAddressRange(address, Math.max(1, byteLength)))
  453. : formatAddressRange(address, registerCount))
  454. const addressText = isBitField
  455. ? (byteAddressed
  456. ? `${isStorageMemory
  457. ? formatStorageAddressRange(address, Math.max(1, byteLength), storageAddressWidth)
  458. : formatAddressRange(address, Math.max(1, byteLength))}.b${bitOffset}${bitWidth > 1 ? `..b${bitOffset + bitWidth - 1}` : ''}`
  459. : formatBitFieldAddressText(address, byteOffset, bitOffset, bitWidth))
  460. : (byteAddressed
  461. ? (isStorageMemory
  462. ? formatStorageAddressRange(address, Math.max(1, byteLength), storageAddressWidth)
  463. : formatAddressRange(address, Math.max(1, byteLength)))
  464. : formatRegisterAddressText(address, byteOffset, byteLength, registerType))
  465. const registerStartAddressText = isStorageMemory
  466. ? formatStorageStartAddressText(address, storageAddressWidth)
  467. : formatRegisterStartAddressText(address)
  468. const metaText = isStorageMemory
  469. ? `${registerStartAddressText} ${rawValueText}`
  470. : `${addressText} ${rawValueText}`.trim()
  471. return {
  472. address,
  473. addressRangeText,
  474. addressText,
  475. bitOffset: isBitField ? bitOffset : '',
  476. bitWidth: isBitField ? bitWidth : '',
  477. byteLength,
  478. byteLengthText: isBitRegisterType(registerType)
  479. ? '1bit'
  480. : (isBitField
  481. ? (bitWidth === 1 ? `1bit/占${byteLength}B` : `${bitWidth}bit/占${byteLength}B`)
  482. : (textByteLength && textByteLength !== byteLength ? `${textByteLength}B/占${byteLength}B` : `${byteLength}B`)),
  483. byteStart: Math.max(0, Math.floor(Number(register.byteStart) || 0)),
  484. dataType,
  485. dataTypeIndex: getDataTypeIndex(dataType),
  486. dataTypeText: getRegisterValueTypeLabel(dataType),
  487. defaultValue,
  488. displayMetaText,
  489. displayValue: convertedDisplayValue,
  490. id: register.id || createId('gm-reg'),
  491. inputType: isTextRegister(dataType) ? 'text' : 'text',
  492. inputValue,
  493. isStructField: layout === GROUP_LAYOUT_STRUCT || !!register.isStructField,
  494. layout,
  495. isBitField,
  496. isPlaceholderByteField,
  497. isDirty: !!register.isDirty,
  498. maxValue: normalizeTextValue(register.maxValue),
  499. metaText,
  500. minValue: normalizeTextValue(register.minValue),
  501. memoryEndian,
  502. name: register.name || `寄存器 ${index + 1}`,
  503. conversionFormula,
  504. conversionFormulaErrorText: conversionResult && conversionResult.errorText ? conversionResult.errorText : '',
  505. rawValue,
  506. rawValueText,
  507. rawBytes,
  508. rawWords,
  509. registerCount,
  510. byteOffset,
  511. registerType,
  512. showDataType: !isBitRegisterType(registerType),
  513. showRange: !isBitRegisterType(registerType) && !isBitField && supportsRange(dataType),
  514. showTextLength: !isBitRegisterType(registerType) && isTextRegister(dataType),
  515. showUnit: canShowUnit,
  516. textByteLength,
  517. unit: canShowUnit ? normalizeTextValue(register.unit).trim() : '',
  518. structByteLength: register.structByteLength,
  519. remark: register.remark || '',
  520. ...pickFields(register, SOURCE_REGISTER_FIELDS),
  521. sourceMetaText: createRegisterSourceMetaText(register)
  522. }
  523. }
  524. function normalizeGroup(group) {
  525. const registerType = getRegisterType(group.registerType || group.type || DEFAULT_REGISTER_TYPE)
  526. const sourceMemoryArea = String(group.sourceMemoryArea || '').trim()
  527. const isStorageGroupSource = !!sourceMemoryArea || group.addressUnit === 'byte' || group.addressUnit === 'bytes'
  528. const startAddress = isStorageGroupSource
  529. ? normalizeStorageAddress(
  530. group.sourceAddress !== undefined && group.sourceAddress !== null && group.sourceAddress !== ''
  531. ? group.sourceAddress
  532. : group.startAddress,
  533. 0
  534. )
  535. : normalizeAddress(group.startAddress, 0)
  536. const maxQuantity = getMaxQuantity(registerType.key)
  537. const sourceRegisters = Array.isArray(group.registers) ? group.registers : []
  538. const codeInfoContext = group.codeInfoContext || {}
  539. const layout = isStructLayout(group.layout) ? GROUP_LAYOUT_STRUCT : GROUP_LAYOUT_REGISTER
  540. const byteAddressed = isByteAddressedGroup(group)
  541. const hasExplicitQuantity = group.quantity !== undefined && group.quantity !== null && group.quantity !== ''
  542. const quantity = hasExplicitQuantity
  543. ? clampInteger(group.quantity, 1, maxQuantity, 1)
  544. : clampInteger(sourceRegisters.length || 1, 1, maxQuantity, 1)
  545. const baseGroup = {
  546. deleteVisible: !!group.deleteVisible,
  547. expanded: group.expanded === true,
  548. id: group.id || createId('gm-group'),
  549. isStructLayout: layout === GROUP_LAYOUT_STRUCT,
  550. layout,
  551. name: String(group.name || group.groupName || '寄存器组').trim() || '寄存器组',
  552. pollEnabled: group.pollEnabled === false ? false : true,
  553. quantity,
  554. registerType: registerType.key,
  555. startAddress,
  556. touchStartX: 0,
  557. codeInfoContext,
  558. ...pickFields(group, SOURCE_GROUP_FIELDS)
  559. }
  560. const registers = []
  561. let nextAddress = startAddress
  562. let nextByteOffset = 0
  563. for (let index = 0; index < quantity; index += 1) {
  564. const sourceRegister = sourceRegisters[index] || {}
  565. let normalizedSourceRegister = sourceRegister
  566. const dataType = normalizeRegisterDataType(sourceRegister, baseGroup.registerType)
  567. const textByteLength = isTextRegister(dataType)
  568. ? normalizeTextByteLength(sourceRegister.textByteLength, DEFAULT_TEXT_BYTE_LENGTH)
  569. : ''
  570. const isBitRegister = isBitRegisterType(baseGroup.registerType)
  571. let address = startAddress + index
  572. let byteOffset = 0
  573. if (!isBitRegister) {
  574. const explicitByteStart = Number(sourceRegister.byteStart)
  575. const hasExplicitByteStart = layout === GROUP_LAYOUT_STRUCT && Number.isFinite(explicitByteStart)
  576. const byteLength = getRegisterByteLength(dataType, { ...sourceRegister, layout, textByteLength })
  577. if (layout !== GROUP_LAYOUT_STRUCT && !isByteRegister(dataType) && nextByteOffset % 2 !== 0) {
  578. nextByteOffset += 1
  579. }
  580. const currentByteStart = hasExplicitByteStart
  581. ? Math.max(0, Math.floor(explicitByteStart))
  582. : nextByteOffset
  583. address = byteAddressed ? startAddress + currentByteStart : startAddress + Math.floor(currentByteStart / 2)
  584. byteOffset = byteAddressed ? 0 : currentByteStart % 2
  585. normalizedSourceRegister = {
  586. ...sourceRegister,
  587. byteStart: currentByteStart
  588. }
  589. nextByteOffset = Math.max(nextByteOffset, currentByteStart + byteLength)
  590. }
  591. const register = normalizeRegister(normalizedSourceRegister, baseGroup, index, address, byteOffset)
  592. registers.push(register)
  593. if (isBitRegister) nextAddress += register.registerCount
  594. }
  595. const byteLength = isBitRegisterType(baseGroup.registerType)
  596. ? Math.max(1, nextAddress - startAddress)
  597. : Math.max(1, nextByteOffset)
  598. const paddedByteLength = isBitRegisterType(baseGroup.registerType)
  599. ? byteLength
  600. : (byteAddressed ? byteLength : alignEvenByteLength(byteLength))
  601. const wordQuantity = isBitRegisterType(baseGroup.registerType)
  602. ? Math.max(1, nextAddress - startAddress)
  603. : (byteAddressed ? Math.max(1, byteLength) : Math.max(1, paddedByteLength / 2))
  604. const addressSpan = byteAddressed ? Math.max(1, byteLength) : wordQuantity
  605. const storageMemory = isStorageMemoryGroup(baseGroup)
  606. const storageAddressWidth = storageMemory ? resolveStorageAddressWidth(baseGroup, startAddress) : 0
  607. const storageAddressMax = getStorageAddressMax(storageAddressWidth, startAddress)
  608. const addressOverflow = storageMemory
  609. ? isStorageAddressRangeOverflow(startAddress, addressSpan, storageAddressWidth)
  610. : isAddressRangeOverflow(startAddress, addressSpan)
  611. const endAddress = startAddress + addressSpan - 1
  612. const addressMax = storageMemory ? storageAddressMax : MAX_MODBUS_ADDRESS
  613. const startAddressText = storageMemory
  614. ? formatStorageStartAddressText(startAddress, storageAddressWidth)
  615. : formatStartAddressText(startAddress, addressMax)
  616. const endAddressText = storageMemory
  617. ? (addressOverflow ? `0x${padStorageHex(addressMax, storageAddressWidth)}+` : `0x${padStorageHex(endAddress, storageAddressWidth)}`)
  618. : (addressOverflow ? `0x${padWordHex(addressMax)}+` : `0x${padWordHex(endAddress)}`)
  619. const displayName = storageMemory
  620. ? stripStorageAreaPrefix(baseGroup.name)
  621. : baseGroup.name
  622. const listMetaText = storageMemory
  623. ? `${getStorageAreaText(baseGroup)} ${startAddressText}-${endAddressText} ${byteLength}B`
  624. : `${formatAddressRange(startAddress, addressSpan)} · ${quantity}/${wordQuantity}${createGroupSourceMetaText(baseGroup) ? ` · ${createGroupSourceMetaText(baseGroup)}` : ''}${addressOverflow ? ' · 地址超出 0xFFFF' : ''}`
  625. const detailMetaText = storageMemory
  626. ? `${getStorageAreaText(baseGroup, true)} ${startAddressText} ${quantity}/${byteLength}B`
  627. : `${formatAddressRange(startAddress, addressSpan)} · ${quantity}/${wordQuantity}${createGroupSourceMetaText(baseGroup) ? ` · ${createGroupSourceMetaText(baseGroup)}` : ''}${addressOverflow ? ' · 地址超出 0xFFFF' : ''}`
  628. const sourceMetaText = createGroupSourceMetaText(baseGroup)
  629. return {
  630. ...baseGroup,
  631. addressRangeText: storageMemory
  632. ? formatStorageAddressRange(startAddress, addressSpan, storageAddressWidth)
  633. : formatAddressRange(startAddress, addressSpan),
  634. addressOverflow,
  635. addressWarningText: addressOverflow
  636. ? `地址超出 0x${storageMemory ? padStorageHex(addressMax, storageAddressWidth) : padWordHex(addressMax)}`
  637. : '',
  638. detailMetaText,
  639. detailTitleText: displayName,
  640. displayName,
  641. endAddressText,
  642. functionCode: registerType.functionCode,
  643. isReadOnly: !registerType.writable,
  644. byteLength,
  645. listMetaText,
  646. maxQuantity,
  647. registerTypeIndex: getRegisterTypeIndex(registerType.key),
  648. registerTypeText: registerType.label,
  649. registers,
  650. paddedByteLength,
  651. sourceMetaText,
  652. startAddressText,
  653. wordQuantity,
  654. writable: registerType.writable
  655. }
  656. }
  657. function normalizeGroupConfig(config = {}) {
  658. const registerType = config.registerTypeIndex !== undefined && config.registerTypeIndex !== null
  659. ? (REGISTER_TYPE_OPTIONS[Number(config.registerTypeIndex)] || REGISTER_TYPE_OPTIONS[0])
  660. : getRegisterType(config.registerType || config.type || DEFAULT_REGISTER_TYPE)
  661. const maxQuantity = getMaxQuantity(registerType.key)
  662. const isStorageConfig = !!String(config.sourceMemoryArea || '').trim()
  663. || config.addressUnit === 'byte'
  664. || config.addressUnit === 'bytes'
  665. const storageAddressWidth = isStorageConfig ? resolveStorageAddressWidth(config, 0) : 0
  666. const storageAddressByteLength = storageAddressWidth === 16 ? 2 : 4
  667. return {
  668. ...(isStorageConfig ? {
  669. addressUnit: config.addressUnit || 'byte',
  670. sourceAddressByteLength: config.sourceAddressByteLength || storageAddressByteLength,
  671. sourceAddressWidth: storageAddressWidth,
  672. sourceMemoryArea: config.sourceMemoryArea || 'XDATA'
  673. } : {}),
  674. layout: isStructLayout(config.layout) ? GROUP_LAYOUT_STRUCT : GROUP_LAYOUT_REGISTER,
  675. name: String(config.name || config.groupName || '寄存器组').trim() || '寄存器组',
  676. pollEnabled: config.pollEnabled === false ? false : true,
  677. quantity: parseConfigQuantity(config.quantity, maxQuantity),
  678. registerType: registerType.key,
  679. startAddress: parseConfigAddress(config.startAddress, {
  680. label: isStorageConfig ? '内存起始地址' : '寄存器起始地址',
  681. maxAddress: isStorageConfig ? getStorageAddressMax(storageAddressWidth) : MAX_MODBUS_ADDRESS
  682. })
  683. }
  684. }
  685. function getRegisterJsonValue(register) {
  686. if (register.inputValue !== undefined && register.inputValue !== null) {
  687. return normalizeTextValue(register.inputValue)
  688. }
  689. if (register.defaultValue !== undefined && register.defaultValue !== null) {
  690. return normalizeTextValue(register.defaultValue)
  691. }
  692. if (register.displayValue !== undefined && register.displayValue !== null && register.displayValue !== '--') {
  693. return normalizeTextValue(register.displayValue)
  694. }
  695. return ''
  696. }
  697. function normalizeImportedRegisterDataType(register) {
  698. const dataType = register.dataType || register.type || DEFAULT_DATA_TYPE
  699. return getDataType(dataType).key
  700. }
  701. function cloneImportedGroup(group) {
  702. return {
  703. layout: isStructLayout(group.layout) ? GROUP_LAYOUT_STRUCT : GROUP_LAYOUT_REGISTER,
  704. name: group.name,
  705. pollEnabled: group.pollEnabled === false ? false : true,
  706. quantity: group.quantity,
  707. registerType: group.registerType || group.type || DEFAULT_REGISTER_TYPE,
  708. registers: (Array.isArray(group.registers) ? group.registers : []).map((register) => ({
  709. conversionFormula: register.conversionFormula,
  710. dataType: normalizeImportedRegisterDataType(register),
  711. defaultValue: register.defaultValue,
  712. inputValue: register.inputValue,
  713. maxValue: register.maxValue,
  714. minValue: register.minValue,
  715. name: register.name,
  716. isStructField: !!register.isStructField,
  717. textByteLength: register.textByteLength,
  718. remark: register.remark,
  719. unit: register.unit,
  720. value: register.value,
  721. ...pickFields(register, STRUCT_REGISTER_FIELDS),
  722. ...pickFields(register, SOURCE_REGISTER_FIELDS)
  723. })),
  724. startAddress: group.startAddress,
  725. ...pickFields(group, SOURCE_GROUP_FIELDS)
  726. }
  727. }
  728. module.exports = {
  729. DATA_TYPE_OPTIONS,
  730. DEFAULT_DATA_TYPE,
  731. DEFAULT_REGISTER_TYPE,
  732. GROUP_LAYOUT_REGISTER,
  733. GROUP_LAYOUT_STRUCT,
  734. MAX_MODBUS_ADDRESS,
  735. MAX_STORAGE_ADDRESS,
  736. REGISTER_TYPE_OPTIONS,
  737. cloneImportedGroup,
  738. decodeRegisterFromByteCache,
  739. decodeRegisterFromWordCache,
  740. decodeRegisterValue,
  741. formatCoilDisplayValue,
  742. formatRegisterValue,
  743. getDataType,
  744. getRegisterEncodedBytes,
  745. getRegisterEncodedWords,
  746. getGroupEncodedBytes,
  747. getGroupEncodedWords,
  748. getRegisterWordCount,
  749. getRegisterJsonValue,
  750. getRegisterBytesFromByteCache,
  751. getRegisterWordsFromByteCache,
  752. getRegisterWordsFromWordCache,
  753. getRegisterWriteValueText,
  754. isAddressRangeOverflow,
  755. isBitRegisterType,
  756. isByteRegister,
  757. isParameterGroupPollEnabled,
  758. isStorageMemoryGroup,
  759. isStorageStructGroup,
  760. isTextRegister,
  761. normalizeGroup,
  762. normalizeGroupConfig,
  763. normalizeStorageCodeInfoCard,
  764. normalizeRegister,
  765. parseCoilValue,
  766. registerTypeIsBit,
  767. splitWordSpans,
  768. validateRegisterValue
  769. }