filter-calculator.js 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303
  1. const {
  2. formatMagnitudeNumber,
  3. formatScaledValue,
  4. getOption,
  5. normalizeIndex,
  6. parsePositiveNumber,
  7. selectBestUnit
  8. } = require('./calculator-helpers')
  9. const TWO_PI = 2 * Math.PI
  10. const NETWORK_OPTIONS = [
  11. { key: 'rc', label: 'RC' },
  12. { key: 'rl', label: 'RL' }
  13. ]
  14. const RESPONSE_OPTIONS = [
  15. { key: 'lowpass', label: '低通' },
  16. { key: 'highpass', label: '高通' }
  17. ]
  18. const RESISTANCE_UNIT_OPTIONS = [
  19. { label: 'mΩ', factor: 1e-3 },
  20. { label: 'Ω', factor: 1 },
  21. { label: 'kΩ', factor: 1e3 },
  22. { label: 'MΩ', factor: 1e6 },
  23. { label: 'GΩ', factor: 1e9 }
  24. ]
  25. const CAPACITANCE_UNIT_OPTIONS = [
  26. { label: 'pF', factor: 1e-12 },
  27. { label: 'nF', factor: 1e-9 },
  28. { label: 'μF', factor: 1e-6 },
  29. { label: 'mF', factor: 1e-3 },
  30. { label: 'F', factor: 1 }
  31. ]
  32. const INDUCTANCE_UNIT_OPTIONS = [
  33. { label: 'pH', factor: 1e-12 },
  34. { label: 'nH', factor: 1e-9 },
  35. { label: 'μH', factor: 1e-6 },
  36. { label: 'mH', factor: 1e-3 },
  37. { label: 'H', factor: 1 },
  38. { label: 'kH', factor: 1e3 }
  39. ]
  40. const FREQUENCY_UNIT_OPTIONS = [
  41. { label: 'Hz', factor: 1 },
  42. { label: 'kHz', factor: 1e3 },
  43. { label: 'MHz', factor: 1e6 },
  44. { label: 'GHz', factor: 1e9 },
  45. { label: 'THz', factor: 1e12 }
  46. ]
  47. function getReactiveUnitOptions(networkKey) {
  48. return networkKey === 'rl' ? INDUCTANCE_UNIT_OPTIONS : CAPACITANCE_UNIT_OPTIONS
  49. }
  50. function getDiagramComponents(networkKey, responseKey) {
  51. if (networkKey === 'rc' && responseKey === 'highpass') {
  52. return { series: 'c', seriesLabel: 'C', shunt: 'r', shuntLabel: 'R' }
  53. }
  54. if (networkKey === 'rl' && responseKey === 'lowpass') {
  55. return { series: 'l', seriesLabel: 'L', shunt: 'r', shuntLabel: 'R' }
  56. }
  57. if (networkKey === 'rl' && responseKey === 'highpass') {
  58. return { series: 'r', seriesLabel: 'R', shunt: 'l', shuntLabel: 'L' }
  59. }
  60. return { series: 'r', seriesLabel: 'R', shunt: 'c', shuntLabel: 'C' }
  61. }
  62. function calculateMissingValue(networkKey, values) {
  63. const hasResistance = Number.isFinite(values.resistance)
  64. const hasReactive = Number.isFinite(values.reactive)
  65. const hasFrequency = Number.isFinite(values.frequency)
  66. const validCount = [hasResistance, hasReactive, hasFrequency].filter(Boolean).length
  67. if (validCount < 2) {
  68. return {
  69. computedKey: '',
  70. errorText: ''
  71. }
  72. }
  73. if (validCount > 2) {
  74. return {
  75. computedKey: '',
  76. errorText: '保留两项,清空一项用于计算'
  77. }
  78. }
  79. if (networkKey === 'rl') {
  80. if (!hasResistance) {
  81. return {
  82. computedKey: 'resistance',
  83. value: TWO_PI * values.frequency * values.reactive
  84. }
  85. }
  86. if (!hasReactive) {
  87. return {
  88. computedKey: 'reactive',
  89. value: values.resistance / (TWO_PI * values.frequency)
  90. }
  91. }
  92. return {
  93. computedKey: 'frequency',
  94. value: values.resistance / (TWO_PI * values.reactive)
  95. }
  96. }
  97. if (!hasResistance) {
  98. return {
  99. computedKey: 'resistance',
  100. value: 1 / (TWO_PI * values.frequency * values.reactive)
  101. }
  102. }
  103. if (!hasReactive) {
  104. return {
  105. computedKey: 'reactive',
  106. value: 1 / (TWO_PI * values.frequency * values.resistance)
  107. }
  108. }
  109. return {
  110. computedKey: 'frequency',
  111. value: 1 / (TWO_PI * values.resistance * values.reactive)
  112. }
  113. }
  114. function buildState(source = {}) {
  115. const networkIndex = normalizeIndex(source.filterNetworkIndex, NETWORK_OPTIONS, 0)
  116. const responseIndex = normalizeIndex(source.filterResponseIndex, RESPONSE_OPTIONS, 0)
  117. const network = getOption(NETWORK_OPTIONS, networkIndex)
  118. const response = getOption(RESPONSE_OPTIONS, responseIndex)
  119. const reactiveUnitOptions = getReactiveUnitOptions(network.key)
  120. const resistanceUnitIndex = normalizeIndex(source.filterResistanceUnitIndex, RESISTANCE_UNIT_OPTIONS, 1)
  121. const capacitanceUnitIndex = normalizeIndex(source.filterCapacitanceUnitIndex, CAPACITANCE_UNIT_OPTIONS, 1)
  122. const inductanceUnitIndex = normalizeIndex(source.filterInductanceUnitIndex, INDUCTANCE_UNIT_OPTIONS, 3)
  123. const frequencyUnitIndex = normalizeIndex(source.filterFrequencyUnitIndex, FREQUENCY_UNIT_OPTIONS, 0)
  124. const reactiveUnitIndex = network.key === 'rl' ? inductanceUnitIndex : capacitanceUnitIndex
  125. const resistanceUnit = getOption(RESISTANCE_UNIT_OPTIONS, resistanceUnitIndex)
  126. const reactiveUnit = getOption(reactiveUnitOptions, reactiveUnitIndex)
  127. const frequencyUnit = getOption(FREQUENCY_UNIT_OPTIONS, frequencyUnitIndex)
  128. const resistanceText = String(source.filterResistanceValue || '')
  129. const reactiveText = String(source.filterReactiveValue || '')
  130. const frequencyText = String(source.filterFrequencyValue || '')
  131. const resistanceNumber = parsePositiveNumber(resistanceText)
  132. const reactiveNumber = parsePositiveNumber(reactiveText)
  133. const frequencyNumber = parsePositiveNumber(frequencyText)
  134. const invalidInput = [resistanceNumber, reactiveNumber, frequencyNumber].some((value) => Number.isNaN(value))
  135. const values = {
  136. frequency: Number.isFinite(frequencyNumber) ? frequencyNumber * frequencyUnit.factor : null,
  137. reactive: Number.isFinite(reactiveNumber) ? reactiveNumber * reactiveUnit.factor : null,
  138. resistance: Number.isFinite(resistanceNumber) ? resistanceNumber * resistanceUnit.factor : null
  139. }
  140. const calculated = invalidInput
  141. ? { computedKey: '', errorText: '输入值需大于 0' }
  142. : calculateMissingValue(network.key, values)
  143. const computedValue = Number.isFinite(calculated.value) ? calculated.value : null
  144. const diagram = getDiagramComponents(network.key, response.key)
  145. let resistanceDisplayUnit = resistanceUnit
  146. let resistanceDisplayUnitIndex = resistanceUnitIndex
  147. let reactiveDisplayUnit = reactiveUnit
  148. let reactiveDisplayUnitIndex = reactiveUnitIndex
  149. let frequencyDisplayUnit = frequencyUnit
  150. let frequencyDisplayUnitIndex = frequencyUnitIndex
  151. let resistanceDisplayValue = Number.isFinite(resistanceNumber)
  152. ? formatMagnitudeNumber(resistanceNumber, { fallbackText: '' })
  153. : resistanceText
  154. let reactiveDisplayValue = Number.isFinite(reactiveNumber)
  155. ? formatMagnitudeNumber(reactiveNumber, { fallbackText: '' })
  156. : reactiveText
  157. let frequencyDisplayValue = Number.isFinite(frequencyNumber)
  158. ? formatMagnitudeNumber(frequencyNumber, { fallbackText: '' })
  159. : frequencyText
  160. if (computedValue !== null && calculated.computedKey === 'resistance') {
  161. const selected = selectBestUnit(RESISTANCE_UNIT_OPTIONS, computedValue, resistanceUnitIndex)
  162. resistanceDisplayUnit = selected.unit
  163. resistanceDisplayUnitIndex = selected.index
  164. resistanceDisplayValue = formatScaledValue(computedValue, resistanceDisplayUnit, { fallbackText: '' })
  165. } else if (computedValue !== null && calculated.computedKey === 'reactive') {
  166. const selected = selectBestUnit(reactiveUnitOptions, computedValue, reactiveUnitIndex)
  167. reactiveDisplayUnit = selected.unit
  168. reactiveDisplayUnitIndex = selected.index
  169. reactiveDisplayValue = formatScaledValue(computedValue, reactiveDisplayUnit, { fallbackText: '' })
  170. } else if (computedValue !== null && calculated.computedKey === 'frequency') {
  171. const selected = selectBestUnit(FREQUENCY_UNIT_OPTIONS, computedValue, frequencyUnitIndex)
  172. frequencyDisplayUnit = selected.unit
  173. frequencyDisplayUnitIndex = selected.index
  174. frequencyDisplayValue = formatScaledValue(computedValue, frequencyDisplayUnit, { fallbackText: '' })
  175. }
  176. return {
  177. filterCapacitanceUnitIndex: network.key === 'rc' ? reactiveDisplayUnitIndex : capacitanceUnitIndex,
  178. filterComputedKey: calculated.computedKey || '',
  179. filterErrorText: calculated.errorText || '',
  180. filterFrequencyDisplayValue: frequencyDisplayValue,
  181. filterFrequencyUnitIndex: frequencyDisplayUnitIndex,
  182. filterFrequencyUnitOptions: FREQUENCY_UNIT_OPTIONS,
  183. filterFrequencyUnitText: frequencyDisplayUnit.label,
  184. filterFrequencyValue: frequencyText,
  185. filterFormulaText: network.key === 'rl' ? 'fc = R / (2πL)' : 'fc = 1 / (2πRC)',
  186. filterInductanceUnitIndex: network.key === 'rl' ? reactiveDisplayUnitIndex : inductanceUnitIndex,
  187. filterNetworkIndex: networkIndex,
  188. filterNetworkKey: network.key,
  189. filterNetworkOptions: NETWORK_OPTIONS,
  190. filterNetworkText: network.label,
  191. filterReactiveDisplayValue: reactiveDisplayValue,
  192. filterReactiveName: network.key === 'rl' ? '电感' : '电容',
  193. filterReactiveSymbol: network.key === 'rl' ? 'L' : 'C',
  194. filterReactiveUnitIndex: reactiveDisplayUnitIndex,
  195. filterReactiveUnitOptions: reactiveUnitOptions,
  196. filterReactiveUnitText: reactiveDisplayUnit.label,
  197. filterReactiveValue: reactiveText,
  198. filterResistanceDisplayValue: resistanceDisplayValue,
  199. filterResistanceUnitIndex: resistanceDisplayUnitIndex,
  200. filterResistanceUnitOptions: RESISTANCE_UNIT_OPTIONS,
  201. filterResistanceUnitText: resistanceDisplayUnit.label,
  202. filterResistanceValue: resistanceText,
  203. filterResponseIndex: responseIndex,
  204. filterResponseKey: response.key,
  205. filterResponseOptions: RESPONSE_OPTIONS,
  206. filterResponseText: response.label,
  207. filterSeriesComponentKey: diagram.series,
  208. filterSeriesComponentLabel: diagram.seriesLabel,
  209. filterShuntComponentKey: diagram.shunt,
  210. filterShuntComponentLabel: diagram.shuntLabel
  211. }
  212. }
  213. function createInitialState() {
  214. return buildState({
  215. filterCapacitanceUnitIndex: 1,
  216. filterFrequencyUnitIndex: 0,
  217. filterInductanceUnitIndex: 3,
  218. filterNetworkIndex: 0,
  219. filterReactiveValue: '',
  220. filterResistanceUnitIndex: 1,
  221. filterResistanceValue: '',
  222. filterResponseIndex: 0
  223. })
  224. }
  225. function updateState(state, changedData = {}) {
  226. return buildState({
  227. ...state,
  228. ...changedData
  229. })
  230. }
  231. function normalizeValue(state, fieldKey, fieldValue) {
  232. const source = buildState(state)
  233. const config = {
  234. frequency: {
  235. options: FREQUENCY_UNIT_OPTIONS,
  236. unitIndex: source.filterFrequencyUnitIndex,
  237. unitIndexKey: 'filterFrequencyUnitIndex',
  238. valueKey: 'filterFrequencyValue'
  239. },
  240. reactive: {
  241. options: source.filterReactiveUnitOptions,
  242. unitIndex: source.filterReactiveUnitIndex,
  243. unitIndexKey: source.filterNetworkKey === 'rl'
  244. ? 'filterInductanceUnitIndex'
  245. : 'filterCapacitanceUnitIndex',
  246. valueKey: 'filterReactiveValue'
  247. },
  248. resistance: {
  249. options: RESISTANCE_UNIT_OPTIONS,
  250. unitIndex: source.filterResistanceUnitIndex,
  251. unitIndexKey: 'filterResistanceUnitIndex',
  252. valueKey: 'filterResistanceValue'
  253. }
  254. }[fieldKey]
  255. if (!config) return source
  256. const text = fieldValue === undefined ? source[config.valueKey] : fieldValue
  257. const numberValue = parsePositiveNumber(text)
  258. if (!Number.isFinite(numberValue)) {
  259. return updateState(source, {
  260. [config.valueKey]: text
  261. })
  262. }
  263. const currentUnit = getOption(config.options, config.unitIndex)
  264. const baseValue = numberValue * currentUnit.factor
  265. const selected = selectBestUnit(config.options, baseValue, config.unitIndex)
  266. return updateState(source, {
  267. [config.unitIndexKey]: selected.index,
  268. [config.valueKey]: formatScaledValue(baseValue, selected.unit, { fallbackText: '' })
  269. })
  270. }
  271. module.exports = {
  272. CAPACITANCE_UNIT_OPTIONS,
  273. FREQUENCY_UNIT_OPTIONS,
  274. INDUCTANCE_UNIT_OPTIONS,
  275. NETWORK_OPTIONS,
  276. RESISTANCE_UNIT_OPTIONS,
  277. RESPONSE_OPTIONS,
  278. createInitialState,
  279. normalizeValue,
  280. updateState
  281. }