1
0

model.js 29 KB

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