LinkedIn Learning videos download directly from listing out (basic version for videos only)

LinkedIn Learning videos download directly from listing out (basic version for videos only).

您需要先安装一个扩展,例如 篡改猴Greasemonkey暴力猴,之后才能安装此脚本。

您需要先安装一个扩展,例如 篡改猴暴力猴,之后才能安装此脚本。

您需要先安装一个扩展,例如 篡改猴暴力猴,之后才能安装此脚本。

您需要先安装一个扩展,例如 篡改猴Userscripts ,之后才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。

您需要先安装用户脚本管理器扩展后才能安装此脚本。

(我已经安装了用户脚本管理器,让我安装!)

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

(我已经安装了用户样式管理器,让我安装!)

// ==UserScript==
// @name         LinkedIn Learning videos download directly from listing out (basic version for videos only)
// @namespace    http://lyndasub.mybanzou.net/
// @version      0.1.2
// @description  LinkedIn Learning videos download directly from listing out (basic version for videos only).
// @author       ▷Please ask me for scripts that can download subtitles, translate subtitles directly, and batch download videos from LinkedIn Learning.▷ [email protected]
// @match        https://www.linkedin.com/learning/*/*u=*
// @include      about:blank
// @icon         
// @grant        GM_xmlhttpRequest
// @grant        GM_setClipboard
// ==/UserScript==

(function() {
    'use strict';

    function leftPad(number, targetLength) {
        var output = number + '';
        while (output.length < targetLength) {
            output = '0' + output;
        }
        return output;
    }

    function validateTitle(title) {
        var newTitle = title.replace(/[\\\/:*?"<>|\r\n]/g, '_');
        return newTitle.trim();
    }
    if (document.body.innerHTML.match('My Learning')) {
        let newWindow = unsafeWindow.open('', '_blank', 'toolbar=yes,scrollbars=yes,resizable=yes,top=10,left=10,width=640,height=800');
        newWindow.document.body.innerHTML = '<div id="myPanel" style="background-color:rgba(88,201,98,0.7); color: #1F792C; "><div style="width:30%; text-align: center; height:800px; background-color:rgba(88,201,98,0.4); float:right;"><div style="padding-top: 100px;"><a href="mailto:[email protected]"><img src="" width="150px">[email protected]</a></div><div style="padding: 10px;">Please Donate Me!</div><a href="http://paypal.me/pendave" target="_blank"><img src="https://www.paypalobjects.com/webstatic/paypalme/images/social/pplogo384.png" width="100px">Paypal.me/pendave</a><hr>▲ Please ask me for scripts that can download subtitles, translate subtitles directly, and batch download videos from LinkedIn Learning.<br />Not FREE!<br />Accept Paypal.▲<hr></div><div id="myfloat" style="width:70%; height:779px; background-color:rgba(88,201,98,0.7); float:left; overflow-y: auto;"></div></div>';
        var newDiv = newWindow.document.body.querySelector('#myfloat');
        var chaptersNodes = document.querySelectorAll('section.classroom-toc-section');
        for (var n = 0; n < chaptersNodes.length; n++) {
            if (chaptersNodes[n].querySelector('button').getAttribute('aria-expanded') == "false") {
                chaptersNodes[n].querySelector('button').click();
            }
        }
        setTimeout(function() {
            if (!chaptersNodes[1].querySelector('span.classroom-toc-section__toggle-title').textContent.match(/1\.\s.+/)) {
                for (var h = 0; h < chaptersNodes.length; h++) {
                    chaptersNodes[h].querySelector('span.classroom-toc-section__toggle-title').textContent = h + '. ' + chaptersNodes[1].querySelector('span.classroom-toc-section__toggle-title').textContent.trim().replace(/\d+\.\s/, '');
                }
            } else {
                chaptersNodes[0].querySelector('span.classroom-toc-section__toggle-title').textContent = '0. ' + chaptersNodes[0].querySelector('span.classroom-toc-section__toggle-title').textContent.trim();
                if (!chaptersNodes[chaptersNodes.length - 1].querySelector('span.classroom-toc-section__toggle-title').textContent.match(/\d+\.\s.+/)) {
                    chaptersNodes[chaptersNodes.length - 1].querySelector('span.classroom-toc-section__toggle-title').textContent = chaptersNodes.length - 1 + '. ' + chaptersNodes[chaptersNodes.length - 1].querySelector('span.classroom-toc-section__toggle-title').textContent.trim();
                }
            }
            var videoNodes = document.querySelectorAll('a.classroom-toc-item__link');
            var videoTitleNodes = document.querySelectorAll('div.classroom-toc-item__title');
            for (var j = 0; j < videoNodes.length; j++) {
                var videoUrl = videoNodes[j].href;
                var videoTitle = validateTitle(videoTitleNodes[j].firstChild.wholeText.trim());
                console.log(videoTitle);
                console.log(j + ' / ' + videoTitleNodes.length + ' : ' + videoUrl);
                var idSrtPart = 'srtLoad' + j;
                var idVideoPart = 'videoLoad' + j;
                var idCopyName = 'copyName' + j;
                newDiv.innerHTML += '<div style="border-radius:10px; border: 1px solid white; padding: 3px; margin: 3px 2px;"><div id=' + idSrtPart + '></div><div id=' + idVideoPart + '>🎞️<button style="float: right;" id=' + idCopyName + '>✂ Video name</button></div></div>';
                let indexTimeArray = [];
                let textLineTransArray = [];
                var ret = GM_xmlhttpRequest({
                    method: "GET",
                    url: videoUrl,
                    onload: callback_function.bind({}, j, newDiv, videoTitle, indexTimeArray, textLineTransArray)
                });
            }
        }, 6000);

        function callback_function(num, newBuiltDiv, video_title, indexTimeArray, textLineTransArray, responseDetails) {
            let container = document.implementation.createHTMLDocument().documentElement;
            container.innerHTML = responseDetails.responseText;
            var codeNodes = container.querySelectorAll('code');
            for (var k = 0; k < codeNodes.length; k++) {
                if (codeNodes[k].textContent.match('learningApiTranscript')) {
                    var result = JSON.parse(codeNodes[k].textContent),
                        downloadVideoLinks = result['included'][2]['presentation']['videoPlay']['videoPlayMetadata']['progressiveStreams'],
                        downloadVideoLink;
                    for (var l = 0; l < downloadVideoLinks.length; l++) {
                        if (downloadVideoLinks[l]["height"] == 720) {
                            downloadVideoLink = downloadVideoLinks[l]["streamingLocations"][0]["url"];
                        }
                    }
                    var span = document.createElement('span');
                    span.download = leftPad(num, 2) + '. ' + video_title;
                    var linkText = document.createTextNode(leftPad(num, 2) + '. ' + video_title);
                    span.appendChild(linkText);
                    console.log('Callback with parameters (' + num + ').');
                    newBuiltDiv.querySelector('#srtLoad' + num).appendChild(span);
                    newBuiltDiv.querySelector('#videoLoad' + num).innerHTML += '<a href="' + downloadVideoLink + '" title="' + video_title + '" download="' + video_title + '.mp4" style="background-color: #46ED52;">' + leftPad(num, 2) + ' ⇩ video</a>';
                    newBuiltDiv.querySelector('#videoLoad' + num).innerHTML += '<span style="float: right; color: white;">' + downloadVideoLink.match(/\d{13}/g)[0] + ' </span>';
                }
            }
            var srtNodes = newBuiltDiv.querySelectorAll('div[id*="srtLoad"]');
            for (var q = 0; q < srtNodes.length; q++) {
                (function(q) {
                    newBuiltDiv.querySelector('#copyName' + q).onclick = function() {
                        GM_setClipboard(newBuiltDiv.querySelector('#srtLoad' + q + ' > span').innerText + '.mp4');
                        this.style.backgroundColor = '#46ED52';
                    };
                })(q);
            }
        }
    }
})();