1
0

generic-modbus-model.js 28 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923
  1. const {
  2. formatFixedValue
  3. } = require('./conversions')
  4. const {
  5. clampInteger,
  6. createId,
  7. normalizeTextValue,
  8. padHex
  9. } = require('./base-utils')
  10. const {
  11. bytesToWords,
  12. getByteFromWord,
  13. trimTrailingNullBytes,
  14. wordsToBytes
  15. } = require('./binary-utils')
  16. const MAX_MODBUS_ADDRESS = 0xFFFF
  17. const MAX_GENERIC_MODBUS_ITEMS = 256
  18. const DEFAULT_TEXT_BYTE_LENGTH = 32
  19. const MAX_TEXT_BYTE_LENGTH = 32
  20. const REGISTER_TYPE_OPTIONS = [
  21. {
  22. functionCode: 0x03,
  23. key: 'holding',
  24. label: '保持寄存器',
  25. writable: true
  26. },
  27. {
  28. functionCode: 0x01,
  29. key: 'coil',
  30. label: '线圈',
  31. writable: true
  32. },
  33. {
  34. functionCode: 0x02,
  35. key: 'discrete',
  36. label: '离散输入状态',
  37. writable: false
  38. },
  39. {
  40. functionCode: 0x04,
  41. key: 'input',
  42. label: '输入寄存器',
  43. writable: false
  44. }
  45. ]
  46. const DATA_TYPE_OPTIONS = [
  47. {
  48. byteLength: 1,
  49. key: 'int8_t',
  50. label: 'int8_t',
  51. kind: 'number',
  52. wordCount: 1
  53. },
  54. {
  55. byteLength: 1,
  56. key: 'uint8_t',
  57. label: 'uint8_t',
  58. kind: 'number',
  59. wordCount: 1
  60. },
  61. {
  62. byteLength: 2,
  63. key: 'int16_t',
  64. label: 'int16_t',
  65. kind: 'number',
  66. wordCount: 1
  67. },
  68. {
  69. byteLength: 2,
  70. key: 'uint16_t',
  71. label: 'uint16_t',
  72. kind: 'number',
  73. wordCount: 1
  74. },
  75. {
  76. byteLength: 4,
  77. key: 'int32_t',
  78. label: 'int32_t',
  79. kind: 'number',
  80. wordCount: 2
  81. },
  82. {
  83. byteLength: 4,
  84. key: 'uint32_t',
  85. label: 'uint32_t',
  86. kind: 'number',
  87. wordCount: 2
  88. },
  89. {
  90. byteLength: 4,
  91. key: 'float',
  92. label: 'float',
  93. kind: 'number',
  94. wordCount: 2
  95. },
  96. {
  97. byteLength: 32,
  98. key: 'utf8',
  99. label: 'UTF-8',
  100. kind: 'text',
  101. maxByteLength: MAX_TEXT_BYTE_LENGTH,
  102. wordCount: 16
  103. },
  104. {
  105. byteLength: 32,
  106. key: 'ascii',
  107. label: 'ASCII',
  108. kind: 'text',
  109. maxByteLength: MAX_TEXT_BYTE_LENGTH,
  110. wordCount: 16
  111. },
  112. {
  113. byteLength: 2,
  114. key: 'hex',
  115. label: 'HEX',
  116. kind: 'hex',
  117. wordCount: 1
  118. }
  119. ]
  120. const DEFAULT_REGISTER_TYPE = REGISTER_TYPE_OPTIONS[0].key
  121. const DEFAULT_DATA_TYPE = 'uint16_t'
  122. function normalizeAddress(value, fallback = 0) {
  123. if (typeof value === 'number') {
  124. return Number.isFinite(value) ? clampInteger(value, 0, MAX_MODBUS_ADDRESS, fallback) : fallback
  125. }
  126. const text = String(value === undefined || value === null ? '' : value).trim()
  127. if (!text) return fallback
  128. const hexText = text.toUpperCase().startsWith('0X') ? text.slice(2) : text
  129. if (/^[0-9A-F]+$/i.test(hexText)) {
  130. const parsedHex = parseInt(hexText, 16)
  131. return Number.isFinite(parsedHex) ? clampInteger(parsedHex, 0, MAX_MODBUS_ADDRESS, fallback) : fallback
  132. }
  133. const numberValue = Number(text)
  134. return Number.isFinite(numberValue) ? clampInteger(numberValue, 0, MAX_MODBUS_ADDRESS, fallback) : fallback
  135. }
  136. function parseConfigAddress(value) {
  137. if (typeof value === 'number') {
  138. return clampInteger(value, 0, MAX_MODBUS_ADDRESS, 0)
  139. }
  140. const text = String(value === undefined || value === null ? '' : value).trim()
  141. const hexText = text.toUpperCase().startsWith('0X') ? text.slice(2) : text
  142. if (!/^[0-9A-F]{1,4}$/i.test(hexText)) {
  143. throw new Error('寄存器起始地址无效')
  144. }
  145. return parseInt(hexText, 16)
  146. }
  147. function parseConfigQuantity(value, maxQuantity) {
  148. const text = String(value === undefined || value === null ? '' : value).trim()
  149. const quantity = Number(text)
  150. if (!Number.isInteger(quantity) || quantity < 1 || quantity > maxQuantity) {
  151. throw new Error(`寄存器数量需为 1 - ${maxQuantity}`)
  152. }
  153. return quantity
  154. }
  155. function getRegisterType(typeKey) {
  156. return REGISTER_TYPE_OPTIONS.find((item) => item.key === typeKey) || REGISTER_TYPE_OPTIONS[0]
  157. }
  158. function getRegisterTypeIndex(typeKey) {
  159. return Math.max(0, REGISTER_TYPE_OPTIONS.findIndex((item) => item.key === getRegisterType(typeKey).key))
  160. }
  161. function getDataType(dataType) {
  162. return DATA_TYPE_OPTIONS.find((item) => item.key === dataType)
  163. || DATA_TYPE_OPTIONS.find((item) => item.key === DEFAULT_DATA_TYPE)
  164. || DATA_TYPE_OPTIONS[0]
  165. }
  166. function getDataTypeIndex(dataType) {
  167. return Math.max(0, DATA_TYPE_OPTIONS.findIndex((item) => item.key === getDataType(dataType).key))
  168. }
  169. function normalizeTextByteLength(value, fallback = DEFAULT_TEXT_BYTE_LENGTH) {
  170. const numberValue = Number(value)
  171. const rounded = Number.isFinite(numberValue) ? Math.round(numberValue) : fallback
  172. return Math.min(Math.max(rounded, 1), MAX_TEXT_BYTE_LENGTH)
  173. }
  174. function alignEvenByteLength(byteLength) {
  175. const length = Math.max(1, Math.round(Number(byteLength) || 1))
  176. return length % 2 === 0 ? length : length + 1
  177. }
  178. function getRegisterTextByteLength(register = {}) {
  179. return normalizeTextByteLength(register.textByteLength, DEFAULT_TEXT_BYTE_LENGTH)
  180. }
  181. function getRegisterByteLength(dataType, register = {}) {
  182. const type = getDataType(dataType)
  183. if (type.kind === 'text') return alignEvenByteLength(getRegisterTextByteLength(register))
  184. return type.byteLength || ((type.wordCount || 1) * 2)
  185. }
  186. function getRegisterWordCount(dataType, register = {}) {
  187. return Math.max(1, Math.ceil(getRegisterByteLength(dataType, register) / 2))
  188. }
  189. function getRegisterWordCountAtOffset(dataType, byteOffset, register = {}) {
  190. const byteLength = getRegisterByteLength(dataType, register)
  191. return Math.max(1, Math.ceil((byteOffset + byteLength) / 2))
  192. }
  193. function getEncodeByteLimit(register) {
  194. return isTextRegister(register.dataType) ? getRegisterTextByteLength(register) : getRegisterByteLength(register.dataType, register)
  195. }
  196. function isTextRegister(dataType) {
  197. return getDataType(dataType).kind === 'text'
  198. }
  199. function isByteRegister(dataType) {
  200. const key = getDataType(dataType).key
  201. return key === 'int8_t' || key === 'uint8_t'
  202. }
  203. function isBitRegisterType(registerType) {
  204. return registerType === 'coil' || registerType === 'discrete'
  205. }
  206. function isHexRegister(dataType) {
  207. return getDataType(dataType).key === 'hex'
  208. }
  209. function isNumericRegister(dataType) {
  210. return getDataType(dataType).kind === 'number'
  211. }
  212. function supportsRange(dataType) {
  213. return isNumericRegister(dataType) || isHexRegister(dataType)
  214. }
  215. function supportsUnit(dataType) {
  216. return isNumericRegister(dataType)
  217. }
  218. function padWordHex(value) {
  219. return Number(value || 0).toString(16).toUpperCase().padStart(4, '0')
  220. }
  221. function formatRawWordText(words = []) {
  222. if (!Array.isArray(words) || !words.length) return '--'
  223. return words.map((word) => `0x${padWordHex(word)}`).join(' ')
  224. }
  225. function formatAddressRange(startAddress, wordCount) {
  226. const address = normalizeAddress(startAddress, 0)
  227. const count = Math.max(1, Number(wordCount) || 1)
  228. const endAddress = address + count - 1
  229. const safeEndAddress = Math.min(endAddress, MAX_MODBUS_ADDRESS)
  230. const overflowText = endAddress > MAX_MODBUS_ADDRESS ? '+' : ''
  231. if (count <= 1) return `0x${padWordHex(address)}`
  232. return `0x${padWordHex(address)}-0x${padWordHex(safeEndAddress)}${overflowText}`
  233. }
  234. function formatRegisterAddressText(address, byteOffset, byteLength, registerType) {
  235. if (isBitRegisterType(registerType)) return `0x${padHex(address)}`
  236. if (byteLength === 1) return `0x${padHex(address)}${byteOffset === 0 ? 'H' : 'L'}`
  237. return `0x${padHex(address)}`
  238. }
  239. function isAddressRangeOverflow(startAddress, wordCount) {
  240. const address = normalizeAddress(startAddress, 0)
  241. const count = Math.max(1, Number(wordCount) || 1)
  242. return address + count - 1 > MAX_MODBUS_ADDRESS
  243. }
  244. function encodeAsciiBytes(text, byteLimit = 32) {
  245. const bytes = []
  246. const stringValue = normalizeTextValue(text)
  247. for (let index = 0; index < stringValue.length; index += 1) {
  248. const code = stringValue.charCodeAt(index)
  249. if (code > 0x7F) {
  250. throw new Error('ASCII 文本只能包含 0x00 - 0x7F 字符')
  251. }
  252. bytes.push(code)
  253. if (bytes.length > byteLimit) break
  254. }
  255. if (bytes.length > byteLimit) {
  256. throw new Error(`长文本最长 ${byteLimit} 字节`)
  257. }
  258. return bytes
  259. }
  260. function encodeUtf8Bytes(text, byteLimit = 32) {
  261. const bytes = []
  262. const encoded = encodeURIComponent(normalizeTextValue(text))
  263. for (let index = 0; index < encoded.length; index += 1) {
  264. const char = encoded[index]
  265. if (char === '%') {
  266. const byte = parseInt(encoded.slice(index + 1, index + 3), 16)
  267. if (!Number.isFinite(byte)) break
  268. bytes.push(byte & 0xFF)
  269. index += 2
  270. } else {
  271. bytes.push(char.charCodeAt(0) & 0xFF)
  272. }
  273. if (bytes.length > byteLimit) break
  274. }
  275. if (bytes.length > byteLimit) {
  276. throw new Error(`长文本最长 ${byteLimit} 字节`)
  277. }
  278. return bytes
  279. }
  280. function decodeAsciiBytes(bytes = []) {
  281. return String.fromCharCode.apply(null, trimTrailingNullBytes(bytes).map((byte) => byte & 0xFF))
  282. }
  283. function decodeUtf8Bytes(bytes = []) {
  284. const trimmed = trimTrailingNullBytes(bytes)
  285. if (!trimmed.length) return ''
  286. let encoded = ''
  287. trimmed.forEach((byte) => {
  288. encoded += `%${(byte & 0xFF).toString(16).padStart(2, '0').toUpperCase()}`
  289. })
  290. try {
  291. return decodeURIComponent(encoded)
  292. } catch (error) {
  293. return decodeAsciiBytes(trimmed)
  294. }
  295. }
  296. function encodeTextBytes(text, dataType, byteLimit = MAX_TEXT_BYTE_LENGTH) {
  297. const normalizedType = getDataType(dataType).key
  298. if (normalizedType === 'ascii') return encodeAsciiBytes(text, byteLimit)
  299. return encodeUtf8Bytes(text, byteLimit)
  300. }
  301. function decodeTextBytes(bytes, dataType) {
  302. const normalizedType = getDataType(dataType).key
  303. return normalizedType === 'ascii'
  304. ? decodeAsciiBytes(bytes)
  305. : decodeUtf8Bytes(bytes)
  306. }
  307. function formatIntegerValue(value, dataType) {
  308. const type = getDataType(dataType).key
  309. const numberValue = Number(value)
  310. if (!Number.isFinite(numberValue)) return '--'
  311. if (type === 'int8_t') return String(((Math.round(numberValue) << 24) >> 24))
  312. if (type === 'uint8_t') return String(Math.round(numberValue) & 0xFF)
  313. if (type === 'int16_t') return String(((Math.round(numberValue) << 16) >> 16))
  314. if (type === 'uint16_t') return String(Math.round(numberValue) & 0xFFFF)
  315. if (type === 'int32_t') return String((Math.round(numberValue) | 0))
  316. if (type === 'uint32_t') return String(Math.round(numberValue) >>> 0)
  317. return String(Math.round(numberValue))
  318. }
  319. function formatHexValue(value) {
  320. const numberValue = Number(value)
  321. if (!Number.isFinite(numberValue)) return '--'
  322. return `0x${padWordHex(Math.round(numberValue) & 0xFFFF)}`
  323. }
  324. function formatFloatValue(value) {
  325. return formatFixedValue(value, 6).replace(/(\.\d*?)0+$/, '$1').replace(/\.$/, '')
  326. }
  327. function parseIntegerText(value) {
  328. const text = String(value === undefined || value === null ? '' : value).trim()
  329. if (!text) return null
  330. const isHex = /^[-+]?0x[0-9a-f]+$/i.test(text) || /^0x[0-9a-f]+$/i.test(text)
  331. const parsed = isHex ? parseInt(text, 16) : Number(text)
  332. return Number.isFinite(parsed) ? parsed : null
  333. }
  334. function parseHexText(value) {
  335. const text = String(value === undefined || value === null ? '' : value).trim()
  336. if (!text) return null
  337. const hexText = text.toUpperCase().startsWith('0X') ? text.slice(2) : text
  338. if (/^[0-9A-F]{1,4}$/i.test(hexText)) {
  339. const parsedHex = parseInt(hexText, 16)
  340. return Number.isFinite(parsedHex) ? parsedHex : null
  341. }
  342. return null
  343. }
  344. function getRegisterValueTypeLabel(dataType) {
  345. return getDataType(dataType).label
  346. }
  347. function getMaxQuantity() {
  348. return MAX_GENERIC_MODBUS_ITEMS
  349. }
  350. function parseCoilValue(value) {
  351. const text = String(value === undefined || value === null ? '' : value).trim()
  352. if (!text || text === '--') return null
  353. if (['1', 'true', 'TRUE', 'on', 'ON', '开'].includes(text)) return 1
  354. if (['0', 'false', 'FALSE', 'off', 'OFF', '关'].includes(text)) return 0
  355. const coilValue = Number(text)
  356. return Number.isFinite(coilValue) ? (coilValue ? 1 : 0) : null
  357. }
  358. function getNumericRange(dataType) {
  359. const type = getDataType(dataType).key
  360. if (type === 'int8_t') return { max: 127, min: -128 }
  361. if (type === 'uint8_t') return { max: 0xFF, min: 0 }
  362. if (type === 'int16_t') return { max: 32767, min: -32768 }
  363. if (type === 'uint16_t') return { max: 0xFFFF, min: 0 }
  364. if (type === 'int32_t') return { max: 2147483647, min: -2147483648 }
  365. if (type === 'uint32_t') return { max: 0xFFFFFFFF, min: 0 }
  366. if (type === 'hex') return { max: 0xFFFF, min: 0 }
  367. return { max: Number.POSITIVE_INFINITY, min: Number.NEGATIVE_INFINITY }
  368. }
  369. function parseNumberText(value, dataType) {
  370. const text = String(value === undefined || value === null ? '' : value).trim()
  371. if (!text || text === '--') return null
  372. if (getDataType(dataType).key === 'float') {
  373. const parsed = Number(text)
  374. return Number.isFinite(parsed) ? parsed : null
  375. }
  376. if (isHexRegister(dataType)) return parseHexText(text)
  377. return parseIntegerText(text)
  378. }
  379. function parseRangeBoundary(value, dataType, label) {
  380. const text = String(value === undefined || value === null ? '' : value).trim()
  381. if (!text) return null
  382. const parsed = parseNumberText(text, dataType)
  383. if (parsed === null) {
  384. throw new Error(`${label}无效`)
  385. }
  386. return parsed
  387. }
  388. function validateNumericValue(register, value) {
  389. const dataType = getDataType(register.dataType).key
  390. const range = getNumericRange(dataType)
  391. const numberValue = Number(value)
  392. if (!Number.isFinite(numberValue)) return false
  393. if (dataType !== 'float' && Math.round(numberValue) !== numberValue) {
  394. throw new Error(`${register.name || '寄存器'} 需要整数`)
  395. }
  396. if (numberValue < range.min || numberValue > range.max) {
  397. throw new Error(`${register.name || '寄存器'} 超出 ${dataType} 范围`)
  398. }
  399. const minValue = parseRangeBoundary(register.minValue, dataType, `${register.name || '寄存器'} 最小值`)
  400. const maxValue = parseRangeBoundary(register.maxValue, dataType, `${register.name || '寄存器'} 最大值`)
  401. if (minValue !== null && numberValue < minValue) {
  402. throw new Error(`${register.name || '寄存器'} 小于限制最小值`)
  403. }
  404. if (maxValue !== null && numberValue > maxValue) {
  405. throw new Error(`${register.name || '寄存器'} 大于限制最大值`)
  406. }
  407. return true
  408. }
  409. function floatToWords(value) {
  410. const buffer = new ArrayBuffer(4)
  411. const view = new DataView(buffer)
  412. view.setFloat32(0, Number(value), false)
  413. return [view.getUint16(0, false), view.getUint16(2, false)]
  414. }
  415. function wordsToFloat(words) {
  416. if (!Array.isArray(words) || words.length < 2) return null
  417. const buffer = new ArrayBuffer(4)
  418. const view = new DataView(buffer)
  419. view.setUint16(0, Number(words[0]) & 0xFFFF, false)
  420. view.setUint16(2, Number(words[1]) & 0xFFFF, false)
  421. return view.getFloat32(0, false)
  422. }
  423. function encodeRegisterWords(register) {
  424. const dataType = getDataType(register.dataType).key
  425. const valueText = normalizeTextValue(register.inputValue)
  426. if (isTextRegister(dataType)) {
  427. const byteLimit = getEncodeByteLimit(register)
  428. const byteLength = getRegisterByteLength(dataType, register)
  429. const bytes = encodeTextBytes(valueText, dataType, byteLimit)
  430. const paddedBytes = bytes.slice()
  431. while (paddedBytes.length < byteLength) {
  432. paddedBytes.push(0)
  433. }
  434. return bytesToWords(paddedBytes.slice(0, byteLength))
  435. }
  436. const numberValue = parseNumberText(valueText, dataType)
  437. if (numberValue === null) return null
  438. validateNumericValue(register, numberValue)
  439. if (dataType === 'float') return floatToWords(numberValue)
  440. const rounded = Math.round(numberValue)
  441. if (dataType === 'int8_t') return [((rounded < 0 ? 0x100 + rounded : rounded) & 0xFF)]
  442. if (dataType === 'uint8_t') return [rounded & 0xFF]
  443. if (dataType === 'int16_t' || dataType === 'uint16_t' || dataType === 'hex') return [rounded & 0xFFFF]
  444. const unsignedValue = rounded < 0 ? 0x100000000 + rounded : rounded
  445. return [
  446. Math.floor(unsignedValue / 0x10000) & 0xFFFF,
  447. unsignedValue & 0xFFFF
  448. ]
  449. }
  450. function decodeRegisterValue(register, words) {
  451. const dataType = getDataType(register.dataType).key
  452. if (!Array.isArray(words) || words.length < getRegisterWordCount(dataType, register)) return null
  453. if (isTextRegister(dataType)) {
  454. return decodeTextBytes(wordsToBytes(words, getEncodeByteLimit(register)), dataType)
  455. }
  456. if (dataType === 'float') {
  457. return wordsToFloat(words)
  458. }
  459. if (dataType === 'int8_t') {
  460. const byteValue = getByteFromWord(words[0], register.byteOffset)
  461. return byteValue & 0x80 ? byteValue - 0x100 : byteValue
  462. }
  463. if (dataType === 'uint8_t') {
  464. return getByteFromWord(words[0], register.byteOffset)
  465. }
  466. if (dataType === 'int16_t') {
  467. const wordValue = Number(words[0]) & 0xFFFF
  468. return wordValue & 0x8000 ? wordValue - 0x10000 : wordValue
  469. }
  470. if (dataType === 'uint16_t') {
  471. return Number(words[0]) & 0xFFFF
  472. }
  473. if (dataType === 'hex') {
  474. return Number(words[0]) & 0xFFFF
  475. }
  476. const highWord = Number(words[0]) & 0xFFFF
  477. const lowWord = Number(words[1]) & 0xFFFF
  478. const unsignedValue = highWord * 0x10000 + lowWord
  479. if (dataType === 'int32_t') {
  480. return unsignedValue >= 0x80000000 ? unsignedValue - 0x100000000 : unsignedValue
  481. }
  482. return unsignedValue
  483. }
  484. function formatRegisterValue(register, rawValue) {
  485. if (rawValue === null || rawValue === undefined) return '--'
  486. const dataType = getDataType(register.dataType).key
  487. if (isTextRegister(dataType)) return normalizeTextValue(rawValue)
  488. if (dataType === 'hex') return formatHexValue(rawValue)
  489. if (dataType === 'float') return formatFloatValue(rawValue)
  490. return formatIntegerValue(rawValue, dataType)
  491. }
  492. function formatCoilDisplayValue(value) {
  493. return Number(value) ? '1' : '0'
  494. }
  495. function getRegisterSavedValue(register) {
  496. if (register.inputValue !== undefined && register.inputValue !== null) return normalizeTextValue(register.inputValue)
  497. if (register.value !== undefined && register.value !== null) return normalizeTextValue(register.value)
  498. return null
  499. }
  500. function normalizeRegisterDataType(register, registerType) {
  501. if (isBitRegisterType(registerType)) return DEFAULT_DATA_TYPE
  502. return getDataType(register.dataType || register.type || DEFAULT_DATA_TYPE).key
  503. }
  504. function normalizeRegister(register, group, index, address, byteOffset = 0) {
  505. const registerType = getRegisterType(group.registerType).key
  506. const dataType = normalizeRegisterDataType(register, registerType)
  507. const textByteLength = isTextRegister(dataType)
  508. ? normalizeTextByteLength(register.textByteLength, DEFAULT_TEXT_BYTE_LENGTH)
  509. : ''
  510. const defaultValue = normalizeTextValue(register.defaultValue)
  511. const savedValue = getRegisterSavedValue(register)
  512. const inputValue = savedValue === null ? defaultValue : savedValue
  513. const rawValue = register.rawValue === undefined ? null : register.rawValue
  514. const byteLength = isBitRegisterType(registerType) ? 1 : getRegisterByteLength(dataType, { textByteLength })
  515. const registerCount = isBitRegisterType(registerType) ? 1 : getRegisterWordCountAtOffset(dataType, byteOffset, { textByteLength })
  516. const canShowUnit = !isBitRegisterType(registerType) && supportsUnit(dataType)
  517. const rawWords = Array.isArray(register.rawWords)
  518. ? register.rawWords.slice(0, registerCount).map((word) => Number(word) & 0xFFFF)
  519. : []
  520. const rawValueText = rawValue === null
  521. ? '--'
  522. : (isBitRegisterType(registerType) ? formatCoilDisplayValue(rawValue) : formatRawWordText(rawWords))
  523. const displayValue = rawValue === null
  524. ? (inputValue.trim() ? inputValue : '--')
  525. : formatRegisterValue({ ...register, dataType, byteOffset }, rawValue)
  526. return {
  527. address,
  528. addressRangeText: isBitRegisterType(registerType)
  529. ? `0x${padHex(address)}`
  530. : formatAddressRange(address, registerCount),
  531. addressText: formatRegisterAddressText(address, byteOffset, byteLength, registerType),
  532. byteLength,
  533. byteLengthText: isBitRegisterType(registerType)
  534. ? '1bit'
  535. : (textByteLength && textByteLength !== byteLength ? `${textByteLength}B/占${byteLength}B` : `${byteLength}B`),
  536. dataType,
  537. dataTypeIndex: getDataTypeIndex(dataType),
  538. dataTypeText: getRegisterValueTypeLabel(dataType),
  539. defaultValue,
  540. displayValue,
  541. id: register.id || createId('gm-reg'),
  542. inputType: isTextRegister(dataType) ? 'text' : 'text',
  543. inputValue,
  544. isDirty: !!register.isDirty,
  545. maxValue: normalizeTextValue(register.maxValue),
  546. minValue: normalizeTextValue(register.minValue),
  547. name: register.name || `寄存器 ${index + 1}`,
  548. rawValue,
  549. rawValueText,
  550. rawWords,
  551. registerCount,
  552. byteOffset,
  553. registerType,
  554. showDataType: !isBitRegisterType(registerType),
  555. showRange: !isBitRegisterType(registerType) && supportsRange(dataType),
  556. showTextLength: !isBitRegisterType(registerType) && isTextRegister(dataType),
  557. showUnit: canShowUnit,
  558. textByteLength,
  559. unit: canShowUnit ? normalizeTextValue(register.unit).trim() : '',
  560. remark: register.remark || ''
  561. }
  562. }
  563. function normalizeGroup(group) {
  564. const registerType = getRegisterType(group.registerType || group.type || DEFAULT_REGISTER_TYPE)
  565. const startAddress = normalizeAddress(group.startAddress, 0)
  566. const maxQuantity = getMaxQuantity(registerType.key)
  567. const sourceRegisters = Array.isArray(group.registers) ? group.registers : []
  568. const hasExplicitQuantity = group.quantity !== undefined && group.quantity !== null && group.quantity !== ''
  569. const quantity = hasExplicitQuantity
  570. ? clampInteger(group.quantity, 1, maxQuantity, 1)
  571. : clampInteger(sourceRegisters.length || 1, 1, maxQuantity, 1)
  572. const baseGroup = {
  573. deleteVisible: !!group.deleteVisible,
  574. expanded: group.expanded === true,
  575. id: group.id || createId('gm-group'),
  576. name: String(group.name || group.groupName || '寄存器组').trim() || '寄存器组',
  577. quantity,
  578. registerType: registerType.key,
  579. startAddress,
  580. touchStartX: 0
  581. }
  582. const registers = []
  583. let nextAddress = startAddress
  584. let nextByteOffset = 0
  585. for (let index = 0; index < quantity; index += 1) {
  586. const sourceRegister = sourceRegisters[index] || {}
  587. const dataType = normalizeRegisterDataType(sourceRegister, baseGroup.registerType)
  588. const textByteLength = isTextRegister(dataType)
  589. ? normalizeTextByteLength(sourceRegister.textByteLength, DEFAULT_TEXT_BYTE_LENGTH)
  590. : ''
  591. const isBitRegister = isBitRegisterType(baseGroup.registerType)
  592. let address = startAddress + index
  593. let byteOffset = 0
  594. if (!isBitRegister) {
  595. const byteLength = getRegisterByteLength(dataType, { textByteLength })
  596. if (!isByteRegister(dataType) && nextByteOffset % 2 !== 0) {
  597. nextByteOffset += 1
  598. }
  599. address = startAddress + Math.floor(nextByteOffset / 2)
  600. byteOffset = nextByteOffset % 2
  601. nextByteOffset += byteLength
  602. }
  603. const register = normalizeRegister(sourceRegister, baseGroup, index, address, byteOffset)
  604. registers.push(register)
  605. if (isBitRegister) nextAddress += register.registerCount
  606. }
  607. const wordQuantity = isBitRegisterType(baseGroup.registerType)
  608. ? Math.max(1, nextAddress - startAddress)
  609. : Math.max(1, Math.ceil(nextByteOffset / 2))
  610. const addressOverflow = isAddressRangeOverflow(startAddress, wordQuantity)
  611. const endAddress = startAddress + wordQuantity - 1
  612. return {
  613. ...baseGroup,
  614. addressRangeText: formatAddressRange(startAddress, wordQuantity),
  615. addressOverflow,
  616. addressWarningText: addressOverflow ? '地址超出 0xFFFF' : '',
  617. endAddressText: addressOverflow ? `0x${padHex(MAX_MODBUS_ADDRESS)}+` : `0x${padHex(endAddress)}`,
  618. functionCode: registerType.functionCode,
  619. isReadOnly: !registerType.writable,
  620. maxQuantity,
  621. registerTypeIndex: getRegisterTypeIndex(registerType.key),
  622. registerTypeText: registerType.label,
  623. registers,
  624. startAddressText: `0x${padHex(startAddress)}`,
  625. wordQuantity,
  626. writable: registerType.writable
  627. }
  628. }
  629. function normalizeGroupConfig(config = {}) {
  630. const registerType = config.registerTypeIndex !== undefined && config.registerTypeIndex !== null
  631. ? (REGISTER_TYPE_OPTIONS[Number(config.registerTypeIndex)] || REGISTER_TYPE_OPTIONS[0])
  632. : getRegisterType(config.registerType || config.type || DEFAULT_REGISTER_TYPE)
  633. const maxQuantity = getMaxQuantity(registerType.key)
  634. return {
  635. name: String(config.name || config.groupName || '寄存器组').trim() || '寄存器组',
  636. quantity: parseConfigQuantity(config.quantity, maxQuantity),
  637. registerType: registerType.key,
  638. startAddress: parseConfigAddress(config.startAddress)
  639. }
  640. }
  641. function getRegisterJsonValue(register) {
  642. if (register.inputValue !== undefined && register.inputValue !== null) {
  643. return normalizeTextValue(register.inputValue)
  644. }
  645. if (register.defaultValue !== undefined && register.defaultValue !== null) {
  646. return normalizeTextValue(register.defaultValue)
  647. }
  648. if (register.displayValue !== undefined && register.displayValue !== null && register.displayValue !== '--') {
  649. return normalizeTextValue(register.displayValue)
  650. }
  651. return ''
  652. }
  653. function normalizeImportedRegisterDataType(register) {
  654. const dataType = register.dataType || register.type || DEFAULT_DATA_TYPE
  655. return getDataType(dataType).key
  656. }
  657. function cloneImportedGroup(group) {
  658. return {
  659. name: group.name,
  660. quantity: group.quantity,
  661. registerType: group.registerType || group.type || DEFAULT_REGISTER_TYPE,
  662. registers: (Array.isArray(group.registers) ? group.registers : []).map((register) => ({
  663. dataType: normalizeImportedRegisterDataType(register),
  664. defaultValue: register.defaultValue,
  665. inputValue: register.inputValue,
  666. maxValue: register.maxValue,
  667. minValue: register.minValue,
  668. name: register.name,
  669. textByteLength: register.textByteLength,
  670. remark: register.remark,
  671. unit: register.unit,
  672. value: register.value
  673. })),
  674. startAddress: group.startAddress
  675. }
  676. }
  677. function splitWordSpans(startAddress, quantity, maxQuantity) {
  678. const spans = []
  679. let address = normalizeAddress(startAddress, 0)
  680. let remaining = Math.max(0, Math.floor(Number(quantity) || 0))
  681. while (remaining > 0) {
  682. const spanQuantity = Math.min(remaining, maxQuantity)
  683. spans.push({
  684. address,
  685. quantity: spanQuantity
  686. })
  687. address += spanQuantity
  688. remaining -= spanQuantity
  689. }
  690. return spans
  691. }
  692. function getRegisterWriteValueText(register) {
  693. if (register.inputValue !== undefined && register.inputValue !== null) return normalizeTextValue(register.inputValue)
  694. if (register.defaultValue !== undefined && register.defaultValue !== null) return normalizeTextValue(register.defaultValue)
  695. return ''
  696. }
  697. function getRegisterWordsFromWordCache(register, wordCache) {
  698. const words = []
  699. for (let offset = 0; offset < register.registerCount; offset += 1) {
  700. const word = wordCache[register.address + offset]
  701. if (word === undefined) return null
  702. words.push(word)
  703. }
  704. return words
  705. }
  706. function decodeRegisterFromWordCache(register, wordCache) {
  707. const words = getRegisterWordsFromWordCache(register, wordCache)
  708. if (!words) return null
  709. return decodeRegisterValue(register, words)
  710. }
  711. function getRegisterEncodedWords(register) {
  712. return encodeRegisterWords({
  713. ...register,
  714. inputValue: getRegisterWriteValueText(register)
  715. })
  716. }
  717. function validateRegisterValue(register, value) {
  718. const valueText = normalizeTextValue(
  719. value === undefined || value === null ? getRegisterWriteValueText(register) : value
  720. ).trim()
  721. if (!valueText || valueText === '--') return true
  722. if (registerTypeIsBit(register)) {
  723. if (parseCoilValue(valueText) === null) {
  724. throw new Error(`${register.name || '线圈'} 只能填写 0 或 1`)
  725. }
  726. return true
  727. }
  728. const dataType = getDataType(register.dataType).key
  729. if (isTextRegister(dataType)) {
  730. encodeTextBytes(valueText, dataType, getEncodeByteLimit(register))
  731. return true
  732. }
  733. const numberValue = parseNumberText(valueText, dataType)
  734. if (numberValue === null) {
  735. throw new Error(`${register.name || '寄存器'} 输入值无效`)
  736. }
  737. return validateNumericValue(register, numberValue)
  738. }
  739. function registerTypeIsBit(register) {
  740. return !!register && isBitRegisterType(register.registerType)
  741. }
  742. module.exports = {
  743. DATA_TYPE_OPTIONS,
  744. DEFAULT_DATA_TYPE,
  745. DEFAULT_REGISTER_TYPE,
  746. MAX_MODBUS_ADDRESS,
  747. REGISTER_TYPE_OPTIONS,
  748. cloneImportedGroup,
  749. decodeRegisterFromWordCache,
  750. decodeRegisterValue,
  751. formatCoilDisplayValue,
  752. formatRegisterValue,
  753. getDataType,
  754. getRegisterEncodedWords,
  755. getRegisterJsonValue,
  756. getRegisterWordsFromWordCache,
  757. getRegisterWriteValueText,
  758. isAddressRangeOverflow,
  759. isBitRegisterType,
  760. isByteRegister,
  761. normalizeGroup,
  762. normalizeGroupConfig,
  763. parseCoilValue,
  764. registerTypeIsBit,
  765. splitWordSpans,
  766. validateRegisterValue
  767. }