global-text-processor.js 9.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257
  1. import axios from 'axios'
  2. import Vue from 'vue'
  3. import { delayPerform } from 'js-perform-lock'
  4. const WindowsTranslateApi = 'https://jiasm.zfire.top/translate'
  5. // const WindowsTranslateApi = 'http://127.0.0.1:7001'
  6. // 数据过滤校验
  7. function isNumberRegex(str) {
  8. const regexPatterns = [
  9. /^\d+$/, // 纯数字
  10. /^\d*\.\d+$/, // 浮点数
  11. /^([01]\d|2[0-3]):([0-5]\d)$/, // 时间 (HH:MM)
  12. /^([01]\d|2[0-3]):([0-5]\d):([0-5]\d)$/, // 时间 (HH:MM:SS)
  13. /^\d{4}-\d{2}-\d{2} \d{2}:\d{2}:\d{2}$/, // 完整时间戳
  14. /^[A-Za-z]+$/, // 纯字母
  15. /^(?=.*[A-Za-z])(?=.*\d)[A-Za-z\d]+$/, // 字母和数字
  16. /^(?=.*[A-Za-z])(?=.*[\p{P}\p{S}])[\p{L}\p{P}\p{S}]+$/u, // 字母和标点符号
  17. /^(?=.*\d)(?=.*[\p{P}\p{S}])[\d\p{P}\p{S}]+$/u, // 数字和标点符号
  18. /^[\p{P}\s\dA-Za-z]+$/u // 字母、数字和标点符号
  19. ]
  20. return regexPatterns.some(pattern => pattern.test(str)) || !!window.Vue_Translation_Of_Text_Old_Data[str]
  21. }
  22. // 文本转译请求
  23. function pushNewText(data) {
  24. return new Promise(function (r, j) {
  25. axios
  26. .post(WindowsTranslateApi + '/api/v1/common/translationOfText', data)
  27. .then(response => {
  28. if (response.data.code === 0) {
  29. var obj = {}
  30. var obj2 = {}
  31. response.data.data.map(val => {
  32. obj[val[0]] = val[1]
  33. obj2[val[1]] = true
  34. })
  35. window.Vue_Translation_Of_Text_Old_Data = { ...(window.Vue_Translation_Of_Text_Old_Data || {}), ...obj2 }
  36. window.Vue_Translation_Of_Text_Data = { ...(window.Vue_Translation_Of_Text_Data || {}), ...obj }
  37. r({ ...obj })
  38. }
  39. })
  40. .catch(j)
  41. })
  42. }
  43. var deferredReplacement = (function () {
  44. var allKeywords = []
  45. var keywords = []
  46. var zhixingfun = []
  47. var index = 0
  48. var index2 = 0
  49. const dInit = new delayPerform(250).refactor(function () {
  50. let len = index
  51. let len2 = index2
  52. let keywords_ = keywords.splice(0, len)
  53. let zhixingfun_ = zhixingfun.splice(0, len2)
  54. index -= len
  55. index2 -= len2
  56. if (keywords_.length > 0) {
  57. pushNewText({
  58. keywords: keywords_,
  59. targetLanguage: window?.Vue_Translation_Of_Text_Type
  60. }).then(data => {
  61. for (var i = 0; i < len2; i++) {
  62. zhixingfun_?.[i]?.()
  63. }
  64. })
  65. } else if (zhixingfun_.length > 0) {
  66. setTimeout(function () {
  67. for (var i = 0; i < len2; i++) {
  68. zhixingfun_?.[i]?.()
  69. }
  70. }, 250)
  71. }
  72. })
  73. return function (text, cb) {
  74. dInit()
  75. if (!allKeywords.includes(text)) {
  76. index++
  77. allKeywords.push(text)
  78. keywords.push(text)
  79. }
  80. index2++
  81. zhixingfun.push(cb)
  82. }
  83. })()
  84. export const GlobalTextProcessor = {
  85. install(Vue) {
  86. Vue.mixin({
  87. updated() {
  88. this.$nextTick(() => {
  89. this.processTextNodes(this.$el)
  90. // this.autoSelect('operation-btns')
  91. // this.autoSelect('list-display-control-view')
  92. this.autoHtmlSelect('title')
  93. })
  94. },
  95. methods: {
  96. // 通过class修改
  97. autoSelect(name) {
  98. document.querySelectorAll(`.${name}`).forEach(element => {
  99. this.processAbsolutely(element)
  100. })
  101. },
  102. // 通过标签名修改
  103. autoHtmlSelect(name) {
  104. document.getElementsByTagName(name).forEach(element => {
  105. this.processAbsolutely(element)
  106. })
  107. },
  108. processTextNodes(el) {
  109. // 遍历子节点并处理文本内容
  110. if (el.nodeType === Node.ELEMENT_NODE) {
  111. // 跳过 table-body中的td 元素
  112. // if (el.nodeName.toLowerCase() === 'td') {
  113. // return
  114. // }
  115. Array.from(el.childNodes).forEach(child => {
  116. if (child.nodeType === Node.TEXT_NODE) {
  117. let text = child.textContent
  118. let _child = child
  119. if (text && !isNumberRegex(text.trim())) {
  120. try {
  121. if (window?.Vue_Translation_Of_Text_Data?.[text.trim()]) {
  122. child.textContent = window?.Vue_Translation_Of_Text_Data?.[text.trim()]
  123. } else if (text.trim()) {
  124. deferredReplacement(text.trim(), function () {
  125. if (window.Vue_Translation_Of_Text_Data[text.trim()]) {
  126. _child.textContent = window.Vue_Translation_Of_Text_Data[text.trim()]
  127. }
  128. })
  129. }
  130. } catch (error) {}
  131. }
  132. } else if (child.nodeName === 'INPUT' || child.nodeName === 'TEXTAREA' || child.nodeName === 'SELECT') {
  133. // 修改输入框和下拉框的 placeholder 内容
  134. let placeholder = child.getAttribute('placeholder')
  135. let _child = child
  136. if (placeholder && !isNumberRegex(placeholder.trim())) {
  137. try {
  138. if (window?.Vue_Translation_Of_Text_Data?.[placeholder.trim() + '']) {
  139. child.setAttribute('placeholder', window?.Vue_Translation_Of_Text_Data?.[placeholder.trim() + ''])
  140. } else if (placeholder.trim() + '') {
  141. deferredReplacement(placeholder.trim() + '', function () {
  142. if (window?.Vue_Translation_Of_Text_Data?.[placeholder.trim() + '']) {
  143. _child.setAttribute(
  144. 'placeholder',
  145. window?.Vue_Translation_Of_Text_Data?.[placeholder.trim() + '']
  146. )
  147. }
  148. })
  149. }
  150. } catch (error) {}
  151. }
  152. } else if (child?.classList?.contains?.('el-select')) {
  153. // 处理 class 为 el-select 的组件
  154. this.processSelect(child)
  155. } else {
  156. // 递归处理子节点
  157. this.processTextNodes(child)
  158. }
  159. })
  160. }
  161. },
  162. processSelect(el) {
  163. // 遍历子节点并处理文本内容
  164. if (el.nodeType === Node.ELEMENT_NODE) {
  165. Array.from(el.childNodes).forEach(child => {
  166. if (child.nodeName === 'INPUT') {
  167. // 修改输入框和下拉框的 placeholder 内容
  168. let value = child.value
  169. let _child = child
  170. if (value && !isNumberRegex(value.trim())) {
  171. try {
  172. if (window?.Vue_Translation_Of_Text_Data?.[value.trim() + '']) {
  173. child.value = window?.Vue_Translation_Of_Text_Data?.[value.trim() + '']
  174. } else if (value.trim() + '') {
  175. deferredReplacement(value.trim() + '', function () {
  176. if (window?.Vue_Translation_Of_Text_Data?.[value.trim() + '']) {
  177. _child.value = window?.Vue_Translation_Of_Text_Data?.[value.trim() + '']
  178. }
  179. })
  180. }
  181. } catch (error) {}
  182. }
  183. } else {
  184. // 递归处理子节点
  185. this.processSelect(child)
  186. }
  187. })
  188. }
  189. },
  190. processAbsolutely(el) {
  191. // 遍历子节点并处理文本内容
  192. if (el.nodeType === Node.ELEMENT_NODE) {
  193. Array.from(el.childNodes).forEach(child => {
  194. if (child.nodeType === Node.TEXT_NODE) {
  195. let text = child.textContent
  196. let _child = child
  197. if (text && !isNumberRegex(text.trim())) {
  198. try {
  199. if (window?.Vue_Translation_Of_Text_Data?.[text.trim()]) {
  200. child.textContent = window?.Vue_Translation_Of_Text_Data?.[text.trim()]
  201. } else if (text.trim()) {
  202. deferredReplacement(text.trim(), function () {
  203. if (window.Vue_Translation_Of_Text_Data[text.trim()]) {
  204. _child.textContent = window.Vue_Translation_Of_Text_Data[text.trim()]
  205. }
  206. })
  207. }
  208. } catch (error) {}
  209. }
  210. } else {
  211. // 递归处理子节点
  212. this.processAbsolutely(child)
  213. }
  214. })
  215. }
  216. }
  217. }
  218. })
  219. }
  220. }
  221. export function translaBeforeRegistration(cb) {
  222. window.Vue_Translation_Of_Text_Type = window.localStorage.getItem('Vue_Translation_Of_Text_Type')
  223. if (
  224. window?.Vue_Translation_Of_Text_Type &&
  225. ['en', 'ar', 'tr', 'ru', 'ja'].includes(window?.Vue_Translation_Of_Text_Type)
  226. ) {
  227. Vue.use(GlobalTextProcessor) // 注册插件
  228. axios
  229. .post(WindowsTranslateApi + '/api/v1/common/readTranslationOfText', {
  230. targetLanguage: window?.Vue_Translation_Of_Text_Type
  231. })
  232. .then(({ data }) => {
  233. var obj = {}
  234. var obj2 = {}
  235. data?.data?.map(item => {
  236. obj[item.source] = item.target
  237. obj2[item.target] = true
  238. })
  239. window.Vue_Translation_Of_Text_Old_Data = { ...(window.Vue_Translation_Of_Text_Old_Data || {}), ...obj2 }
  240. window.Vue_Translation_Of_Text_Data = { ...(window.Vue_Translation_Of_Text_Data || {}), ...obj }
  241. cb?.()
  242. })
  243. .catch(error => {
  244. cb?.()
  245. })
  246. } else {
  247. cb?.()
  248. }
  249. }