global-text-processor.js 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152
  1. import axios from 'axios'
  2. import { delayPerform } from 'js-perform-lock'
  3. function isNumberRegex(str) {
  4. const pureNumberRegex = /^\d+$/
  5. const floatNumberRegex = /^\d*\.\d+$/
  6. const timeRegexHHMM = /^([01]\d|2[0-3]):([0-5]\d)$/
  7. const timeRegexHHMMSS = /^([01]\d|2[0-3]):([0-5]\d):([0-5]\d)$/
  8. const timeRegex = /^\d{4}-\d{2}-\d{2} \d{2}:\d{2}:\d{2}$/
  9. const pureLetterRegex = /^[A-Za-z]+$/
  10. const letterAndDigitRegex = /^(?=.*[A-Za-z])(?=.*\d)[A-Za-z\d]+$/
  11. const letterAndPunctuationRegex = /^(?=.*[A-Za-z])(?=.*[\p{P}\p{S}])[\p{L}\p{P}\p{S}]+$/u
  12. const digitAndPunctuationRegex = /^(?=.*\d)(?=.*[\p{P}\p{S}])[\d\p{P}\p{S}]+$/u
  13. if (pureNumberRegex.test(str)) {
  14. return true
  15. } else if (floatNumberRegex.test(str)) {
  16. return true
  17. } else if (timeRegexHHMM.test(str)) {
  18. return true
  19. } else if (timeRegexHHMMSS.test(str)) {
  20. return true
  21. } else if (pureLetterRegex.test(str)) {
  22. return true
  23. } else if (letterAndDigitRegex.test(str)) {
  24. return true
  25. } else if (letterAndPunctuationRegex.test(str)) {
  26. return true
  27. } else if (digitAndPunctuationRegex.test(str)) {
  28. return true
  29. } else if (timeRegex.test(str)) {
  30. return true
  31. } else {
  32. return false
  33. }
  34. }
  35. var deferredReplacement = (function () {
  36. var keywords = []
  37. var zhixingfun = []
  38. var index = 0
  39. var index2 = 0
  40. function pushNewText(data) {
  41. return new Promise(function (r, j) {
  42. axios
  43. .post('https://jiasm.zfire.top/translate/api/v1/common/translationOfText', data)
  44. .then(response => {
  45. if (response.data.code === 0) {
  46. var obj = {}
  47. response.data.data.map(val => {
  48. obj[val[0]] = val[1]
  49. })
  50. window.Vue_Translation_Of_Text_Data = { ...(window.Vue_Translation_Of_Text_Data || {}), ...obj }
  51. r({ ...obj })
  52. }
  53. })
  54. .catch(j)
  55. })
  56. }
  57. const d = new delayPerform(250).refactor(function (/**可接收参数**/) {
  58. let len = index
  59. let len2 = index2
  60. pushNewText({
  61. keywords: [...keywords],
  62. targetLanguage: window?.Vue_Translation_Of_Text_Type
  63. }).then(data => {
  64. for (var i = 0; i < len2; i++) {
  65. zhixingfun[i]()
  66. }
  67. keywords.splice(0, len)
  68. zhixingfun.splice(0, len2)
  69. index -= len
  70. index2 -= len2
  71. })
  72. })
  73. return function (text, cb) {
  74. d()
  75. if (!keywords.includes(text)) {
  76. index++
  77. keywords.push(text)
  78. }
  79. index2++
  80. zhixingfun.push(cb)
  81. }
  82. })()
  83. export default {
  84. install(Vue) {
  85. Vue.mixin({
  86. updated() {
  87. this.$nextTick(() => {
  88. this.processTextNodes(this.$el)
  89. })
  90. },
  91. methods: {
  92. processTextNodes(el) {
  93. // 遍历子节点并处理文本内容
  94. if (el.nodeType === Node.ELEMENT_NODE) {
  95. // 跳过 table-body中的td 元素
  96. if (el.nodeName.toLowerCase() === 'td') {
  97. return
  98. }
  99. Array.from(el.childNodes).forEach(child => {
  100. if (child.nodeType === Node.TEXT_NODE) {
  101. let text = child.textContent
  102. let _child = child
  103. if (text && !isNumberRegex(text.trim())) {
  104. try {
  105. if (window?.Vue_Translation_Of_Text_Data?.[text.trim()]) {
  106. child.textContent = window?.Vue_Translation_Of_Text_Data?.[text.trim()]
  107. } else if (text.trim()) {
  108. deferredReplacement(text.trim(), function () {
  109. if (window.Vue_Translation_Of_Text_Data[text.trim()]) {
  110. _child.textContent = window.Vue_Translation_Of_Text_Data[text.trim()]
  111. }
  112. })
  113. }
  114. } catch (error) {}
  115. }
  116. } else if (child.nodeName === 'INPUT' || child.nodeName === 'TEXTAREA' || child.nodeName === 'SELECT') {
  117. // 修改输入框和下拉框的 placeholder 内容
  118. let placeholder = child.getAttribute('placeholder')
  119. let _child = child
  120. if (placeholder && !isNumberRegex(placeholder.trim())) {
  121. try {
  122. if (window?.Vue_Translation_Of_Text_Data?.[placeholder.trim() + '']) {
  123. child.setAttribute('placeholder', window?.Vue_Translation_Of_Text_Data?.[placeholder.trim() + ''])
  124. } else if (placeholder.trim() + '') {
  125. deferredReplacement(placeholder.trim() + '', function () {
  126. if (window?.Vue_Translation_Of_Text_Data?.[placeholder.trim() + '']) {
  127. _child.setAttribute(
  128. 'placeholder',
  129. window?.Vue_Translation_Of_Text_Data?.[placeholder.trim() + '']
  130. )
  131. }
  132. })
  133. }
  134. } catch (error) {}
  135. }
  136. } else {
  137. // 递归处理子节点
  138. this.processTextNodes(child)
  139. }
  140. })
  141. }
  142. }
  143. }
  144. })
  145. }
  146. }