Pixiv Bookmark async

Add a bookmark in Pixiv on background using Fetch API.

目前为 2017-09-21 提交的版本。查看 最新版本

  1. /*
  2. // ==UserScript==
  3. // @namespace com.whoopsworkshop.hkgsherlock.userjs.pixiv_bookmark_async
  4. // @id com.whoopsworkshop.hkgsherlock.userjs.pixiv_bookmark_async
  5. // @icon 
  6. // @name Pixiv Bookmark async
  7. // @version 201709211745+8
  8. // @author hkgsherlock
  9. // @license Unlicense
  10. // @description Add a bookmark in Pixiv on background using Fetch API.
  11. // @include /^https?:\/\/www\.pixiv\.net\/member_illust\.php?.*mode=(medium|manga).*$/
  12. // @grant none
  13. // ==/UserScript==
  14. */
  15.  
  16. const PIXIV_API_URL = 'https://www.pixiv.net/rpc/index.php';
  17. const BUTTON_TARGET_QUERY = '.bookmark-container .add-bookmark';
  18.  
  19. (() => {
  20. const target = document.querySelector(BUTTON_TARGET_QUERY);
  21.  
  22. target.addEventListener('click', (e) => {
  23. e.preventDefault();
  24.  
  25. const queryString = document.location.search.substr(1)
  26. .split('&')
  27. .map((e) => {
  28. const [key, value] = e.split('=');
  29. return {[key]: value};
  30. })
  31. .reduce((a,b)=>Object.assign(a,b),{});
  32. // https://google.com/search?q=pixiv+context+token
  33. const pixiv_context_token = window.pixiv.context.token; // not ranbu sry (; _ ;) i miss my manba-chan dayo~
  34. const illust_id = queryString.illust_id;
  35. const restricted = !!document.querySelector('.work-info .meta [class^="r-18"]');
  36. const comment = null;
  37. const tags = [...document.querySelectorAll('.tags .tag .text')].map((e) => e.innerText).join('+');
  38.  
  39. fetch(PIXIV_API_URL, {
  40. method: "POST",
  41. headers: {
  42. "Content-Type": "application/x-www-form-urlencoded; charset=UTF-8"
  43. },
  44. credentials: "same-origin",
  45. body: [
  46. 'mode=save_illust_bookmark',
  47. `illust_id=${illust_id}`,
  48. `restrict=${!!restricted?1:0}`,
  49. 'comment'+(!!comment?`=${comment}`:''),
  50. 'tags'+(!!tags?`=${tags}`:''),
  51. `tt=${pixiv_context_token}`,
  52. ].join('&'),
  53. })
  54. .then((r) => r.json())
  55. .then((data) => {
  56. if (data.error) throw new Error(message || body.message);
  57. document.querySelector(BUTTON_TARGET_QUERY)
  58. .replaceWith(((id) => {
  59. const a = document.createElement('a');
  60. a.setAttribute('href', `bookmark_add.php?type=illust&illust_id=${id}`);
  61. a.classList.add('_bookmark-toggle-button');
  62. a.classList.add('bookmarked');
  63. a.classList.add('edit-bookmark');
  64. a.appendChild((() => {
  65. const span = document.createElement('span');
  66. span.classList.add('description');
  67. span.innerText = "ブックマークを編集";
  68. return span;
  69. })());
  70. a.appendChild((() => {
  71. const span = document.createElement('span');
  72. span.setAttribute('style', 'animation:nice-zoom-smile 1s forwards cubic-bezier(0.9, -1.5, 0.4, 1);'
  73. +'position:absolute;'
  74. +'top:50%;'
  75. +'left:50%;'
  76. +'width:32px;'
  77. +'height:32px;'
  78. +'margin:-16px 0 0 -16px;'
  79. +'background-image: url();'
  80. +'background-position:center;'
  81. +'background-repeat:no-repeat;'
  82. +'background-size:cover;'
  83. +'visibility:hidden;');
  84. return span;
  85. })());
  86. return a;
  87. })(illust_id));
  88. })
  89. .catch((err) => console.error(err));
  90. });
  91.  
  92. target.querySelector('.description').textContent += '*';
  93. })();