天翼云盘cloud.189.cn批量转存

当天翼云文件夹超过1000时使用最合适,低于1000文件请用天翼云自带功能!

  1. // ==UserScript==
  2. // @name 天翼云盘cloud.189.cn批量转存
  3. // @namespace StartMenu
  4. // @version 1.1
  5. // @description 当天翼云文件夹超过1000时使用最合适,低于1000文件请用天翼云自带功能!
  6. // @author StartMenu
  7. // @license MIT
  8. // @match https://cloud.189.cn/web/share?code=*
  9. // @icon https://www.google.com/s2/favicons?domain=189.cn
  10. // @grant GM_addStyle
  11. // @grant GM_addElement
  12.  
  13. // @ 测试用: https://cloud.189.cn/web/share?code=iqQjMn3e6Nfu
  14. // @ 测试用2 : https://cloud.189.cn/web/share?code=reIJv2rE7Fni
  15.  
  16. // ==/UserScript==
  17. //动作间隔,单位毫秒,根据自己的网速和电脑性能调节
  18. let ActionInterval=1000;
  19. //一次保存数量,普通用户最多1000,设太大容易卡住
  20. let SaveNumber=200;
  21. //保存文件夹名
  22. let SaveDirectoryName="批量转存-";
  23.  
  24. //以下变量不要修改
  25. //文件列表,它自己的分页都有问题,出现重复文件,会导致保存失败
  26. let NowFileList=[];
  27. //加载是否完成
  28. let LoadFinish=false;
  29. let MYloadingText;
  30.  
  31. function RemoveRepeatItem() {
  32. document.querySelectorAll(".file-list-ul>li").forEach(function (Item) {
  33. if (NowFileList.indexOf(Item.querySelector(".file-item-name-fileName-span").textContent.trim())>-1){
  34. Item.remove();
  35. }
  36. });
  37. }
  38.  
  39. function SelectListFile() {
  40. let Items=document.querySelectorAll(".file-list-ul>li");
  41. for (let i = 0; i <Items.length; i++) {
  42. if (i<=SaveNumber){
  43. let Name=Items[i].querySelector(".file-item-name-fileName-span").textContent.trim();
  44. NowFileList.push(Name);
  45. Items[i].querySelector(".file-item-check").click();
  46. }
  47. else
  48. {
  49. break;
  50. }
  51. }
  52. //console.log("完成文件选择",NowFileList);
  53. console.log("完成文件选择:"+NowFileList.length);
  54. }
  55.  
  56. async function ClickNewSaveButton() {
  57. return new Promise(resolve => {
  58. let Interval=setInterval(function () {
  59. //点击转存
  60. let SaveButton = document.querySelector("[class^=FileListHead_file-list-li-head-select]").nextSibling.firstChild;
  61. if(SaveButton!==null){
  62. clearInterval(Interval);
  63. SaveButton.click();
  64. return resolve();
  65. }
  66. },0);
  67. });
  68. }
  69.  
  70. // document.querySelector("[class^=Directory_directory-button]")
  71. // 生成需要的文件夹
  72. async function makeDir(){
  73. return new Promise(resolve => {
  74. let makeDirCheck=setInterval(function () {
  75. //let makeDirectory=document.querySelector("[class^=Directory_directory-button]").firstChild;
  76. //makeDirectory.click();
  77.  
  78. // console.log('查找新建按钮');
  79. // console.log(makeDirectory);
  80. let url = 'https://cloud.189.cn/api/open/file/createFolder.action?noCache=0.'+Date.now();
  81. // document.querySelectorAll(".info-detail-name span")[0].textContent.trim()
  82. let data = {
  83. parentFolderId: "-11",
  84. folderName: SaveDirectoryName+document.querySelector("[class^=FileListHead_file-list-nav]").lastChild.textContent.trim()
  85. };
  86.  
  87. // 将数据转换为x-www-form-urlencoded格式
  88. let params = Object.keys(data).map(function(key) {
  89. return encodeURIComponent(key) + '=' + encodeURIComponent(data[key]);
  90. }).join('&');
  91.  
  92. fetch(url, {
  93. method: 'POST',
  94. headers: {
  95. 'Content-Type': 'application/x-www-form-urlencoded',
  96. },
  97. body: params,
  98. })
  99. .then(function(){
  100. clearInterval(makeDirCheck);
  101. })
  102. .catch((error) => {
  103. });
  104.  
  105. return resolve();
  106. },ActionInterval);
  107. });
  108. }
  109.  
  110. /**
  111. 保存时覆盖已有的文件
  112. */
  113. function autoRewrite(){
  114. let myAutoRewrite=setInterval(function () {
  115. if(document.querySelector("[class^=HandleConflict_conflict-choose] input[type='checkbox']:not(:checked)")){
  116. document.querySelector("[class^=HandleConflict_conflict-choose] input[type='checkbox']:not(:checked)").click();
  117. }
  118. if(document.querySelector("[class^=HandleConflict_conflict-bottom-button-replace]")){
  119. document.querySelector("[class^=HandleConflict_conflict-bottom-button-replace]").click();
  120. }
  121. },ActionInterval);
  122. }
  123.  
  124.  
  125. async function SelectSaveDirectory() {
  126. let selectedDir = false;
  127. return new Promise(resolve => {
  128.  
  129. let Check=setInterval(function () {
  130. let AllDirectory=document.querySelectorAll("[class^=DirectoryTree_directory-li] [class^=DirectoryTree_c-directory-tree]>div");
  131. for (let i = 0; i < AllDirectory.length; i++) {
  132. let Name=AllDirectory[i].querySelector(".directory-name").textContent.trim();
  133. if (Name===SaveDirectoryName+document.querySelector("[class^=FileListHead_file-list-nav]").lastChild.textContent.trim()){
  134. clearInterval(Check)
  135. AllDirectory[i].childNodes[0].childNodes[1].dispatchEvent(new MouseEvent("mousedown"));
  136. selectedDir = true;
  137. break;
  138. }
  139. }
  140. //目录加载完了还是没看到文件夹,刷新一下
  141. if(AllDirectory.length > 0 && !selectedDir){
  142. document.querySelector("[class^=Directory_directory-button]").nextElementSibling.click();
  143. }
  144. return resolve();
  145. },ActionInterval);
  146. });
  147. }
  148.  
  149. function ClickYesButton() {
  150. let YesButton=document.querySelector("[class^=Directory_directory-button-confirm]");
  151. YesButton.click();
  152. }
  153.  
  154. async function SaveIsDone() {
  155. // 页面切到后台会导致此方法失效,不管用下面哪种方式。 都无法检测当转存的项已经完成!
  156. /*
  157. return new Promise(resolve => {
  158. let Loading=setInterval(function () {
  159. let LoadingBlock=document.querySelector("[class^=loading_c-loading]");
  160. if (LoadingBlock.style.display==="none"){
  161. clearInterval(Loading);
  162. return resolve();
  163. }
  164. },ActionInterval);
  165. });
  166. */
  167. return new Promise(resolve => {
  168. let Loading=setInterval(function () {
  169. let LoadingBlock=document.querySelector(".ant-message .ant-message-notice");
  170. if (LoadingBlock){
  171. console.log("消息完成!");
  172. clearInterval(Loading);
  173. return resolve();
  174. }
  175. },100);
  176. });
  177.  
  178. }
  179.  
  180. function RemoveSelectItem() {
  181. let List=document.querySelectorAll(".file-list-ul .selected");
  182. for (let i = 0; i < List.length; i++) {
  183. List[i].querySelector("input").click();
  184. List[i].remove();
  185. }
  186. }
  187.  
  188.  
  189. let nowsaveIndex = 0; //当前保存到第几个文件了
  190.  
  191. async function StartSave() {
  192.  
  193. console.log("开始保存");
  194. MYloadingText.textContent="正在保存选中的文件(页面不要切到后台)";
  195. //删除重复项目
  196. RemoveRepeatItem();
  197. //选择列表文件
  198. SelectListFile();
  199. //点击转存
  200. await ClickNewSaveButton();
  201. //选择保存文件夹
  202. await SelectSaveDirectory();
  203.  
  204. //点击确定
  205. ClickYesButton();
  206. //等待转存完成
  207. await SaveIsDone();
  208. //移除选择项
  209. RemoveSelectItem();
  210. console.log("当前保存完成");
  211. if (LoadFinish&&document.querySelectorAll(".file-list-ul>li").length===0){
  212. //alert("全部保存完成");
  213. MYloadingText.textContent="全部保存完成";
  214. }
  215. else{
  216. StartSave();
  217. }
  218. }
  219.  
  220. function InitHtml() {
  221. //
  222. //document.querySelector(".info-detail")
  223. let Holder=document.querySelector(".info-detail h2");
  224. let SaveAllButton=GM_addElement(Holder,"button",{textContent:"全部保存",class:"JXNButton"});
  225.  
  226. //绑定事件
  227. SaveAllButton.onclick=function () {
  228. //隐藏保存按钮
  229. SaveAllButton.style.display="none";
  230. MYloadingText=GM_addElement(Holder,"button",{textContent:"正在加载(页面不要切到后台)",class:"JXNButton"});
  231. //先强制排一下序
  232. document.querySelector('.file-list-ul').previousElementSibling.childNodes[1].childNodes[1].click();
  233. //生成目录
  234. makeDir();
  235. //设置全局显示
  236. ScrollFileList2();
  237. }
  238. }
  239.  
  240. let trytimes = 0; //加载时尝试次数
  241. function ScrollFileList2() {
  242. let LastName=""; // 最后一个元素
  243. MYloadingText.textContent="正在加载所有文件(页面不要切到后台)";
  244. let ScrollFileList2xx=setInterval(function () {
  245. if (document.querySelector(".file-list-load")===null){
  246. //多等5秒再执行
  247. if(trytimes < 5){
  248. trytimes++;
  249. }
  250. else{
  251. //取当前最后一个文件名 .querySelector(".file-list-ul")
  252. let CheckLastName=document.querySelector(".selDiv").previousSibling.querySelector(".file-item-name-fileName-span").textContent.trim();
  253. console.log(CheckLastName);
  254. if(CheckLastName===LastName){
  255. console.log("列表全部加载完成");
  256. LoadFinish=true;
  257. MYloadingText.textContent="完成加载所有文件(页面不要切到后台)";
  258. let NowLength=document.querySelector(".file-list-ul").querySelectorAll("li").length;
  259. console.log("当前加载完成,数量"+NowLength);
  260. clearInterval(ScrollFileList2xx);
  261. StartSave();
  262. }
  263. else{
  264. LastName=CheckLastName;
  265. document.querySelector(".file-list-ul").scrollTo(0,document.querySelector(".file-list-ul").scrollHeight);
  266. }
  267. }
  268. }
  269. else {
  270. console.log("正在加载");
  271. }
  272. },ActionInterval);
  273. }
  274.  
  275.  
  276. function InitCss() {
  277. GM_addStyle(`.JXNButton {
  278. left: 0;
  279. padding: 5px 10px;
  280. background-color: #ff1eee;
  281. color: white;
  282. border: none;
  283. cursor: pointer;
  284. border-radius: 5px;
  285. }`);
  286. }
  287.  
  288. /**
  289. 调用自己,直到file-list-ul出现,再返回执行下面的操作
  290. */
  291. async function WaitUI() {
  292. return new Promise(resolve => {
  293. let CheckHtml=setInterval(function () {
  294. if (document.querySelector(".file-list-ul")!==null){
  295. clearInterval(CheckHtml);
  296. resolve();
  297. }
  298. },ActionInterval);
  299. });
  300. }
  301.  
  302. function Run() {
  303. //等待界面加载完成
  304. WaitUI().then(()=>{
  305. InitCss();
  306. InitHtml();
  307. });
  308. }
  309. Run();

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址