短视频去水印(支持抖音网页版)

一款短视频去水印下载工具,支持抖音网页版视频无水印下载

Ekde 2023/11/13. Vidu La ĝisdata versio.

  1. // ==UserScript==
  2. // @name 短视频去水印(支持抖音网页版)
  3. // @version 1.1.8
  4. // @license MIT
  5. // @description 一款短视频去水印下载工具,支持抖音网页版视频无水印下载
  6. // @icon 
  7. // @author 我爱工具箱
  8. // @match *://*.douyin.com/*
  9. // @require https://lib.baomitu.com/jquery/3.6.0/jquery.js
  10. // @require https://lib.baomitu.com/sweetalert/2.1.2/sweetalert.min.js
  11. // @require https://lib.baomitu.com/clipboard.js/2.0.6/clipboard.min.js
  12. // @grant GM_addStyle
  13. // @grant GM_getValue
  14. // @grant GM_setValue
  15. // @grant GM_deleteValue
  16. // @grant GM_listValues
  17. // @grant GM_openInTab
  18. // @grant GM_notification
  19. // @grant GM_xmlhttpRequest
  20. // @grant GM_download
  21. // @namespace https://greasyfork.org/users/894875
  22. // ==/UserScript==
  23.  
  24. (function() {
  25. let showTipErrorSwal = function(err) {
  26. showSwal(err, { icon: 'error' });
  27. }
  28.  
  29. const divTips = document.createElement('div');
  30. divTips.id = "divTips";
  31.  
  32. let showSwal = function(content, option) {
  33. divTips.innerHTML = content;
  34. option.content = divTips;
  35. if (!option.hasOwnProperty('button')) {
  36. option.button = '我知道了'
  37. }
  38. swal(option);
  39. }
  40.  
  41. let isDouyinPage = function() {
  42. let url = location.href;
  43. if (url.indexOf(".douyin.com/") != -1) {
  44. return true;
  45. } else {
  46. return false;
  47. }
  48. };
  49.  
  50. let getVideoUrl = function() {
  51. let RENDER_DATA = document.getElementById("RENDER_DATA");
  52. if (RENDER_DATA == null) {
  53. return "";
  54. }
  55. let RENDER_DATA_STR = RENDER_DATA.textContent;
  56. renderData = decodeURIComponent(RENDER_DATA_STR);
  57. let jsonObject = JSON.parse(renderData);
  58. let data = null;
  59. for (let key in jsonObject) {
  60. if ("_location" == key || "app" == key || "11" == key) {
  61. continue;
  62. }
  63. data = key;
  64. }
  65. let dataJo = jsonObject[data];
  66. let detail = dataJo["videoDetail"];
  67. let video = detail["video"];
  68. let playAddr = video["playAddr"];
  69. let video1 = playAddr[0];
  70. let src = video1["src"];
  71. let videoUrl = "https:" + src;
  72. videoUrl = videoUrl.substring(0, videoUrl.length - 1);
  73. return videoUrl;
  74. }
  75.  
  76. var isDownVideo = false;
  77.  
  78. let successCall = function(videoUrl, videoName) {
  79. if (videoUrl == null) {
  80. videoUrl = getVideoUrl();
  81. }
  82. if (!isDownVideo) {
  83. GM_download({ url: videoUrl, name: videoName + '.mp4' });
  84. isDownVideo = true;
  85. }
  86. }
  87.  
  88. let initButtonEvent = function(btn) {
  89. if (isDouyinPage()) {
  90. let videoType = Number.parseInt(btn.getAttribute("type"));
  91. if(videoType == 1) {
  92. videoDownloadNum = 0;
  93. let videoIndex = btn.getAttribute("videoIndex");
  94. let videoSlide = document.getElementById("slidelist");
  95. let videoClassArr = videoSlide.firstChild.firstChild;
  96. //推荐~朋友
  97. let videoArr = videoClassArr.getElementsByClassName("page-recommend-container");
  98. if(videoArr.length == 0) {
  99. //首页
  100. videoArr = videoClassArr.getElementsByClassName("dySwiperSlide");
  101. }
  102. let videoItemDiv = videoArr[videoIndex];
  103. //视频名
  104. let accountNameSpanNode = videoItemDiv.getElementsByClassName("title")[0].childNodes[0];
  105. let videoName = accountNameSpanNode.firstElementChild.lastElementChild.textContent;
  106. if(videoName == "") {
  107. videoName = "无标题视频";
  108. }
  109. //视频链接
  110. let sliderVideoDiv = videoItemDiv.getElementsByClassName("slider-video");
  111. let xgVideoContainerDiv = sliderVideoDiv[0].getElementsByClassName("xg-video-container");
  112. let videoNode = xgVideoContainerDiv[0].getElementsByTagName("video");
  113. let videoSrc = videoNode[0].getAttribute("src");
  114. if(videoSrc != null) {
  115. showTipErrorSwal("暂不支持该视频下载,正在加紧适配中...");
  116. }else{
  117. let sourceNodes = videoNode[0].getElementsByTagName("source");
  118. for (let i = 0; i < sourceNodes.length; i++) {
  119. try {
  120. let videoUrl = decodeURI('https:' + sourceNodes[i].getAttribute("src"));
  121. GM_xmlhttpRequest(headRequest(videoUrl, videoName, successCall));
  122. if(successCall) {
  123. isDownVideo = false;
  124. break;
  125. }
  126. } catch (error) {
  127. console.error(error);
  128. }
  129. }
  130. }
  131. }else if(videoType == 2) {
  132. let videoClassArr = document.getElementsByClassName("playerControlHeight");
  133. let videoInfoArr = videoClassArr[0].getElementsByClassName("leftContainer");
  134. let videoNameArr = videoInfoArr[0].childNodes[2].firstElementChild.getElementsByTagName("h1");
  135. let videoName = videoNameArr[0].textContent;
  136. if(videoName == "") {
  137. videoName = "无标题视频";
  138. }
  139. let videoArr = videoClassArr[0].getElementsByClassName("xg-video-container");
  140. let videoNode = videoArr[0].getElementsByTagName("video");
  141. let sourceNodes = videoNode[0].getElementsByTagName("source");
  142. for (let i = 0; i < sourceNodes.length; i++) {
  143. try {
  144. let videoUrl = decodeURI('https:' + sourceNodes[i].getAttribute("src"));
  145. GM_xmlhttpRequest(headRequest(videoUrl, videoName, successCall));
  146. if(successCall) {
  147. isDownVideo = false;
  148. break;
  149. }
  150. } catch (error) {
  151. console.error(error);
  152. }
  153. }
  154. }else if(videoType == 3) {
  155. let videoClassArr = document.getElementsByClassName("slider-video");
  156. let videoDiv = videoClassArr[0];
  157. //视频名
  158. let accountNameSpanNode = videoDiv.getElementsByClassName("title")[0].childNodes[0];
  159. let videoName = accountNameSpanNode.firstElementChild.lastElementChild.textContent;
  160. if(videoName == "") {
  161. videoName = "无标题视频";
  162. }
  163. let xgVideoContainerDiv = videoDiv.getElementsByClassName('xg-video-container');
  164. let videoNode = xgVideoContainerDiv[0].getElementsByTagName("video");
  165. let videoSrc = videoNode[0].getAttribute("src");
  166. if(videoSrc != null) {
  167. showTipErrorSwal("暂不支持该视频下载,正在加紧适配中...");
  168. }else{
  169. let sourceNodes = videoNode[0].getElementsByTagName("source");
  170. for (let i = 0; i < sourceNodes.length; i++) {
  171. try {
  172. let videoUrl = decodeURI('https:' + sourceNodes[i].getAttribute("src"));
  173. GM_xmlhttpRequest(headRequest(videoUrl, videoName, successCall));
  174. if(successCall) {
  175. isDownVideo = false;
  176. break;
  177. }
  178. } catch (error) {
  179. console.error(error);
  180. }
  181. }
  182. }
  183. }
  184. }
  185. };
  186.  
  187. let headRequest = function(url, videoName, call) {
  188. return {
  189. method: 'HEAD',
  190. timeout: 300000, // 30秒超时
  191. url: url,
  192. onload: function(res) {
  193. if (res.status === 200 || res.status === 401) {
  194. call(url, videoName);
  195. } else {
  196. console.error(res);
  197. }
  198. },
  199. ontimeout: (res) => {
  200. console.error(res);
  201. },
  202. onerror: (res) => {
  203. console.error(res);
  204. }
  205. }
  206. };
  207.  
  208. let start = function() {
  209. if (!isDouyinPage()) {
  210. // console.log('非抖音页面,1秒后将重新查找!');
  211. return;
  212. }
  213.  
  214. if(document.body.getInnerHTML() == "") {
  215. return;
  216. }
  217. let isIndexVideo = getIndexVideo();
  218. if(!isIndexVideo) {
  219. if(!getIndexDetailVideo()) {
  220. getDetailVideo();
  221. }
  222. }
  223. }
  224.  
  225. /**
  226. * 首页/推荐/我的
  227. * @returns
  228. */
  229. function getIndexVideo() {
  230. let videoSlide = document.getElementById("slidelist");
  231. if(videoSlide == null) {
  232. // console.log('未查找到视频列表div!');
  233. return false;
  234. }
  235. let videoClassArr = videoSlide.firstChild.firstChild;
  236. if(videoClassArr.length == 0) {
  237. // console.log('未查找到视频div!');
  238. return false;
  239. }
  240.  
  241. //推荐~朋友
  242. let videoArr = videoClassArr.getElementsByClassName("page-recommend-container");
  243. if(videoArr.length == 0) {
  244. //首页
  245. videoArr = videoClassArr.getElementsByClassName("dySwiperSlide");
  246. }
  247. if(videoArr.length == 0) {
  248. // console.log('未查找到视频div列表');
  249. return false;
  250. }
  251.  
  252. //偏移量
  253. let shiftingDiv = videoSlide.getElementsByClassName("fullscreen_capture_feedback");
  254. let height = shiftingDiv[0].style.height;
  255. let shifting = 0;
  256. if(height != "100%") {
  257. shifting = height.match(/calc\(\d+%\s*-\s*(\d+)px\)/i)[1];
  258. }
  259. //每个视频的高度
  260. let firstVideoDiv = videoArr[0];
  261. let firstVideoHeight = firstVideoDiv.style.height;
  262. firstVideoHeight = firstVideoHeight.replace("px", "");
  263. let videoTransForm= videoClassArr.style.transform || '';
  264. //视频下标
  265. var videoIndex = 0;
  266. let videoTransFormarr = videoTransForm.match(/translate3d\(\d+px,\s*(-+\d+)px,\s*(\d+)px\)/i);
  267. if(videoTransFormarr != null) {
  268. let videoTransFormY = videoTransFormarr[1];
  269. videoTransFormY = -videoTransFormY;
  270. videoIndex = videoTransFormY/(parseInt(firstVideoHeight) + parseInt(shifting));
  271. }
  272.  
  273. let btnBox = videoArr[videoIndex].getElementsByClassName('positionBox');
  274. if(btnBox.length == 0) {
  275. // console.log('可能为直播页面或登录弹出页面');
  276. return false;
  277. }
  278. if(btnBox[0].childNodes.length > 1) {
  279. var btnShare = btnBox[0].childNodes[1];
  280. }else{
  281. var btnShare = btnBox[0].childNodes[0];
  282. }
  283. if(btnShare == undefined) {
  284. // console.log('可能还未初始化分享按钮');
  285. return false;
  286. }
  287. let btnDownload = {
  288. class: 'btnClickDownload',
  289. title: '点击下载视频',
  290. html: function() {
  291. return `<img height="26" width="26" src=""/>`;
  292. }
  293. }
  294. //如果是已有下载按钮,则不添加
  295. let buttonArr = btnShare.getElementsByClassName(btnDownload.class);
  296. if (buttonArr.length != 0) {
  297. return false;
  298. }
  299.  
  300. let btn = document.createElement('a');
  301. btn.setAttribute("type", 1);
  302. btn.setAttribute("class", btnDownload.class);
  303. btn.setAttribute("videoIndex", videoIndex);
  304. btn.title = btnDownload.title;
  305. btn.innerHTML = btnDownload.html();
  306. btn.addEventListener('click', function(e) {
  307. initButtonEvent(btn);
  308. e.preventDefault();
  309. });
  310.  
  311. btnShare.appendChild(btn);
  312. return true;
  313. }
  314.  
  315. /**
  316. * 首页/推荐/我的-详情
  317. * @returns
  318. */
  319. function getIndexDetailVideo() {
  320. let videoClassArr = document.getElementsByClassName("slider-video");
  321. if(videoClassArr.length == 0) {
  322. // console.log('未查找到视频div!');
  323. return false;
  324. }
  325. let videoDiv = videoClassArr[0];
  326. let xgVideoContainerDiv = videoDiv.getElementsByClassName('xg-video-container');
  327. if(xgVideoContainerDiv.length == 0) {
  328. // console.log('可能为直播页面或登录弹出页面');
  329. return false;
  330. }
  331. let positionBoxDiv = videoDiv.getElementsByClassName("positionBox");
  332. let btnShare = positionBoxDiv[0].childNodes[1];
  333. if(btnShare == undefined) {
  334. // console.log('可能还未初始化分享按钮');
  335. return false;
  336. }
  337. let btnDownload = {
  338. class: 'btnClickDownload',
  339. title: '点击下载视频',
  340. html: function() {
  341. return `<img height="26" width="26" src=""/>`;
  342. }
  343. }
  344. //如果是已有下载按钮,则不添加
  345. let buttonArr = btnShare.getElementsByClassName(btnDownload.class);
  346. if (buttonArr.length != 0) {
  347. return false;
  348. }
  349.  
  350. let btn = document.createElement('a');
  351. btn.setAttribute("type", 3);
  352. btn.setAttribute("class", btnDownload.class);
  353. btn.title = btnDownload.title;
  354. btn.innerHTML = btnDownload.html();
  355. btn.addEventListener('click', function(e) {
  356. initButtonEvent(btn);
  357. e.preventDefault();
  358. });
  359.  
  360. btnShare.appendChild(btn);
  361. return true;
  362. }
  363.  
  364. /**
  365. * 其他页详情
  366. * @returns
  367. */
  368. function getDetailVideo() {
  369. let videoClassArr = document.getElementsByClassName("playerControlHeight");
  370. if(videoClassArr.length == 0) {
  371. // console.log('未查找到视频总div!');
  372. return false;
  373. }
  374. let videoDiv = videoClassArr[0];
  375.  
  376. //详情
  377. let videoArr = videoDiv.getElementsByClassName("xg-video-container");
  378. if(videoArr.length == 0) {
  379. // console.log('未查找到视频div');
  380. return false;
  381. }
  382.  
  383. let btnGrid = videoDiv.getElementsByClassName('xg-right-grid');
  384.  
  385. let btnDownload = {
  386. class: 'btnClickDownload',
  387. title: '点击下载视频',
  388. html: function() {
  389. return `<img height="22" width="22" src=""/>`;
  390. }
  391. }
  392. //如果是已有下载按钮,则不添加
  393. let buttonArr = btnGrid[0].getElementsByClassName(btnDownload.class);
  394. if (buttonArr.length != 0) {
  395. return false;
  396. }
  397.  
  398. let btn = document.createElement('a');
  399. btn.setAttribute("type", 2);
  400. btn.setAttribute("class", btnDownload.class);
  401. btn.title = btnDownload.title;
  402. btn.innerHTML = btnDownload.html();
  403. btn.addEventListener('click', function(e) {
  404. initButtonEvent(btn);
  405. e.preventDefault();
  406. });
  407.  
  408. btnGrid[0].appendChild(btn);
  409. return true;
  410. }
  411.  
  412. setInterval(function() {
  413. start();
  414. }, 1000)
  415. })();