阿里云云效增强

阿里云效平台功能增强辅助脚本

目前为 2021-09-03 提交的版本。查看 最新版本

// ==UserScript==
// @name         阿里云云效增强
// @namespace    http://bmqy.net/
// @version      1.0.1
// @description  阿里云效平台功能增强辅助脚本
// @author       bmqy
// @match        https://packages.aliyun.com/repos/*
// @icon         
// @grant        GM_getValue
// @grant        GM_setValue
// @grant        GM_deleteValue
// ==/UserScript==

(function () {
	'use strict';

	// Your code here...
	const app = {
		init: function () {
			this.delStorage();
			this.checkboxOnClick();
			this.pkgUploadFormOnAddEvent();
		},

		// 脚本存储数据key
		sotrageKey: 'BmqyAliyunPackage',

		/**
		 * 获取脚本存储数据值
		 * @returns {Object}
		 */
		getStorage: function () {
			return GM_getValue(this.sotrageKey);
		},

		/**
		 * 设置脚本存储
		 * @param {Object} value 存储数据值
		 */
		setStorage: function (value) {
			GM_setValue(this.sotrageKey, value);
		},

		/**
		 * 清除脚本存储
		 * @param {Object} value 存储数据值
		 */
		delStorage: function () {
			GM_deleteValue(this.sotrageKey);
		},

		/**
		 * 更新制品上传表单
		 */
		pkgUploadFormUpdate: function () {
			let _this = this;
			let $groupId = document.querySelector('#groupId');
			let $artifactId = document.querySelector('#artifactId');
			let $version = document.querySelector('#version');
			if (_this.getStorage()) {
				$groupId.value = _this.getStorage().group;
				_this.reactInputEmit($groupId);
				$artifactId.value = _this.getStorage().artifact;
				_this.reactInputEmit($artifactId);
				$version.value = _this.getStorage().version;
				_this.reactInputEmit($version);
			}
		},

		// js如何在外部改变react受控组件的状态量?参考:https://github.com/ILovePing/ILovePing.github.io/issues/22
		reactInputEmit: function (element) {
			let event = new Event('input', { bubbles: true });
			// hack React15
			event.simulated = true;
			// hack React16 内部定义了descriptor拦截value,此处重置状态
			let tracker = element._valueTracker;
			if (tracker) {
				tracker.setValue(element);
			}
			element.dispatchEvent(event);
		},

		// 监听当前选中的制品
		checkboxOnClick: function () {
			let _this = this;
			document.body.addEventListener('click', (e) => {
				let $element = e.target;
				if ($element.classList.contains('next-checkbox-input')) {
					if ($element.checked) {
						let $parent = _this.getParent(
							$element,
							'next-table-row'
						);
						_this.currentPackageUpdate($parent);
					} else {
						// 取消当前选中则检测列表中第一个选中的行,如果有更新存储值
						setTimeout(() => {
							let $checkboxList = document.querySelectorAll(
								'.next-table-row.selected'
							);
							if ($checkboxList.length == 0) {
								_this.delStorage();
								return false;
							}

							_this.currentPackageUpdate($checkboxList[0]);
						}, 200);
					}
				}
			});
		},

		// 更新本地存储值
		currentPackageUpdate: function (element) {
			let _this = this;
			if (!element) return false;

			let $tr = element.querySelectorAll('.next-table-cell-wrapper');
			let $GA = $tr[1],
				$ver = $tr[2];
			let GAStr = $GA.innerText;
			let data = {
				group: GAStr.split(':')[0],
				artifact: GAStr.split(':')[1],
				version: _this.versionFormat($ver.innerText),
			};
			_this.setStorage(data);
		},

        versionFormat: function (version) {
            if(!version) return version;

            let arr = version.split('.');
            arr.forEach((e, i) => {
                if(i >= (arr.length-1)){
                    arr[arr.length-1] = ++e;
                }
            });
            return arr.join('.');
        },

		/**
		 * 监听制品上传表单加载完成
		 */
		pkgUploadFormOnAddEvent: function () {
			let _this = this;
			let mos = new MutationObserver(function (mutations, observer) {
				for (const mutation in mutations) {
					if (Object.hasOwnProperty.call(mutations, mutation)) {
						const element = mutations[mutation];
						if (
							element.target.classList.contains(
								'next-dialog-body'
							)
						) {
							_this.pkgUploadFormUpdate();
						}
					}
				}
			});

			mos.observe(document.body, {
				attributes: true,
				childList: true,
				subtree: true,
			});
		},

		/**
		 * 获取指定元素的目标父元素
		 * @param {Dom} element
		 * @param {String} className
		 * @returns Dom[]
		 */
		getParent: function (element, className) {
			if (!element || !className) {
				return false;
			}

			let _this = this;
			while (element.parentNode) {
				element = element.parentNode;
				if (
					element.classList &&
					element.classList.contains(className)
				) {
					return element;
				}
				_this.getParent(element, className);
			}

			return null;
		},
	};
	app.init();
})();