content.js 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348
  1. // entrypoints/content.ts
  2. import { Loading } from 'element-plus/es/components/loading/src/service.mjs';
  3. import src1 from '../assets/images/begin.png'
  4. import PageAnalyzer from '../utils/page-analyzer'
  5. import { log } from 'console';
  6. export default defineContentScript({
  7. matches: ["<all_urls>"],
  8. main(ctx) {
  9. let page = document.getElementsByTagName('body')[0];
  10. const src = chrome.runtime.getURL('images/begin.png')
  11. window.pageAnalyzer = new PageAnalyzer();
  12. window.onload = () => {
  13. chrome.runtime.sendMessage(getPageInfo())
  14. }
  15. let form = null
  16. let formChildren = []
  17. let excelDataA = {}
  18. chrome.runtime.onMessage.addListener(async (message, sender, sendResponse) => {
  19. if (message.type === 'GET_PAGE_INFO') {
  20. sendResponse({
  21. data: getPageInfo()
  22. })
  23. }
  24. let dom = null
  25. if (message.type === "GET_TAG_ACTION") {
  26. const data = message.data
  27. if (data.id) {
  28. dom = document.getElementById(data.id)
  29. }
  30. if (data.class) {
  31. dom = [...document.getElementsByTagName(data.tag)]
  32. // .filter(_ => _.className.includes(data.class))
  33. .filter(_ => _.innerText.includes(data.text))
  34. }
  35. console.log(dom);
  36. dom[0].click()
  37. sendResponse({ data: '完成' })
  38. return true
  39. }
  40. if (message.type === 'GET_PAGE_FORM') {
  41. const len = document.querySelectorAll("form").length
  42. if (len) {
  43. const forms = document.querySelectorAll("form")
  44. if (len > 1) {
  45. form = forms[len - 1]
  46. } else form = forms[0]
  47. form.querySelectorAll("svg")
  48. .forEach((el) => el.remove());
  49. formChildren = [...form.elements]
  50. }
  51. // if (!form && document.querySelector("input")) {
  52. // const arr = []
  53. // const inputs = document.querySelectorAll("input")
  54. // formChildren = [...inputs]
  55. // for (const element of inputs) {
  56. // arr.push(element.outerHTML)
  57. // }
  58. // form = { outerHTML: JSON.stringify(arr) }
  59. // }
  60. if (!form) {
  61. sendResponse({
  62. status: 'error',
  63. message: '没有找到表单'
  64. })
  65. return
  66. }
  67. sendResponse({
  68. status: 'ok',
  69. data: form.outerHTML
  70. })
  71. }
  72. if (message.type === "INPUT_FORM") {
  73. const { formData, excelData } = message.data
  74. excelDataA = excelData
  75. console.log(formData, excelData);
  76. await handleFillInput(formData, 0)
  77. }
  78. return true
  79. });
  80. function getPageInfo() {
  81. const favIconUrl = getFavicon()
  82. return {
  83. type: "PAGE_INFO",
  84. data: {
  85. favIconUrl,
  86. title: document.title,
  87. url: window.location.href,
  88. content: window.pageAnalyzer.analyzePage()
  89. },
  90. }
  91. }
  92. function getFavicon() {
  93. // 尝试获取动态favicon
  94. const iconLinks = Array.from(
  95. document.querySelectorAll('link[rel*="icon"]')
  96. );
  97. const favIconUrl = iconLinks
  98. .sort((a, b) => {
  99. // 优先使用大尺寸图标
  100. const sizeA = parseInt(a.sizes?.value) || 0;
  101. const sizeB = parseInt(b.sizes?.value) || 0;
  102. return sizeB - sizeA;
  103. })
  104. .map((link) => link.href)
  105. .find(Boolean);
  106. if (favIconUrl) return favIconUrl;
  107. // 如果没有找到,返回网站根目录的favicon.ico
  108. const url = new URL(window.location.href);
  109. return `${url.protocol}//${url.hostname}/favicon.ico`;
  110. }
  111. const handleFillInput = async (data, index) => {
  112. console.log(data);
  113. for (let i = 0; i < data.length; i++) {
  114. const item = data[i]
  115. if (item.findBy === 'id') {
  116. const input = formChildren.find(child => child.id === item.findByValue)
  117. if (item.type === 'text' || item.type === 'textarea' || item.type === 'number') {
  118. if (!input) {
  119. if (item.label) {
  120. const label = [...form.getElementsByTagName('label')].find(label => label.innerText.includes(item.label))
  121. if (label) {
  122. const input = findLabelForInput(label)
  123. if (input) {
  124. await simulateUserInput(input, excelDataA[item.excelColumn][index])
  125. }
  126. }
  127. }
  128. }
  129. if (input) {
  130. await simulateUserInput(input, excelDataA[item.excelColumn][index])
  131. }
  132. }
  133. if (item.type === 'radio' || item.type === 'checkbox') {
  134. if (item.label) {
  135. const label = [...form.getElementsByTagName('label')].find(label => label.innerText.includes(item.label))
  136. if (label) {
  137. const span = findLabelForSpan(label)
  138. span.forEach(span => {
  139. span.innerText === excelDataA[item.excelColumn][index] && span.click()
  140. })
  141. }
  142. }
  143. }
  144. if (item.type === 'date') {
  145. const input = formChildren.find(child => child.id === item.findByValue)
  146. if (excelDataA[item.excelColumn][index]) {
  147. await simulateCompleteUserAction(input, input, formatDate(excelDataA[item.excelColumn][index]), formatDate(excelDataA[item.excelColumn][index]))
  148. }
  149. }
  150. } else if (item.findBy === 'placeholder') {
  151. const input = formChildren.find(child => child.placeholder === item.findByValue)
  152. if (input) {
  153. simulateUserInput(input, excelDataA[item.excelColumn][index])
  154. }
  155. }
  156. }
  157. }
  158. const simulateUserInput = async (element, value) => {
  159. // 设置值
  160. // if (element.tagName.toLowerCase() === 'textarea') {
  161. // element.focus()
  162. // element.value = value
  163. // element.blur()
  164. // return
  165. // }
  166. element.value = value;
  167. // 创建并触发 input 事件
  168. const inputEvent = new Event('input', { bubbles: true });
  169. element.dispatchEvent(inputEvent);
  170. // 创建并触发 change 事件
  171. const changeEvent = new Event('change', { bubbles: true });
  172. element.dispatchEvent(changeEvent);
  173. // 可选:如果表单使用了 React/Vue 等框架,可能还需要触发以下事件
  174. element.dispatchEvent(new Event('blur'));
  175. element.dispatchEvent(new KeyboardEvent('keyup', { key: 'Enter' }));
  176. }
  177. const findLabelForInput = (label) => {
  178. let p = label.parentElement
  179. for (let i = 0; i < 10; i++) {
  180. const input = p.getElementsByTagName('input')
  181. if (input.length > 0) {
  182. return input[0]
  183. }
  184. p = p.parentElement
  185. }
  186. return null
  187. }
  188. const findLabelForSpan = (label) => {
  189. let p = label.parentElement
  190. for (let i = 0; i < 10; i++) {
  191. const span = p.getElementsByTagName('span')
  192. if (span.length > 0) {
  193. return [...span]
  194. }
  195. p = p.parentElement
  196. }
  197. return null
  198. }
  199. const simulateCompleteUserAction = async (clickElement, inputElement, inputText, tdTitle) => {
  200. // 1. 模拟鼠标弹起事件
  201. const simulateMouseUp = (element) => {
  202. const mouseUpEvent = new MouseEvent('mouseup', {
  203. bubbles: true,
  204. cancelable: true,
  205. view: window,
  206. button: 0,
  207. buttons: 0,
  208. clientX: 0,
  209. clientY: 0,
  210. detail: 1
  211. });
  212. element.dispatchEvent(mouseUpEvent);
  213. };
  214. // 123456qq?
  215. // 2. 模拟键盘输入
  216. const simulateTyping = async (element, text, delay = 50) => {
  217. element.focus();
  218. for (let char of text) {
  219. await new Promise(resolve => setTimeout(resolve, delay));
  220. // 按键按下
  221. const keydownEvent = new KeyboardEvent('keydown', {
  222. key: char,
  223. code: `Key${char.toUpperCase()}`,
  224. bubbles: true,
  225. cancelable: true
  226. });
  227. element.dispatchEvent(keydownEvent);
  228. // 更新输入值
  229. element.value += char;
  230. // 触发输入事件
  231. const inputEvent = new InputEvent('input', {
  232. bubbles: true,
  233. cancelable: true,
  234. data: char,
  235. inputType: 'insertText'
  236. });
  237. element.dispatchEvent(inputEvent);
  238. // 按键弹起
  239. const keyupEvent = new KeyboardEvent('keyup', {
  240. key: char,
  241. code: `Key${char.toUpperCase()}`,
  242. bubbles: true,
  243. cancelable: true
  244. });
  245. element.dispatchEvent(keyupEvent);
  246. }
  247. // 触发change事件
  248. element.dispatchEvent(new Event('change', { bubbles: true }));
  249. };
  250. // 3. 查找td元素
  251. const findTdByTitle = (title, timeout = 5000) => {
  252. return new Promise((resolve, reject) => {
  253. const startTime = Date.now();
  254. const find = () => {
  255. const td = document.querySelector(`td[title="${title}"]`);
  256. if (td) {
  257. resolve(td);
  258. return;
  259. }
  260. if (Date.now() - startTime > timeout) {
  261. reject(new Error(`未找到title为"${title}"的td元素`));
  262. return;
  263. }
  264. requestAnimationFrame(find);
  265. };
  266. find();
  267. });
  268. };
  269. // 4. 模拟点击事件
  270. const simulateClick = (element) => {
  271. const clickEvent = new MouseEvent('click', {
  272. bubbles: true,
  273. cancelable: true,
  274. view: window,
  275. detail: 1
  276. });
  277. element.dispatchEvent(clickEvent);
  278. };
  279. try {
  280. // 执行操作序列
  281. // 1. 触发鼠标弹起
  282. simulateMouseUp(clickElement);
  283. await new Promise(resolve => setTimeout(resolve, 100));
  284. // 2. 模拟键盘输入
  285. await simulateTyping(inputElement, inputText);
  286. await new Promise(resolve => setTimeout(resolve, 200));
  287. // 3. 查找并点击td元素
  288. // const tdElement = await findTdByTitle(tdTitle);
  289. const enterKeyEvent = new KeyboardEvent('keydown', {
  290. key: 'Enter', // 事件键名
  291. code: 'Enter', // 物理按键编码
  292. keyCode: 13, // 传统键码(Enter键为13)
  293. which: 13, // 同keyCode
  294. bubbles: true, // 允许事件冒泡
  295. cancelable: true // 允许事件被取消
  296. });
  297. setTimeout(() => {
  298. inputElement.dispatchEvent(enterKeyEvent);
  299. }, 0)
  300. await new Promise(resolve => setTimeout(resolve, 500));
  301. return true;
  302. } catch (error) {
  303. throw error;
  304. }
  305. };
  306. function formatDate(date) {
  307. // 直接创建北京时间的日期对象
  308. const d = new Date(date);
  309. // 获取年、月、日
  310. const year = d.getFullYear();
  311. const month = String(d.getMonth() + 1).padStart(2, '0'); // 月份从0开始,需要加1
  312. const day = String(d.getDate()).padStart(2, '0');
  313. return `${year}-${month}-${day}`;
  314. }
  315. },
  316. });