您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
The backup and export tool Pinterest forgot, finally revived!
当前为
// ==UserScript== // @name Pinback Next // @namespace https://www.octt.eu.org/ // @match https://*.pinterest.*/* // @grant none // @version 0.5.0 // @author OctoSpacc // @license MIT // @description The backup and export tool Pinterest forgot, finally revived! // @run-at context-menu // @grant GM_registerMenuCommand // ==/UserScript== GM_registerMenuCommand('Open Export Tool', function(){ var version = '0.5.0' , boards = {} , board = {} , pins = [] , pin_count = 0 , username; if (match = location.href.match(/^https:\/\/www\.pinterest\..*?\/([a-z0-9_]{1,30})/i)) { username = match[1]; getResource('Boards', { username: username, field_set_key: 'detailed' }, start); } else { alert('Log in and visit your profile (pinterest.com/username) or board to start'); return false; } function start(json) { if (document.querySelector('#pboverlay')) return false; var overlay = document.createElement('div'); overlay.id = 'pboverlay'; overlay.innerHTML = ` <style> #pboverlay { display: block; bottom: 0; left: 0; right: 0; top: 0; z-index: 9999; position: fixed; background: rgba(0, 0, 0, 0.8); color: white; text-align: center; } #pboverlay .close { color: white; position: absolute; top:10px; right:20px; font-size: 30px; } #pboverlay .standardForm { top: 50%; margin-top: -100px; position: absolute; width: 100%; max-width: none; } #pboverlay h1 { color: white; } #pboverlay .controls a { display: inline-block; } #pboverlay select, #pboverlay meter { width: 100%; max-width: 300px; !important } #pboverlay .btn { -webkit-box-shadow: 0 1px 0 0 rgba(0, 0, 0, 0.34) !important; box-shadow: 0 1px 0 0 rgba(0, 0, 0, 0.34) !important; margin-left: 10px;} #pboverlay a { color: grey; } </style> <a href="#" class="close">×</a> <form class="standardForm"> <h1>Choose a board to export</h1> <p class="controls"> <select></select> <button class="btn"> <span>Export</span> </button> </p> <p><a href="http://pinback.github.io">Pinback v${version}</a></p> </form>`; document.querySelector('body').appendChild(overlay); var select = document.querySelector('#pboverlay select'); var option = document.createElement('option'); option.text = '[ All public pins ]'; option.value = 'all'; select.add(option); Array.prototype.forEach.call(json.resource_response.data, function(b, i) { var option = document.createElement('option'); option.text = b.name; option.value = b.id; option.selected = (location.pathname == b.url); select.add(option); }); document.querySelector('#pboverlay button').onclick = function() { document.querySelector('#pboverlay .controls').innerHTML = '<meter min="0" max="100"></meter>'; status('Exporting...'); var selected = select.querySelector('option:checked'); if (selected.value == 'all') { status('Exporting all pins...'); getResource('User', {username: username}, parseBoard); } else { status('Exporting ' + selected.text + '...'); getResource('Board', {board_id: selected.value}, parseBoard); } return false; }; document.querySelector('#pboverlay .close').onclick = function() { location.href = location.pathname; return false; }; } function parseBoard(json) { board = json.resource_response.data; getFeed(); } function getFeed(bookmarks) { if (board.type == 'user') { getResource('UserPins', { username: username, page_size: 25, bookmarks: bookmarks }, parseFeed); } else { getResource('BoardFeed', { board_id: board.id, page_size: 25, bookmarks: bookmarks }, parseFeed); } } function parseFeed(json) { json.resource_response.data.forEach(function(p, i) { if (p.type == 'pin') { if (!boards[p.board.name]) boards[p.board.name] = { id: p.board.id, name: p.board.name, url: "https://www.pinterest.com"+p.board.url, privacy: p.board.privacy, pins: [] } boards[p.board.name].pins.push({ id: p.id, link: p.link, description: p.description, url: "https://www.pinterest.com/pin/"+p.id, image: p.images.orig.url, color: p.dominant_color, longitude: (p.place && p.place.longitude) || null, latitude: (p.place && p.place.latitude) || null, pinner: p.pinner.username, privacy: p.privacy, date: Date.parse(p.created_at) }); progress(pin_count++, board.pin_count); } }) var bookmarks = json.resource.options.bookmarks; if (bookmarks[0] == '-end-') { done(); } else { getFeed(bookmarks); } } function getResource(resource, options, callback) { var xhr = new XMLHttpRequest(); xhr.onload = function () { if (xhr.status >= 200 && xhr.status < 300) { callback(JSON.parse(xhr.responseText)); } else { alert('An error has occurred.') console.log(JSON.parse(xhr.responseText)); } }; xhr.open('GET', ('/resource/' + resource + 'Resource/get/?data=' + encodeURIComponent(JSON.stringify({ options: options })))); xhr.setRequestHeader('X-Requested-With', 'XMLHttpRequest'); xhr.setRequestHeader('Accept', 'application/json, text/javascript, */*, q=0.01'); xhr.send(); } function status(s) { document.querySelector('#pboverlay h1').innerText = s; } function progress(a, b) { status('Exporting ' + a + ' of ' + b + '...'); document.querySelector('#pboverlay meter').value = (a/b)*100; } function privacy(p) { return (p != 'public') ? 1 : 0; } function escapeHtml(unsafe) { return unsafe.replace(/&/g, "&").replace(/</g, "<").replace(/>/g, ">").replace(/"/g, """).replace(/'/g, "'"); } function done() { status('Export complete!'); var data = `<!DOCTYPE NETSCAPE-Bookmark-file-1> <META HTTP-EQUIV="Content-Type" CONTENT="text/html; charset=UTF-8"> <TITLE>Bookmarks</TITLE> <H1>Bookmarks</H1> <DL><p>`; for (var name in boards) { var b = boards[name]; data += `<DT><H3 GUID="${b.id}" ORIGLINK="${b.url}" PRIVATE="${privacy(b.privacy)}">${escapeHtml(b.name)}</H3>\n<DL><p>\n`; b.pins.forEach(function(p, i) { data += `<DT><A HREF="${p.link || p.url}" GUID="${p.id}" ORIGLINK="${p.url}" IMAGE="${p.image}" COLOR="${p.color}" AUTHOR="${p.pinner}" PRIVATE="${privacy(p.privacy)}">${escapeHtml(p.description?.trim() || p.link || p.url)}</A>\n`; }); data += '</DL><p>\n'; } data += '</DL><p>'; var filename = (board.url || username).replace(/^\/|\/$/g, '').replace(/\//g,'-')+".html"; var blob = new Blob([data], { type: 'text/html' }); var url = URL.createObjectURL(blob); document.querySelector('#pboverlay .controls').innerHTML = `<a href="${url}" download="${filename}" class="btn"><span>Save export file</span></a>`; if (typeof(document.createElement('a').download) === 'undefined') { document.querySelector('#pboverlay .controls a').onclick = function() { alert('Choose File > Save As in your browser to save a copy of your export'); } } } });