百度网盘 - 更高级的重命名

你不知道的百度网盘重命名姿势~支持批量替换重命名、手动批量修改后一次提交、智能重命名,拒绝转圈圈!

// ==UserScript==
// @name            百度网盘 - 更高级的重命名
// @version         1.0.5
// @description     你不知道的百度网盘重命名姿势~支持批量替换重命名、手动批量修改后一次提交、智能重命名,拒绝转圈圈!
// @author          Priate
// @match           https://pan.baidu.com/*
// @grant           GM_setValue
// @grant           GM_getValue
// @grant           GM_addStyle
// @icon            https://nd-static.bdstatic.com/m-static/v20-main/favicon-main.ico
// @require         https://unpkg.com/vue@2
// @require         https://unpkg.com/sweetalert@2.1.2/dist/sweetalert.min.js
// @require         https://unpkg.com/jquery@3.2.1/dist/jquery.min.js
// @require         https://greasyfork.org/scripts/435476-priatelib/code/PriateLib.js?version=1202493
// @supportURL      https://greasyfork.org/scripts/443771/feedback
// @homepageURL     https://greasyfork.org/scripts/443771
// @contributionURL https://afdian.net/@cyberubbish
// @license         MIT
// @namespace       https://greasyfork.org/users/219866
// ==/UserScript==

(function() {
	'use strict';

	function initSetting() {
		var setting;
		if (!GM_getValue('priate_script_bdwp_data')) {
			GM_setValue('priate_script_bdwp_data', {
				left: 20,
				top: 100,
			})
		}
		setting = GM_getValue('priate_script_bdwp_data')
		GM_setValue('priate_script_bdwp_data', setting)
	}

	function injectDiv() {
		var priate_script_div = document.createElement("div")
		priate_script_div.innerHTML = `
<div id="priate_script_div">
<b style='font-size:30px; font-weight:300; margin: 10px 20px'>更高级的重命名</b>
<p id='priate_script_setting' style='margin: 0 0'>
❤️ by <a @click='openDonate' style='color:#337ab7'>Priate</a> |
v <a href="//greasyfork.org/scripts/443771" target="_blank" style='color:#ff6666'>{{version}}</a> |
<a @click="switchDrag">📌</a> | <a @click="recoverText">♻️</a> | <a v-show="data.length > 0" @click="smartRename">🤖️ |</a> <a @click="clearData">❌</a>
<br>
文件夹重命名 : <a @click='todo' :style='"color:" + (false ? "#00947e" : "#CC0F35")'> 关闭 </a> |
正则 : <a @click='todo' :style='"color:" + (false ? "#00947e" : "#CC0F35")'> 关闭 </a>
<div v-show="data.length">
<a style='color:#946C00;' @click="changeOriginText" v-show="showOriginText">{{'【' + originText + '】'}}</a>
<textarea class="replaceArea" ref="originTextInput" v-show="!showOriginText" @blur="recoverOriginText" type="text" v-model="originText" ></textarea>
<a @click="autoRename">➡️ ➡️ ➡️</a>
<a style='color:#00947e' @click="changeReplaceText" v-show="showReplaceText">{{'【' + replaceText + '】' }}</a>
<textarea class="replaceArea" ref="replaceTextInput" v-show="!showReplaceText" @blur="recoverReplaceText" type="text" v-model="replaceText" ></textarea>
</div>
</p>
<button @click="loadData">{{data.length > 0 ? '重载数据' : '加载数据'}}</button>
<button @click="postRename" v-show="musicList.length > 0"> 确定修改 </button>
</br>
<table v-show="data.length > 0">
<thead><tr><th></th><th>原始文件名</th><th>修改后文件名</th></tr></thead>
<tbody id="priate_script_table">
<tr v-for="(item, index) in data" :key="index">
<td><input class="checkMusicBox" v-model="musicList" :value='item' type="checkbox" :disabled="item.replace == item.origin"></td>
<td><a @click="originText = item.origin" style='color:#337ab7'>{{item.origin}}</a></td>
<td><a v-show="!item.isChanging" @click="manualRename(item)" style='color:#C01D07'>{{item.replace}}</a><textarea class="replaceArea" :ref="item.id" @blur="modifyReplace(item)" v-show="item.isChanging" v-model="item.replace"></textarea></td>
</tr>
</tbody>
</table>
</div>
`
		GM_addStyle(`
#priate_script_div{
font-size : 15px;
position: fixed;
background-color: rgb(240, 223, 175);
color : #660000;
text-align : center;
padding: 10px;
z-index : 9999;
border-radius : 20px;
border:2px solid #660000;
font-weight: 300;
text-stroke: 0.5px;
box-shadow: 5px 15px 15px rgba(0,0,0,0.4);
user-select : none;
-webkit-user-select : none;
-moz-user-select : none;
-ms-user-select:none;
}
#priate_script_div:hover{
box-shadow: 5px 15px 15px rgba(0,0,0,0.8);
transition: box-shadow 0.3s;
}

#priate_script_div a{
margin-bottom: 2px !important;
}
.priate_script_hide{
padding: 0 !important;
border:none !important;
}
a{
cursor : pointer;
text-decoration : none;
}
/*表格样式*/
#priate_script_div table{
text-align: center;
// border:2px solid #660000;
margin: 5px auto;
padding: 2px;
border-collapse: collapse;
display: block;
height : 400px;
overflow-y: scroll;
}
/*表格框样式*/
#priate_script_div td{
border:2px solid #660000;
padding: 8px 12px;
max-width : 300px;
word-wrap : break-word;
}
/*表头样式*/
#priate_script_div th{
border:2px solid #660000;
padding: 8px 12px;
font-weight: 300;
-webkit-text-stroke: 0.5px;
text-stroke: 0.5px;
}

/*脚本按钮样式*/
#priate_script_div button{
display: inline-block;
border-radius: 4px;
border: 1px solid #660000;
background-color: transparent;
color: #660000;
text-decoration: none;
padding: 5px 10px;
margin : 5px 10px;
font-weight: 300;
-webkit-text-stroke: 0.5px;
text-stroke: 0.5px;
}
/*脚本按钮悬浮样式*/
#priate_script_div button:hover{
cursor : pointer;
color: rgb(240, 223, 175);
background-color: #660000;
transition: background-color 0.2s;
}
/*设置区域 p 标签*/
#priate_script_setting{
user-select : none;
-webkit-user-select : none;
-moz-user-select : none;
-ms-user-select:none;
}
/*swal按钮*/
.swal-button--1{
background-color: #FFFAEB !important;
color: #946C00;
}
.swal-button--2{
background-color: #ebfffc !important;
color: #00947e;
}
.swal-button--3{
background-color: #ECF6FD !important;
color: #55ACEE;
}
.checkMusicBox{
transform: scale(1.7,1.7);
cursor: pointer;
}
.replaceArea{
height : 100%;
width : 100%;
background-color: #fff;
border:1px solid #000000;
padding: 4px;
border-radius: 4px;
}
.replaceInput{
height : 100%;
width : 40%;
background-color: #fff;
border:1px solid #000000;
padding: 4px;
border-radius: 4px;
}
`);
		document.querySelector("html").appendChild(priate_script_div)
		var setting = GM_getValue('priate_script_bdwp_data')
		document.getElementById("priate_script_div").style.left = (setting.left || 20) + "px";
		document.getElementById("priate_script_div").style.top = (setting.top || 100) + "px";
	}

	function dragFunc(id) {
		var Drag = document.getElementById(id);
		var setting = GM_getValue('priate_script_bdwp_data')
		Drag.onmousedown = function(event) {
			var ev = event || window.event;
			event.stopPropagation();
			var disX = ev.clientX - Drag.offsetLeft;
			var disY = ev.clientY - Drag.offsetTop;
			document.onmousemove = function(event) {
				var ev = event || window.event;
				setting.left = ev.clientX - disX
				Drag.style.left = setting.left + "px";
				setting.top = ev.clientY - disY
				Drag.style.top = setting.top + "px";
				Drag.style.cursor = "move";
				GM_setValue('priate_script_bdwp_data', setting)
			};
		};
		Drag.onmouseup = function() {
			document.onmousemove = null;
			this.style.cursor = "default";
		};
	};

	function unDragFunc(id) {
		var Drag = document.getElementById(id);
		Drag.onmousedown = function(event) {
			var ev = event || window.event;
			event.stopPropagation();
		};
		Drag.onmouseup = function() {};
		Drag.onmousemove = function() {};
	};

	//初始化脚本设置
	initSetting()
	//注入脚本div
	injectDiv()

	function getToken() {
		return document.querySelector(".nd-main-list, .nd-new-main-list").__vue__.yunData.bdstoken;
	}

	function getLogid() {
		let ut = unsafeWindow.require("system-core:context/context.js").instanceForSystem.tools.baseService;
		return ut.base64Encode(base.getCookie("BAIDUID"));
	}

	function getPath() {
		var nowPath = location.href.match(/path=(.+?)(?:&|$)/);
		var pathValue;
		if (nowPath)
			pathValue = decodeURIComponent(nowPath[1]);
		else
			pathValue = "/";
		if (pathValue.charAt(0) !== "/")
			pathValue = "/" + pathValue; // 补齐路径前缀斜杠
		if (pathValue.charAt(pathValue.length - 1) !== "/")
			pathValue += "/"
		return pathValue
	}

	function reloadList() {
		document.querySelector(".nd-main-list, .nd-new-main-list").__vue__.reloadList();
	}

	// 处理数据等逻辑
	var vm = new Vue({
		el: '#priate_script_div',
		data: {
			version: "1.0.5",
			setting: GM_getValue('priate_script_bdwp_data'),
			data: [],
			musicList: [],
			showOriginText: true,
			showReplaceText: true,
			originText: "需要替换的内容",
			replaceText: "",
			path: '/',
			isDrag: true,
		},
		methods: {
			loadData() {
				const path = getPath()
				this.path = path
				const all_tr = document.querySelectorAll('table.wp-s-pan-table__body-table tbody>tr');
				var result = [];
				var _this = this
				all_tr.forEach((item) => {
					const type = item.querySelector('img[alt]') ? item.querySelector('img[alt]').getAttribute('alt') : 'unknow'
					if (!item.querySelector('a')) return
					const music = {
						id: item.getAttribute('data-id'),
						origin: item.querySelector('a').getAttribute('title'),
						replace: item.querySelector('a').getAttribute('title'),
						isChanging: false,
						type,
						isFolder: type == 'folder',
					}
					result.push(music)
				})
				// 如果仍未获取到数据
				if (result.length == 0) {
					swal("未获取到数据,请确认此目录下有文件或文件夹!", {
						icon: "error",
						buttons: false,
						timer: 3000,
					});
				}
				this.data = result
				this.musicList = []
			},
			clearData() {
				if (this.data.length == 0) swal(`已经是最简形态了!`, {
					buttons: false,
					timer: 2000,
				});
				this.data = []
				this.musicList = []
			},
			openDonate() {
				showDonate()
			},
			async manualRename(item) {
				this.unDrag()
				item.isChanging = true
				await Sleep(0.01)
				this.$refs[item.id][0].focus();
			},
			modifyReplace(item) {
				this.drag()
				item.isChanging = false
				item.replace = item.replace.trim()
				if (item.replace != item.origin) {
					if (!this.musicList.find(el => el.id == item.id)) {
						this.musicList.push(item)
					}
				} else {
					this.musicList = this.musicList.filter(el => el.id != item.id)
				}

			},
			autoRename() {
				var _this = this
				const origin = this.originText
				const replace = this.replaceText
				var hasChanged = false
				var hasEmpty = false
				this.data.forEach(item => {
					item.replace = item.replace.replace(origin, replace).trim()
					if (item.replace !== item.origin) {
						hasChanged = true
					}
					if (item.replace == "") {
						hasEmpty = true
						item.replace = item.origin
					}
					_this.modifyReplace(item)
				})
				if (!hasChanged) swal(`没有匹配到任何需要修改的文件!`, {
					icon: "error",
					buttons: false,
					timer: 3000,
				});
				if (hasEmpty) swal(`替换后某个文件名为空!`, {
					icon: "error",
					buttons: false,
					timer: 3000,
				});
			},
			async postRename() {
				var _this = this
				const token = getToken()
				// const logid = getLogid()
				const data = this.musicList.map(item => {
					return {
						id: item.id,
						path: `${_this.path}${item.origin}`,
						newname: item.replace,
					}
				})
				$.ajax(`https://pan.baidu.com/api/filemanager?async=2&onnest=fail&opera=rename&bdstoken=${getToken()}&clienttype=0&app_id=250528&web=1`, {
					type: 'post',
					data: {
						filelist: JSON.stringify(data)
					},
					complete: function(res) {
						const resp = res.responseJSON
						if (resp['errno'] == 0) {
							swal(`重命名已完成!\n如部分文件显示未修改请点击 ♻️ 按钮手动刷新!`, {
								icon: "success",
								buttons: false,
								timer: 3000,
							});
						} else {
							swal(`重命名失败!`, {
								icon: "error",
								buttons: false,
								timer: 3000,
							});
						}
						_this.clearData()
					}
				});
				await Sleep(3)
				reloadList()
			},
			switchDrag() {
				if (this.isDrag) {
					this.unDrag()
					swal(`悬浮窗已固定`, {
						buttons: false,
						timer: 2000,
					});
				} else {
					this.drag()
					swal(`悬浮窗可以拖动了`, {
						buttons: false,
						timer: 2000,
					});
				}
			},
			unDrag() {
				this.isDrag = false
				unDragFunc("priate_script_div")
			},
			drag() {
				this.isDrag = true
				dragFunc("priate_script_div")
			},
			async changeOriginText() {
				this.unDrag()
				this.showOriginText = false
				await Sleep(0.01)
				this.$refs.originTextInput.focus();
			},
			async changeReplaceText() {
				this.unDrag()
				this.showReplaceText = false
				await Sleep(0.01)
				this.$refs.replaceTextInput.focus();
			},
			recoverOriginText() {
				this.drag()
				this.showOriginText = true
			},
			recoverReplaceText() {
				this.drag()
				this.showReplaceText = true
			},
			recoverText() {
				reloadList()
			},
			smartRename() {
				var _this = this
				var hasChanged = false
				var hasEmpty = false
				var lengthStatistics = {}
				var maxLength = 0
				this.data.forEach(item => {
					const suffix = item.origin.split('.').length > 1 ? '.' + item.origin.split('.')[item.origin.split('.').length - 1] : ''
					item.numArr = item.origin.replace(suffix, '').match(/\d+/g) || []
					// 寻找出现次数最多的匹配项
					if (lengthStatistics[item.numArr.length] == undefined) lengthStatistics[item.numArr.length] = 1
					else lengthStatistics[item.numArr.length] += 1
				})
				for (var i in lengthStatistics) {
					maxLength = lengthStatistics[i] > maxLength ? i : maxLength
				}
				var seq = -1
				for (var index = 0; index < maxLength; index++) {
					const tempList = this.data.map(item => {
						if (item.numArr.length == maxLength) {
							return item.numArr[index]
						}
					}).filter(l => l != undefined)
					// 判断是否有重复元素
					if (Array.from(new Set(tempList)).length == tempList.length) {
						seq = index
						break
					}
				}
				if (seq < 0) return swal(`智能重命名失败,找不到唯一的数字序号!`, {
					icon: "error",
					buttons: false,
					timer: 3000,
				});
				this.data.forEach(item => {
					if (item.numArr.length == maxLength) {
						const suffix = item.origin.split('.').length > 1 ? '.' + item.origin.split('.')[item.origin.split('.').length - 1] : ''
						item.replace = `${item.numArr[seq]}${suffix}`.trim()
					}
					if (item.replace !== item.origin) {
						hasChanged = true
					}
					if (item.replace == "") {
						hasEmpty = true
						item.replace = item.origin
					}
					_this.modifyReplace(item)
				})

				if (!hasChanged) swal(`没有匹配到任何需要修改的文件!`, {
					icon: "error",
					buttons: false,
					timer: 3000,
				});
				if (hasEmpty) swal(`替换后某个文件名为空!`, {
					icon: "error",
					buttons: false,
					timer: 3000,
				});
			},
			todo() {
				swal(`🈲️🈲️🈲️ 此功能暂不可用,请等待版本更新 🔞🔞🔞`, {
					buttons: false,
					timer: 2000,
				});
			}
		},
		computed: {},
		mounted() {}
	})
	//设置div可拖动
	dragFunc("priate_script_div");
})();