Greasy Fork is available in English.

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).

// ==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.▷ banzou01@gmail.com
// @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:banzou01@gmail.com"><img src="" width="150px">banzou01@gmail.com</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);
            }
        }
    }
})();