您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
快速跳转至指定包或指定分支
// ==UserScript== // @name 快速查包 // @namespace fsh // @version 7.0 // @description 快速跳转至指定包或指定分支 // @author xxtest // @match *://ci.meitu.city/* // @match *://omnibus.meitu-int.com/* // @match *://ios.meitu-int.com/ipa/* // @match *://jira.meitu.com/* // @match *://cf.meitu.com/* // @grant GM_xmlhttpRequest // @grant GM_setValue // @grant GM_getValue // @grant GM_deleteValue // @require https://cdnjs.cloudflare.com/ajax/libs/jquery/3.3.1/jquery.js // @homepage https://greasyfork.org/zh-CN/scripts/454567-%E5%BF%AB%E9%80%9F%E6%9F%A5%E5%8C%85 // @license MIT // @note 7.0 logwork打卡标签新增证件照 // @note 6.9 logwork打卡标签新增订阅、商业化 // @note 6.8 logwork打卡弹窗选项,新增视频美化、视频美容 // @note 6.7 logwork打卡弹框,下拉框选型增加全部;一键复制需求bug修复 // @note 6.6 logwork打卡弹框,支持下拉框填入标签名 // @note 6.5 修复bug,解决鸿蒙和web平台影响版本判断问题 // @note 6.4 创建bug判断平台和影响版本是否一致 // @note 6.3 解决Jira获取分支,偶尔会填写错误的问题 // @note 6.2 解决iOS魔法屋跑图异常问题 // @note 6.1 解决跨年日期获取不到版本号问题 // @note 6.0 一键复制需求列表,增加容错 // ==/UserScript== (function () { 'use strict'; // 设置刷新时间 const refreshTime = 1000; // 添加CSS $('head').append($(` <style> .search{ position: relative; width: 220px; } .search input{ height: 40px; width: 220px; border-radius: 40px; border: 2px solid #324B4E; background: #F9F0DA; transition: .3s linear; float: left; text-indent: 10px; } .search input:focus::placeholder{ opacity: 0; } .search button{ height: 37px; border-radius: 37px; border-right: 1px solid #324B4E; border-left: 0px; background: #F9F0DA; right: 0; position: absolute; } .btn_find_build{ width: 40px; text-align: center; } .search_span{ /*border: 1px dashed #000;*/ cursor: pointer; height: 16px; margin: 0px 10px; padding: 3px; border-radius: 25px; } .to_new_build{ height: 30px; width: 120px; border-radius: 42px; border: 1px solid #324B4E; background: #fff; transition: .3s linear; color: #544d4d; margin: 0px 10px; font-size:14px; } .myimg{ height:16px; } #input_last_build{ border-radius: 30px; border: 1px solid #324B4E; } </style>`)); // 当是ci域名时,才触发后续的操作,如果不是则不触发 // CI // CI // CI if (location.href.indexOf('ci.meitu.city') > 0) { clearInterval(refreshTime); setInterval(function () { let input_find_build = document.getElementById('input_find_build'); let to_new_build = document.getElementById('to_new_build'); // 不存在输入框和跳转按钮则添加 if (!input_find_build) { addFindButtonCicity(); // 当用户按下键盘上的某个键时触发 document.addEventListener("keyup", function (event) { // 如果按下的是回车键 if (event.keyCode === 13) { // 触发按钮的点击事件 $('#btn_find_build').click(); } }); // 跳转按钮点击 $('#btn_find_build').unbind("click").click(function () { // 获取当前url, 用正则表达式获取项目名 let localUrl = window.location.href let project = '' let reg = /(?<=build\/)\w*/ project = localUrl.match(reg)[0] if (project != null) { let baseUrl = 'https://ci.meitu.city/build/' + project + '/' // 获取用户输入 let input_build_content = document.getElementById("input_find_build").value.trim() if (input_build_content === "") { return false; } let targetUrl = "" // 如果输入的是纯数字,视为build id if (/^\d+$/.test(input_build_content)) { targetUrl = baseUrl + 'number/' + input_build_content } else {// 否则按输入的是分支处理 input_build_content = input_build_content.replaceAll("/", "%2F") targetUrl = baseUrl + 'branch/' + input_build_content } // 跳转到指定页面 window.location.href = targetUrl } else { input_find_build.value = "未能正确获取项目名称" } }); } // 不存在跳转至最新按钮则添加 if (!to_new_build) { addToNewCicity(); $('.to_new_build').unbind("click").click(function () { // 判断当前所处的页面, 用正则表达式获取项目名 let localUrl = window.location.href let project = '' let reg = /(?<=build\/)\w*/ project = localUrl.match(reg)[0] // 获取所要跳转的分支名 let full_build_name = this.parentNode.children[0].innerText.trim() let targetUrl = "" let baseUrl = 'https://ci.meitu.city/build/' + project + '/branch/' full_build_name = full_build_name.replaceAll("/", "%2F") targetUrl = baseUrl + full_build_name // 跳转到指定页面 window.location.href = targetUrl }); } }, refreshTime); } if (location.href.indexOf('omnibus.meitu-int.com') > 0) { clearInterval(refreshTime); setInterval(function () { let input_find_build = document.getElementById('input_find_build'); let to_new_build = document.getElementById('to_new_build'); let input_last_build = document.getElementById('input_last_build'); let build_magichouse = document.getElementById('build_magichouse'); // 不存在输入框和跳转按钮则添加 if (!input_find_build) { addFindButtonOmnibus(); // 当用户按下键盘上的某个键时触发 document.addEventListener("keyup", function (event) { // 如果按下的是回车键 if (event.keyCode === 13) { // 触发按钮的点击事件 $('#btn_find_build').click(); } }); // 跳转按钮点击 $('#btn_find_build').unbind("click").click(function () { // 获取当前url, 用正则表达式获取项目名 let localUrl = window.location.href let project = '' let platform = '' let reg_platform = /:\/\/[^\/]+\/apps\/[^:]+:(\w+)/; let reg = /\/apps\/([^/:]+)/ project = localUrl.match(reg)[1] if (project != null) { console.log(project.length) let baseUrl = "" if (project.length < 10) { // 适配iOS的情况 platform = localUrl.match(reg_platform)[1] console.log(platform) baseUrl = ' https://omnibus.meitu-int.com/apps/' + project + ':' + platform + '/build' } else { baseUrl = ' https://omnibus.meitu-int.com/apps/' + project + '/build' } console.log(baseUrl) // 获取用户输入 let input_build_content = document.getElementById("input_find_build").value.trim() if (input_build_content === "") { return false; } let targetUrl = "" // 如果输入的是纯数字,视为build id if (/^\d+$/.test(input_build_content)) { targetUrl = baseUrl + '/number/' + input_build_content } else {// 否则按输入的是分支处理 input_build_content = input_build_content.replaceAll("/", "%2F") targetUrl = baseUrl + '?branch=' + input_build_content } console.log(targetUrl) // 跳转到指定页面 window.location.href = targetUrl } else { input_find_build.value = "未能正确获取项目名称" } }); } // 不存在跳转至最新按钮则添加 if (!to_new_build) { addToNewOmnibus(); $('.to_new_build').unbind("click").click(function () { let localUrl = window.location.href // 项目名 let project = '' // 平台 let platform = '' let reg_platform = /:\/\/[^\/]+\/apps\/[^:]+:(\w+)/; let reg = /\/apps\/([^/:]+)/ project = localUrl.match(reg)[1] // 获取对应节点,因重复节点较多只能这么处理 let parent_node = this.parentNode.parentNode.parentNode let bracnh_span = parent_node.children[2].children[0].children[1].querySelector('span.el-link__inner') // 获取所要跳转的分支名 let full_build_name = bracnh_span.innerText.trim() if (project != null) { let baseUrl = "" // 区分通过appuid还是通过项目:平台 if (project.length < 10) { // 适配iOS的情况,获取对应平台 platform = localUrl.match(reg_platform)[1] baseUrl = ' https://omnibus.meitu-int.com/apps/' + project + ':' + platform + '/build?branch=' + full_build_name } else { baseUrl = ' https://omnibus.meitu-int.com/apps/' + project + '/build?branch=' + full_build_name } // 跳转到指定页面 window.location.href = baseUrl } }); } // 不存在魔法屋跑图按钮则添加 if (!build_magichouse) { addOmnibusMagichouse(); $('.build_magichouse').unbind("click").click(function () { let buildIdB = document.getElementById("input_last_build").value.trim() if (buildIdB === "") { alert("请输入上个版本提交的build id"); return false; } let buildIdA = this.parentNode.parentNode.parentNode.children[0].children[0].children[0].innerText.trim() // 获取请求的URL let url = window.location.href //分割,获取平台(可能是mtxx:ios,也可能是aze3897z9xi8g3dzvw8ce6thc5) let path_url = url.split('/'); let platform = path_url.includes('apps') ? path_url[path_url.indexOf('apps') + 1] : null; if (confirm("确定构建魔法屋任务吗?点击确定构建")) { if (platform == "hydsriakywi7a4wtukhxh6zt6q" || platform.split(":")[1] == 'android') { buildMagichouseTask(2, buildIdA, buildIdB); } else if (platform == "aze3897z9xi8g3dzvw8ce6thc5" || platform.split(":")[1] == 'ios') { buildMagichouseTask(0, buildIdA, buildIdB); } else { console.log(platform); alert("出现错误,请联系阿德"); } } }); } }, refreshTime); } var project_android_omnibus = { "美图秀秀": "mtxx", "美颜相机": "beautycam", "美拍": "meipai", "美妆相机": "makeup", "潮自拍": "selfiecity", "设计室": "mtsjs", "wink": "wink", "BeautyPlus": "beautyplus", "AirBrush": "airbrush", "eve": "eve", "chic": "chic", "美图宜肤V": "eveking", "EveNetAssist": "evenetassist", "VChat": "vchatbeauty", "vcut": "vcut", "Vmake": "beautyplusvideo", "PixEngine": "pixengine", "智肤APP": "skinar", "美图秀秀Starii": "starii" } var appuid_android = { "美图秀秀": "hydsriakywi7a4wtukhxh6zt6q", "美颜相机": "afpqgy5mqyiyejvj7tu5pvkb3s", "美拍": "a3cat9d2fnieu5ghyjr6uvysd7", "美妆相机": "e69r7unz2bi2rm4d5hhmejunjt", "潮自拍": "au89qds8ipjnfmq6y36yxrwwqr", "设计室": "e57ixjbvhjj3pkjtqbircekw4g", "wink": "a7wb9ngehrjhs5s6kbc8jhpmut", "BeautyPlus": "b4ecchhcsgibw5kfamgxdr4z2f", "AirBrush": "ap6s3sujqjjze3nzd8e46sgwg8", "eve": "bqn9z5twij9d3jqjgpg6dvws3", "chic": "ajbmhzdremjtk5nkx9eezbc2ps", "美图宜肤V": "cprbcrdb58jwxjne7z5daitprd", "EveNetAssist": "epynw2drmbjyz5amwe4eirqm3f", "VChat": "envi6ei33biw7kn9zi42t2yqm2", "PixEngine": "pixengine", "智肤APP": "ecsgbwkuj5iffmjg3jzjq82ti4", "美图秀秀Starii": "enri33faz6ibwj5t7f9jpq4cvi" } var appuid_ios = { "美图秀秀": "aze3897z9xi8g3dzvw8ce6thc5", "美颜相机": "cd2z8rxy5uic9iaz6dbjv8xhw6", "美拍": "e79mtyimnwixp46e6hvspus6v2", "美妆相机": "ffp224ksztiqhi58ud7n33fvs2", "潮自拍": "etiv8nbxjpihni6cr9qhba4u3r", "设计室": "gki88wa2nfiqxi7vhyf2xux6cw", "wink": "bhks37k3uaizn4yvcspt4j3nma", "BeautyPlus": "eprvgz2kwujgvmyt7bvw92fae7", "AirBrush": "e8new3bg3eiximneygeizjyrn6", "eve": "b3huanjy32ipcmhkq6cfnvvspp", "chic": "hfkv4chfbgj56kuphtpe656t6g", "美图宜肤V": "dcjubb2t6ajvyjxaymkvtpp5az", "PixEngine": "pixengine", "智肤APP": "ecsgbwkuj5iffmjg3jzjq82ti4", "美图秀秀Starii": "hmeb767rjnjh8i3vmy3rwmkxn7" } var project_ios_omnibus = { "美图秀秀": "mtxx", "美颜相机": "beautycam", "美拍": "meipai", "美妆相机": "makeup", "潮自拍": "selfiecity", "设计室": "mtsjs", "wink": "wink", "BeautyPlus": "beautyplus", "AirBrush": "airbrush", "eve": "eve", "chic": "chic", "美图宜肤V": "eveking", "EveNetAssist": "evenetassist", "VChat": "vchatbeauty", "vcut": "vcut", "Vmake": "beautyplusvideo", "PixEngine": "pixengine", "智肤APP": "skinar", "美图秀秀Starii": "starii" } // omnibus页面,添加输入框和跳转按钮 function addFindButtonOmnibus() { let span = $('<main class="search" style="display:inline-block;margin-left: 150px;" ></main>') let input_find_build = $('<input type="text" class="text" id="input_find_build" placeholder=" 输入Build id或分支名">'); let btn_find_build = $('<button type="submit" class="btn_find_build" id="btn_find_build">🔍</button>'); span.append(input_find_build); span.append(btn_find_build); // jQuery不能直接插入,需要转一下格式,转成HTML才行 let spanHtml = span.prop('outerHTML'); // 获取第一个再插入; let firstDivWithTitleClass = document.querySelector('.is-plain'); firstDivWithTitleClass.insertAdjacentHTML('beforebegin', spanHtml); } // omnibus页面,添加跳转至最新按钮 function addToNewOmnibus() { let to_new_build = $('<button type="submit" class="to_new_build" id="to_new_build">跳转至最新➔</button>'); $(".el-text--primary").after(to_new_build); } // omnibus页面,在其后方添加魔法屋跑图任务按钮 function addOmnibusMagichouse() { let input_last_build = $('<input type="text" class="text" id="input_last_build" placeholder=" 输入上个版本正式包">'); let build_magichouse = $('<button type="button" class="build_magichouse" id="build_magichouse">魔法屋跑图</button>'); $(".to_new_build").after(build_magichouse); $(".to_new_build").after(input_last_build); } function buildMagichouseTask(osType, buildIdA, buildIdB) { let url = "http://mh-mng.meitu-int.com/xianyao/schedule/run/report"; let module_list = [2, 3, 4, 5] for (var i = 0; i < module_list.length; i++) { var module = module_list[i] GM_xmlhttpRequest({ url: url, method: 'POST', headers: { "Content-Type": "application/json" }, data: JSON.stringify({ osType: osType, module: module, buildIdA: buildIdA, buildIdB: buildIdB, creator: "[email protected]" }), onload: function (res) { console.log(res) }, onerror: function (err) { result = '接口请求失败,建议重新关闭开启脚本再试试'; } }); } if (confirm("点击确定跳转魔法屋查看任务,如无任务联系阿德")) { window.open('https://magichouse.meitu-int.com/check/media/run/tasks', '_blank'); } } // CI页面,添加输入框和跳转按钮 function addFindButtonCicity() { let span = $('<span class="search"></span>') let input_find_build = $('<input type="text" class="text" id="input_find_build" placeholder=" 输入Build id或分支名">'); let btn_find_build = $('<button type="submit" class="btn_find_build" id="btn_find_build">🔍</button>'); span.append(input_find_build); span.append(btn_find_build); $(".project-label__name").after(span); } // CI页面,在分支名后方添加跳转至最新按钮 function addToNewCicity() { let to_new_build = $('<button type="submit" class="to_new_build" id="to_new_build">跳转至最新➔</button>'); $(".message-card__subtitle").after(to_new_build); } // 当是granary域名时,才触发后续的操作,如果不是则不触发 // Granary // Granary // Granary if (location.href.indexOf('ios.meitu-int.com') > 0) { clearInterval(refreshTime); setInterval(function () { let input_find_build = document.getElementById('input_find_build'); let to_new_build = document.getElementById('to_new_build'); let input_last_build = document.getElementById('input_last_build'); let build_magichouse = document.getElementById('build_magichouse'); // 项目名 var project = '' var reg = /(?<=ipa\/)\w*/ // 当前页面Url, 用正则表达式获取项目名 var localUrl = window.location.href project = localUrl.match(reg)[0] // 不存在输入框和跳转按钮则添加 if (!input_find_build) { addFindButtonGranary(); // 当用户按下键盘上的某个键时触发 document.addEventListener("keyup", function (event) { // 如果按下的是回车键 if (event.keyCode === 13) { // 触发按钮的点击事件 $('#btn_find_build').click(); } }); // 按钮绑定点击事件 $('#btn_find_build').unbind("click").click(function () { if (project != null) { // 获取用户输入 let input_build_content = document.getElementById("input_find_build").value.trim() if (input_build_content === "") { return false; } let targetUrl = "" // 如果输入的是纯数字,视为build id if (/^\d+$/.test(input_build_content)) { let baseUrl = 'http://ios.meitu-int.com/ipa/' + project + '/build/' targetUrl = baseUrl + input_build_content } else {// 否则按分支处理 let baseUrl = 'http://ios.meitu-int.com/ipa/' + project + '/' input_build_content = input_build_content.replaceAll("/", "%2F") targetUrl = baseUrl + input_build_content } // 跳转到指定页面 window.location.href = targetUrl } else { input_build_content.value = "未能正确获取项目名称" } }); } //不存在跳转至最新按钮则添加 if (!to_new_build) { addToNewGranary(); // 跳转按钮点击 $('.to_new_build').unbind("click").click(function (event) { // 分支名,去掉首尾的字符串,“/”转为“%2F” let parentNode = event.target.parentNode; let branchName = parentNode.querySelector("span.branch-name").innerText.trim() branchName = branchName.substring(1, branchName.length - 1).replaceAll("/", "%2F"); // 跳转链接 let baseUrl = 'http://ios.meitu-int.com/ipa/' + project + '/' let targetUrl = baseUrl + branchName // 跳转 window.location.href = targetUrl }); } if (!build_magichouse) { addGranaryMagichouse(); $('.build_magichouse').unbind("click").click(function () { let buildIdB = document.getElementById("input_last_build").value.trim() if (buildIdB === "") { alert("请输入上个版本提交的build id"); return false; } let parentNode = event.target.parentNode.children[0].innerText.trim(); let buildIdA = parentNode.match(/#Build\s+(\d+)/)[1].trim(); if (confirm("确定构建魔法屋任务吗?点击确定构建")) { buildMagichouseTask(0, buildIdA, buildIdB) } }); } }, refreshTime); } // granary页面,添加输入框和跳转按钮 function addFindButtonGranary() { let span = $('<span class="search"></span>') let input_find_build = $('<input type="text" class="text" id="input_find_build" placeholder=" 输入Build id或分支名">'); let btn_find_build = $('<button type="submit" class="btn_find_build" id="btn_find_build">🔍</button>'); span.append(input_find_build); span.append(btn_find_build); $("#myTab").append(span); } // granary页面,在分支名后方添加跳转至最新按钮 function addToNewGranary() { let to_new_build = $('<button type="submit" class="to_new_build" id="to_new_build">跳转至最新➔</button>'); $("div#list-home span.branch-name").after(to_new_build); } // granary页面,在其他后方添加魔法屋跑图任务按钮 function addGranaryMagichouse() { let input_last_build = $('<input type="text" class="text" id="input_last_build" placeholder=" 输入上个版本企业包">'); let build_magichouse = $('<button type="button" class="build_magichouse" id="build_magichouse">魔法屋跑图</button>'); $(".to_new_build").after(build_magichouse); $(".to_new_build").after(input_last_build); } // 当是jira域名时,才触发后续的操作,如果不是则不触发 // Jira // Jira // Jira if (location.href.indexOf('jira.meitu.com') > 0) { clearInterval(refreshTime); var counter = 0; // 定时器循环操作:页面元素添加等 setInterval(function () { var search_span = document.getElementById('search_span_create'); // 因为切至iOS按钮和切至Android按钮一般都会成对出现,所以这里只获取iOS按钮,用于判断按钮是否已存在 var change_side_button = document.getElementById('change_side_ios'); var create_input = document.getElementById('customfield_10303'); var step_text = document.getElementById('customfield_10203'); var create_issue_dialog = document.getElementById('create-issue-dialog'); var close_bug_dialog = document.getElementById('workflow-transition-21-dialog'); var reopen_bug_dialog = document.getElementById('workflow-transition-31-dialog'); var comment_bug_toolbar = document.getElementById('wiki-edit-wikiEdit0'); // 如果不存在bug跳转按钮则添加一个,需要判断url是bug页面而不是bug列表页,否则会报错 if (!search_span) { addButtonJira(); // 存储bug平台 GM_setValue('platform', $('#customfield_10301-val').text().trim()) } // 在创建问题dialog添加获取分支按钮组 if (create_issue_dialog) { // 加个计时避免按钮显示不出来 // Bug模版按钮 if (change_side_button == undefined || change_side_button.length == 0) { addChangeSideButton(); } // 获取分支按钮 if (create_input !== undefined || create_input.length !== 0) { var branch_span = $('<label for="customfield_10304"></label>'); $("#customfield_10303").after(branch_span) add_get_branch_btn($(branch_span)); } // 重置步骤按钮 if (step_text !== undefined || step_text.length !== 0) { var step_span = $('<label for="customfield_10204"></label>'); $("#customfield_10203").after(step_span); var step_btn = $('<input type="button" class="aui-button" value="重置步骤" id="reset_step">'); step_span.append(step_btn) } // 隐藏starii项目不必要的UI hideUI(); // 自动填充build号 fillBuildIdAuto(); } // 自动往指定的input组件中填build号 function fillBuildIdAuto() { // 获取输入框内容 var inputContent = $("#customfield_10303").val(); // 使用正则表达式检查当前文本内容是否是纯数字 var isCurrentNumber = /^\d+$/.test(inputContent); if (isCurrentNumber) { counter++; console.log(counter) // 如果当前文本内容是纯数字 if (counter % 6 === 0) { // 往输入框内填写 set_branch('get_branch_btn', 'create-issue-dialog') counter = 0; } } } // 在关闭问题dialog 或 重新打开dialog 添加获取分支按钮组 if (close_bug_dialog || reopen_bug_dialog) { var pre_text_button = $('<input class="aui-button" id="close-text" type="button" value="上次填写"></input>'); var pre_text_btn = document.getElementById('close-text'); var branch_span_close = $('<span id="close_text">输入id:</span>'); var input_text_close = $('<input type="text" class="text medium-field" id="build_id_close">'); setTimeout(function () { if (!pre_text_btn) { $(".jira-dialog-content .form-footer").append(branch_span_close).append(input_text_close);// buildid输入框 add_get_branch_btn($(".jira-dialog-content").find(".form-footer"));//获取分支按钮 $(".jira-dialog-content .form-footer").append(pre_text_button);//上次填写按钮 } }, 500); } // 在备注窗口添加获取分支按钮组 if (comment_bug_toolbar) { var pre_text_span = $('<span id="pre_text">输入id:</span>'); var pre_text_span_element = document.getElementById('pre_text'); var input_text = $('<input type="text" class="text medium-field" id="input_text">'); setTimeout(function () { if (!pre_text_span_element && !close_bug_dialog) { var branch_span = $('<span></span>'); branch_span.append(pre_text_span).append(input_text); add_get_branch_btn($(branch_span)); $(".security-level .current-level").after(branch_span); } }, 500); } // 获取分支按钮点击事件 $('#get_branch_btn, #last_branch_btn').unbind("click").click(function (event) { setTimeout(function () { // 当前节点的父节点 var parentNode = event.target.parentNode; // 如果父节点没有Id属性,则往上遍历 while (parentNode != null) { if (parentNode.hasAttribute("id")) { // 找到第一个有id属性的父节点 var parentWithId = parentNode; break; } parentNode = parentNode.parentNode; } // 第一个拥有Id的父节点的id let parentId = parentWithId.getAttribute("id"); // 获取当前节点id var selfId = event.target.id // 往网页中填入分支名 set_branch(selfId, parentId); }, 500); }); // 重置步骤按钮点击事件 $('#reset_step').unbind('click').click(function (event) { let default_text = '[预置条件]\n\n[步骤]\n\n[结果]\n\n[期望]\n\n[备注机型]\n\n[BUG出现频次]\n\n\n' document.getElementById('customfield_10203').value = default_text; }) function hideUI() { // 获取具有id为project-options的div元素 var projectOptionsDiv = document.getElementById('project-options'); if (projectOptionsDiv) { // 获取data-suggestions属性的值 var dataSuggestionsValue = projectOptionsDiv.getAttribute('data-suggestions'); var jsonObject = JSON.parse(dataSuggestionsValue); // 项目名 var project = jsonObject[0]['items'][0].label.trim(); var project_name = project.replace(/\s*\([^)]*\)\s*/, '').trim(); } else { var project_name = $('#project-name-val').text().trim(); // 项目名 } if (project_name === "美图秀秀Starii") { hideParentNodeById("customfield_10422"); hideParentNodeById("customfield_10202"); hideParentNodeById("customfield_10305"); hideParentNodeById("customfield_11100"); hideParentNodeById("customfield_11101"); hideParentNodeById("customfield_11102"); hideParentNodeById("customfield_10304"); hideParentNodeById("fixVersions"); hideParentNodeById("reporter"); hideParentNodeById("customfield_13601"); } } // 隐藏指定ID节点的父节点 function hideParentNodeById(childNodeId) { // 隐藏bug优先级 var element = document.getElementById(childNodeId); // 检查是否找到了元素 if (element) { // 获取父节点并将其样式的display属性设置为"none" element.parentNode.style.display = "none"; } } // 往网页中填入分支名称 async function set_branch(selfId, parentId) { if (selfId === "get_branch_btn") { switch (parentId) { // 创建Bug窗口填写分支 case "create-issue-dialog": // bug平台节点和buildId节点 var $platform = $('input:radio[name="customfield_10301"]:checked'); var $buildId = $('#customfield_10303'); // 通过buildId节点(input)定位到bug平台节点(label) var id = $platform.attr("id") var label = document.querySelector("label[for='" + id + "']"); // bug平台和buildId try { var platform = label.textContent; } catch (error) { GM_setValue('branch_value', "#请先选择Bug平台"); fillInBranch('#customfield_10303'); return; } var buildId = $buildId.val(); if (buildId === undefined || buildId === '' || buildId === null) { GM_setValue('branch_value', "#请先填写Build号"); fillInBranch('#customfield_10303'); return; } console.log(platform) // 获取分支名 await get_branch(platform, buildId); // 填入分支 await fillInBranch('#customfield_10303'); counter = 0; break; // 关闭||重新打开窗口填写分支 case "issue-workflow-transition": var $platform = $('#customfield_10301-val'); var $buildId = $('#build_id_close'); // bug平台和buildId var platform = $platform.text().trim(); var buildId = $buildId.val(); // 获取分支名 await get_branch(platform, buildId); // 填入分支 await fillInBranchTextarea('div#comment-wiki-edit textarea#comment'); // 聚焦到输入框 sleep(500).then(() => { $('#comment-wiki-edit textarea').focus(); }) break; // 备注 case "issue-comment-add": var $platform = $('#customfield_10301-val'); var $buildId = $('#input_text'); // bug平台和buildId var platform = $platform.text().trim(); var buildId = $buildId.val(); // 获取分支名 await get_branch(platform, buildId); // 填入分支 await fillInBranchTextarea('div#comment-wiki-edit textarea#comment'); // 聚焦到输入框 sleep(500).then(() => { $('#comment-wiki-edit textarea').focus(); }) break; default: } } else if (selfId === "last_branch_btn") { switch (parentId) { case "create-issue-dialog": // 填入分支 await fillInBranch('#customfield_10303'); break; case "issue-workflow-transition": // 填入分支 await fillInBranchTextarea('div#comment-wiki-edit textarea#comment'); break; // 备注 case "issue-comment-add": await fillInBranchTextarea('div#comment-wiki-edit textarea#comment'); break; default: } } } // 跳转到创建分支 $('#search_span_create').unbind("click").click(function () { var build_id = getBuildId("customfield_10303-val"); console.log(build_id) // 存储bug平台 GM_setValue('platform', $('#customfield_10301-val').text().trim()) sleep(500).then(() => { var targetUrl = getBaseUrl() + build_id; window.open(targetUrl); }) }); // 跳转到解决分支 $('#search_span_solved').unbind("click").click(function () { var build_id = getBuildId("customfield_10304-val"); // 存储bug平台 GM_setValue('platform', $('#customfield_10301-val').text().trim()) sleep(500).then(() => { var targetUrl = getBaseUrl() + build_id; window.open(targetUrl); }) }); // 切换到iOS bug模版 $('.ios').unbind("click").click(function () { changeBugPlatform('iOS') }); // 切换到Androidbug模版 $('.android').unbind("click").click(function () { changeBugPlatform('Android') }); // 切换到Webbug模版 $('.web').unbind("click").click(function () { changeBugPlatform('Web') }); // 点击关闭问题按钮,记录下填写的内容 var text_area = $('.jira-dialog-content').find('#comment') $('#issue-workflow-transition-submit').unbind("click").click(function () { if ($('#issue-workflow-transition-submit').val().trim() == "关闭问题") { // 存储bug平台 GM_setValue('closeText', text_area.val().trim()) } }) // 点击上次填写按钮,填充上次填写的内容 $('#close-text').unbind("click").click(function () { text_area.val(GM_getValue('closeText')) text_area.focus() }) // TODO:点击「创建」按钮时记录下所有的bug信息 const $createBtn = $('#create-issue-submit'); const $summary = $('#summary'); const $business = $('input:radio[name="customfield_12903"]:checked'); const $platform = $('input:radio[name="customfield_10301"]:checked'); const $path0 = $("#selectCFLevel0 option:selected"); const $path1 = $("#selectCFLevel1 option:selected"); const $assignee = $("#assignee-field"); const $severity = $("#customfield_10406 option:selected"); const $version = $("#versions-multi-select .value-text"); const $find = $("#customfield_10202 option:selected"); const $frequency = $("#customfield_10204 option:selected"); const $branch = $("#customfield_10303"); const $step = $("#customfield_10203"); const $tips = $("#labels-multi-select .representation .value-text"); $createBtn.unbind('click').click(function () { const bugDict = { 'summary': $summary.val(), 'business': $business.attr("id"), 'platform': $platform.attr("id"), 'path0': $path0.text(), 'path1': $path1.text(), 'assignee': $assignee.val(), 'severity': $severity.text(), 'version': $version.text(), 'find': $find.attr("value"), 'frequency': $frequency.attr("value"), 'branch': $branch.val(), 'step': $step.val(), 'tips': $tips.text() }; GM_setValue('bugDict', bugDict); }); // 点击再提一个 // 已知问题:1. 路径2无法填写 $('#once-again').unbind('click').click(function () { const bugDict = GM_getValue('bugDict'); function setValue(selector, value) { $(selector).val(value); } function setChecked(selector, value) { $(selector).attr("checked", value); } function setSelected(selector, value) { $(selector).find(`option[value='${value}']`).attr("selected", true); } console.log(bugDict); setValue("#summary", bugDict.summary); setChecked(`input[id=${bugDict.business}]`, true); setChecked(`input[id=${bugDict.platform}]`, true); setValue("#selectCFLevel0", bugDict.path0); setValue("#selectCFLevel1", bugDict.path1); setValue("#assignee-field", bugDict.assignee); // $('#assignee').append($('<option>', { // value: '[email protected]', // text: '丘文坚', // title: "undefined", // selected: "selected", // style: "background-image: url(\"https://jira.meitu.com/secure/useravatar?size=xsmall&ownerId=qwj%40meitu.com&avatarId=13404\");" // })); setSelected("#customfield_10406", bugDict.severity); setValue("#versions-textarea", bugDict.version); setSelected("#customfield_10202", bugDict.find); //setSelected("#customfield_10204", bugDict.frequency); setValue("#customfield_10303", bugDict.branch); setValue("#customfield_10203", bugDict.step); //setValue("#labels-textarea",bugDict.tips) }); // 点击创建按钮 $('#create-issue-submit').unbind('click').click(function () { var $platform = $('input:radio[name="customfield_10301"]:checked'); // 通过buildId节点(input)定位到bug平台节点(label) var id = $platform.attr("id") var label = document.querySelector("label[for='" + id + "']"); try { var platform = label.textContent; } catch (error) { platform = ""; console.log("没选平台,不提示"); return; } var version = $("#versions-multi-select .value-text").text(); platform = platform.toLowerCase(); version = version.toLowerCase(); if (platform && version && !version.includes(platform) && platform != 'harmony') { console.log("存在不一致情况"); showtipsMessage("平台和影响版本不一致,请检查下!", "#F66666"); }; }); // logWork添加打卡选项下拉框 function addDropdownBeforeLogButton() { // 查找"记录"按钮 const logButton = document.getElementById('log-work-submit'); // 检查下拉框是否已存在 const existingDropdown = document.getElementById('work-hours-dropdown'); // 只有当按钮存在且下拉框不存在时才创建 if (logButton && !existingDropdown) { // 创建下拉框 const dropdown = document.createElement('select'); dropdown.id = 'work-hours-dropdown'; dropdown.className = 'aui-button'; dropdown.style.marginRight = '10px'; // 添加选项 - 替换为指定的类别 const categories = ['闪传', 'AI垂类', '非AI垂类', '美化', '美容', '拼图', '相机', '视频美化', '视频美容', '证件照', '素材中心', '订阅', '商业化', '其他']; // 添加一个默认空选项 const defaultOption = document.createElement('option'); defaultOption.value = ''; defaultOption.text = '选择类别'; defaultOption.disabled = true; defaultOption.selected = true; dropdown.appendChild(defaultOption); // 添加"全部"选项 const allOption = document.createElement('option'); allOption.value = 'all'; allOption.text = '全部'; dropdown.appendChild(allOption); // 添加类别选项 categories.forEach(category => { const option = document.createElement('option'); option.value = category; option.text = category; dropdown.appendChild(option); }); // 将下拉框插入到按钮前面 logButton.parentNode.insertBefore(dropdown, logButton); // 添加change事件监听器 dropdown.addEventListener('change', function () { // 获取选中的值 const selectedValue = this.value; // 更精确地定位当前工作记录对话框中的comment文本区域 // 首先找到当前对话框,然后在其中查找textarea const dialogContainer = logButton.closest('.aui-dialog2, .jira-dialog'); if (dialogContainer) { const commentTextarea = dialogContainer.querySelector('textarea.textarea.long-field.wiki-textfield#comment'); if (commentTextarea) { // 获取当前光标位置 const startPos = commentTextarea.selectionStart; const endPos = commentTextarea.selectionEnd; // 获取当前文本内容 const currentText = commentTextarea.value; // 构建要插入的文本 let insertedText; // 如果选择"全部",则插入所有类别 if (selectedValue === 'all') { insertedText = ""; categories.forEach(category => { insertedText += "【" + category + "】" + ": h\n"; }); // 移除最后一个换行符 insertedText = insertedText.trim(); } else { insertedText = "【" + selectedValue + "】" + ": h"; } // 在光标位置插入选中的值 const newText = currentText.substring(0, startPos) + insertedText + currentText.substring(endPos); // 更新文本区域的值 commentTextarea.value = newText; // 设置光标位置在插入文本之后,-1可以定位到“h”之前 const newCursorPos = startPos + insertedText.length - 1; commentTextarea.setSelectionRange(newCursorPos, newCursorPos); // 聚焦文本区域 commentTextarea.focus(); // 重置下拉框为默认选项 this.selectedIndex = 0; } } }); } } // ... existing code ... addDropdownBeforeLogButton(); }, refreshTime); } // 在输入框中填入分支 // 这里必须加上700ms的延时,否则get_branch尚未填写完成时就会调用该方法,导致填写之前存储的内容 async function fillInBranch(inputSelector) { // await sleep(700); var inputElement = $(inputSelector); inputElement.val("").val(GM_getValue('branch_value')); } // 在文本区域中填入分支 async function fillInBranchTextarea(textareaSelector) { // await sleep(700); var textareaElement = $(textareaSelector); textareaElement.val("").val(GM_getValue('branch_value')); } // 在指定节点后添加获取分支按钮 function add_get_branch_btn(targetElement) { var branch_btn = $('<input type="button" class="aui-button" value="获取分支" id="get_branch_btn">'); var last_branch_btn = $('<input type="button" class="aui-button" value="上次分支" id="last_branch_btn">'); targetElement.append(branch_btn).append(last_branch_btn) } // 根据build号,请求接口获取分支 async function get_branch(platform, build_id) { return new Promise((resolve, reject) => { //console.log(platform+ "接收到的" + build_id) var url = ''; var branch = ''; var result = ''; // 获取具有id为project-options的div元素 var projectOptionsDiv = document.getElementById('project-options'); if (projectOptionsDiv) { // 获取data-suggestions属性的值 var dataSuggestionsValue = projectOptionsDiv.getAttribute('data-suggestions'); var jsonObject = JSON.parse(dataSuggestionsValue); // 项目名 var project = jsonObject[0]['items'][0].label.trim(); var project_name = project.replace(/\s*\([^)]*\)\s*/, '').trim(); } else { var project_name = $('#project-name-val').text().trim(); // 项目名 } if (platform === 'iOS') { url = 'https://omnibus.meitu-int.com/api/apps/' + appuid_ios[project_name] + '/builds/' + build_id; } else if (platform === 'Android') { url = 'https://omnibus.meitu-int.com/api/apps/' + appuid_android[project_name] + '/builds/' + build_id; } else { url = 'https://omnibus.meitu-int.com/api/apps/' + appuid_android[project_name] + '/builds/' + build_id; } GM_xmlhttpRequest({ url: url, method: 'GET', onload: function (res) { if (res.status === 200) { var r = ''; if (platform === 'iOS') { r = 'refs/heads/(.*?)B'; } else if (platform === 'Android') { r = 'refs/heads/(.*?)B'; } else { r = 'refs/heads/(.*?)B'; } branch = res.responseText.match(r)[1]; } else { branch = '' } // branch的值存在$符时,设置为空 // branch的值为空时,设置返回结果为提示语 branch = branch.indexOf('$') != -1 ? '' : branch; result = branch == '' ? "未找到该包的分支," + build_id : branch + '#' + build_id; if (result.indexOf("未找到") !== -1) { GM_setValue('branch_value', result); console.log(GM_getValue('branch_value')) } else { if (result.indexOf(build_id) !== -1) { GM_setValue('branch_value', result); console.log(GM_getValue('branch_value')) } } // 处理完成后 resolve resolve(GM_getValue('branch_value')); }, onerror: function (err) { result = '接口请求失败,建议重新关闭开启脚本再试试'; GM_setValue('branch_value', result); reject(err); } }); }); } //在jira页面添加跳转到分支按钮 function addButtonJira() { // 创建分支按钮添加 var span_create = $('<span class="search_span" id="search_span_create" style=""></span>') // 添加图片 var search_image = "" var img = document.createElement('img'); img.className = "myimg" img.src = search_image; span_create.append(img); $("#customfield_10303-val").after(span_create); // 解决分支按钮添加 var span_solved = $('<span class="search_span" id="search_span_solved"></span>') var img2 = document.createElement('img'); img2.className = "myimg" img2.src = search_image; span_solved.append(img2); $("#customfield_10304-val").after(span_solved); } // 正则表达式获取build号 function getBuildId(elementId) { var create_build_content = document.getElementById(elementId).textContent.trim(); var reg = /\d{4,}/g; var build_id_array = create_build_content.match(reg); var build_id = ''; if (build_id_array == null || build_id_array.length == 0) { console.log("未识别到 build 号"); } else { build_id = build_id_array[0]; var reg_num = /\d{4,}/g; build_id = build_id.match(reg_num)[0].replace('#', ''); } return build_id } // 根据不同的项目,Bug平台拼接url function getBaseUrl() { var baseUrl = ''; var project_name = $('#project-name-val').text().trim(); // 项目名 var platform = $('#customfield_10301-val').text().trim(); //平台 // TODO:兼容不同的项目 switch (platform) { case 'iOS': baseUrl = 'https://omnibus.meitu-int.com/apps/' + project_ios_omnibus[project_name] + ':ios/build/number/' break; case 'Android': baseUrl = 'https://omnibus.meitu-int.com/apps/' + project_android_omnibus[project_name] + ':android/build/number/' break; default: baseUrl = 'https://omnibus.meitu-int.com/apps/' + project_android_omnibus[project_name] + ':android/build/number/' } return baseUrl; } // 在创建Bug页面添加Bug模版(iOS、Android、Web)按钮 function addChangeSideButton() { // 创建分支按钮添加 var btn_ios = $('<button class="ios aui-button" id="change_side_ios" type="button" style="">iOS</button>') var btn_android = $('<button class="android aui-button" id="change_side_android" type="button" style="">Android</button>') var btn_web = $('<button class="web aui-button" id="change_side_web" type="button" style="">Web</button>') var btn_once_again = $('<button class="once-again aui-button" id="once-again" type="button" style="">再提一个</button>') $(".jira-dialog-content").find(".form-footer").append(btn_ios).append(btn_android).append(btn_web).append(btn_once_again) } // 点击按钮更换Bug模版,如:iOS、Android、Web function changeBugPlatform(platform) { var project_name = $('#project-name-val').text().trim(); // 项目名 switch (project_name) { case "美图秀秀": if (platform == "iOS") { // document.getElementById('customfield_12903-1').checked = true document.getElementById('customfield_10301-2').checked = true } else if (platform == "Android") { // document.getElementById('customfield_12903-1').checked = true document.getElementById('customfield_10301-1').checked = true } else if (platform == "Web") { // document.getElementById('customfield_12903-1').checked = true document.getElementById('customfield_10301-3').checked = true } break; case "美图秀秀Starii": if (platform == "iOS") { document.getElementById('customfield_10301-2').checked = true } else if (platform == "Android") { document.getElementById('customfield_10301-1').checked = true } else if (platform == "Web") { document.getElementById('customfield_10301-3').checked = true } break; default: if (platform == "iOS") { document.getElementById('customfield_10301-2').checked = true } else if (platform == "Android") { document.getElementById('customfield_10301-1').checked = true } else if (platform == "Web") { document.getElementById('customfield_10301-3').checked = true } } // 获取当前的月份和日期 const date = new Date() const today = date.getDate() const curmonth = date.getMonth() + 1 // 获取大于且最接近当前日期的版本 let minNum = 99 let similarDate = "" // 获取 <optgroup> 元素 let optgroup = $('.aui-field-versionspicker').find('.multi-select-select').find('[label="未发布版本"]')[0] // 获取 <option> 元素集合 let options = optgroup.getElementsByTagName("option"); for (let i = 0; i < options.length; i++) { let option = options[i]; let text = option.textContent.trim(); if (text.toLowerCase().indexOf(platform.toLowerCase()) < 0) { continue; } else { // console.log(text);//打印获取到的版本号 let startNum = text.lastIndexOf("(") !== -1 ? text.lastIndexOf("(") : text.lastIndexOf("("); let endNum = text.lastIndexOf(")") !== -1 ? text.lastIndexOf(")") : text.lastIndexOf(")"); // theDateStr是形似「1109」的日期形式,下面拆分出月份和日期; theMonth形如「02」,theDate形如「18」 // theDateStr还有可能是「11.9」的形式,需要对有无小数点进行判断 // 2024-11-05-调整,theDateStr改为[11/5]的形式 let theDateStr = text.slice(startNum + 1, endNum) console.log(theDateStr) let theMonth = "" let theDate = "" if (theDateStr.includes('.')) { // 使用 split 方法分割小数点前后的数字 let parts = theDateStr.split("."); // 获取小数点前面的数字 theMonth = parts[0]; // 获取小数点后面的数字 theDate = parts[1]; } else if (theDateStr.includes('/')) { let parts = theDateStr.split("/"); theMonth = parts[0]; theDate = parts[1]; } else { theMonth = parseInt(theDateStr.slice(4, 6))// 2022-12-11调整,theDateStr变为20221109的形式,所以调整slice的区间;原本为(0,2)(2) theDate = parseInt(theDateStr.slice(6)) } if (theMonth < curmonth && theMonth !== "1") { continue; } else if (theMonth == curmonth) { if (theDate - today >= 0 && theDate - today < minNum) { similarDate = theDateStr minNum = theDate - today } } else if (theMonth == curmonth + 1 || theMonth == curmonth - 11) { let monthDuration = getDuration() let daysToEnd = monthDuration - today if ((Number(theDate) + Number(daysToEnd)) < minNum) { similarDate = theDateStr minNum = Number(theDate) + Number(daysToEnd) } } } } // console.log(similarDate) // 删掉「影响版本」文本框中的内容 let div = document.getElementsByClassName('representation')[0] let emarr = div.getElementsByTagName('em') for (let i = 0; i < emarr.length; i++) { // 这里全部都点击的emmarr[0],是因为第一个节点被删除掉之后,后面的素材会顶上来成为新的第0位节点,加个200ms延时,避免点击不到 sleep(200).then(() => { emarr[0].click() }) } // 在「影响版本」文本框填入内容 for (let i = 0; i < options.length; i++) { let option = options[i]; let text = option.textContent.trim(); // 如果a节点当中,存在目标日期字段,且平台与点击的一致,就把a节点的text填入到文本框中 if (text.indexOf(similarDate) > 0 && text.toLowerCase().indexOf(platform.toLowerCase()) > 0) { $("#versions-textarea").val(text) // 获取控件焦点 $("#versions-textarea").focus() // 主动失去当前控件的焦点 $("#versions-textarea").blur() } } } // 获取当前月份有多少天 function getDuration() { let dt = new Date() var month = dt.getMonth() dt.setMonth(dt.getMonth() + 1) dt.setDate(0) return dt.getDate() } // sleep方法,用于延迟一些操作 function sleep(time) { return new Promise((resolve) => setTimeout(resolve, time)); } // 当是CF域名时,才触发后续的操作,如果不是则不触发 // CF // CF // CF if (location.href.indexOf('cf.meitu.com') > 0) { // 获取ul元素 var menuBars = document.getElementsByClassName("ajs-menu-bar"); var menuBar = menuBars[0]; // 创建两个li元素 var li1 = document.createElement("li"); var li2 = document.createElement("li"); li1.className = "ajs-button normal" li2.className = "ajs-button normal" // 创建两个button元素 var button1 = document.createElement("button"); var button2 = document.createElement("button"); // 设置按钮文本 button1.textContent = "复制Android需求"; button2.textContent = "复制iOS需求"; // 设置按钮ID button1.id = "Android" button2.id = "iOS" // 设置按钮Class button1.className = "aui-button aui-button-subtle edit" button2.className = "aui-button aui-button-subtle edit" // 将button元素添加到li元素中 li1.appendChild(button1); li2.appendChild(button2); // 将li元素添加到ul元素中 menuBar.appendChild(li1); menuBar.appendChild(li2); // 将li元素放在menuBar的最前 menuBar.insertBefore(li2, menuBar.firstChild); menuBar.insertBefore(li1, menuBar.firstChild); getRequirementArray("Android"); getRequirementArray("iOS"); } /** * 为平台特定按钮添加点击事件监听器,根据表格中的复选框状态收集需求名称并复制到剪贴板。 * * @function getRequirementArray * @param {string} platform - 平台标识符("Android"或"iOS"),用于查找按钮元素 * @description 当指定平台按钮被点击时,此函数将: * 1. 扫描需求表格中的已勾选项目 * 2. 将需求名称收集到特定平台的数组中 * 3. 根据特定规则处理"中间架构"需求: * - 排除包含"底层先行"的项目 * - 根据平台特定文本将项目路由到适当的平台数组 * 4. 将收集的需求复制到剪贴板 * @requires getArrayByCheckedLi - 根据复选框状态将需求添加到数组的辅助函数 * @requires copyArrayToClipboard - 将数组复制到剪贴板的辅助函数 */ function getRequirementArray(platform) { // 获取相应的按钮 var platformButton = document.getElementById(platform); // 添加点击事件处理程序 platformButton.addEventListener("click", function (event) { // 创建一个空数组,用于存储每个需求名称 var androidArray = []; var iosArray = []; // 记录每个需求所处的列数,用于获取打勾状态 var androidTd = findColumnIndex('Android') + 1; var iosTd = findColumnIndex('iOS') + 1; var cppTd = findColumnIndex('中间架构') + 1; // 获取具有aria-live属性为"polite"的tbody元素 var politeTbodies = document.querySelectorAll('tbody[aria-live="polite"]'); politeTbodies.forEach(function (tbody) { // 获取tbody下的所有tr元素 var rows = tbody.querySelectorAll('tr'); // 遍历tr rows.forEach(function (row) { // 从tr中找到a元素,直接获取需求名称 var anchorElement = row.querySelector('a'); const secondTdText = anchorElement ? anchorElement.textContent.trim() : undefined; // Android有打勾的项目放入Android数组 getArrayByCheckedLi(row, androidTd, secondTdText, androidArray) // iOS有打勾的项目放入iOS数组 getArrayByCheckedLi(row, iosTd, secondTdText, iosArray) // 中间架构有打勾的项目放入双端数组 let blackWord = '底层先行'; // 需求名称中不含“底层先行”的可放入 if (secondTdText && secondTdText.indexOf(blackWord) === -1) { let platformAndroid = "Android"; let platformiOS = "iOS"; // 不含Android含有iOS的,放入iOS数组 if (secondTdText.indexOf(platformAndroid) === -1 && secondTdText.indexOf(platformiOS) !== -1) { getArrayByCheckedLi(row, cppTd, secondTdText, iosArray); // 不含iOS,含有Android的放入Android数组 } else if (secondTdText.indexOf(platformAndroid) !== -1 && secondTdText.indexOf(platformiOS) === -1) { getArrayByCheckedLi(row, cppTd, secondTdText, androidArray); } else { // 其余的两个数组都放入 getArrayByCheckedLi(row, cppTd, secondTdText, androidArray); getArrayByCheckedLi(row, cppTd, secondTdText, iosArray); } } }); }); var buttonPlatform = event.target.id; if (buttonPlatform === "Android") { copyArrayToClipboard(androidArray); } else if (buttonPlatform === "iOS") { copyArrayToClipboard(iosArray); } }); } /** * 查找表头中包含指定字符串的单元格位置 * @param {string} columnName - 要查找的列名 * @return {number} - 列的索引位置(从0开始),如果未找到则返回-1 */ function findColumnIndex(columnName) { // 获取所有表头单元格 const headerCells = document.querySelectorAll('th.confluenceTh'); // 遍历查找匹配的列 for (let i = 0; i < headerCells.length; i++) { // 获取表头单元格中的内部div const headerInner = headerCells[i].querySelector('.tablesorter-header-inner'); if (headerInner && headerInner.textContent.trim() === columnName) { return i; // 返回索引位置(从0开始) } } // 未找到匹配列 console.log(`未找到包含"${columnName}"的列`); return -1; } // 获取tr元素中的第N个td元素,并检查其中的li元素是否为选中状态,选中的话则添加需求名字段到数组 function getArrayByCheckedLi(row, tdNum, requirementName, array) { var tdElement = row.querySelector('td:nth-child(' + tdNum + ')'); var liElement = tdElement.querySelector('li.checked'); if (liElement) { addToUniqueArray(array, requirementName) } } // 往数组内添加不重复的元素 function addToUniqueArray(arr, element) { if (arr.indexOf(element) === -1) { arr.push(element); } } // 将数组转化为字符串进行拷贝 function copyArrayToClipboard(arr) { // 将数组转换为字符串 var arrayString = arr.join('\n'); // 使用逗号和空格分隔数组元素 // 创建一个新的textarea元素 var textarea = document.createElement('textarea'); // 设置textarea的值为数组转换后的字符串 textarea.value = arrayString; // 将textarea元素添加到DOM中 document.body.appendChild(textarea); // 选中textarea中的文本 textarea.select(); try { // 尝试执行复制操作 document.execCommand('copy'); console.log('数组已复制到剪贴板'); } catch (err) { console.error('复制到剪贴板失败: ', err); } finally { // 移除textarea元素 document.body.removeChild(textarea); // js提示复制成功 showCopySuccessMessage() } } /** * 复制成功提示 */ function showCopySuccessMessage() { // 创建一个提示框 var messageBox = document.createElement('div'); messageBox.textContent = '复制成功!'; messageBox.style.position = 'fixed'; messageBox.style.top = '50%'; messageBox.style.left = '50%'; messageBox.style.transform = 'translate(-50%, -50%)'; messageBox.style.padding = '10px'; messageBox.style.background = '#4CAF50'; messageBox.style.color = 'white'; messageBox.style.borderRadius = '5px'; messageBox.style.zIndex = '9999'; // 将提示框添加到DOM中 document.body.appendChild(messageBox); // 2秒后移除提示框 setTimeout(function () { document.body.removeChild(messageBox); }, 2000); } function showtipsMessage(textContent, background) { // 创建一个提示框 var messageBox = document.createElement('div'); messageBox.textContent = textContent; messageBox.style.position = 'fixed'; messageBox.style.top = '50%'; messageBox.style.left = '50%'; messageBox.style.transform = 'translate(-50%, -50%)'; messageBox.style.padding = '10px'; messageBox.style.background = background; messageBox.style.color = 'white'; messageBox.style.borderRadius = '5px'; messageBox.style.zIndex = '9999'; // 将提示框添加到DOM中 document.body.appendChild(messageBox); // 3秒后移除提示框 setTimeout(function () { document.body.removeChild(messageBox); }, 3000); } // Your code here... })();