VRChat Web Pages Extender

Add features into VRChat Web Pages and improve user experience.

目前為 2019-06-02 提交的版本,檢視 最新版本

  1. // ==UserScript==
  2. // @name VRChat Web Pages Extender
  3. // @name:ja VRChat Webページ拡張
  4. // @description Add features into VRChat Web Pages and improve user experience.
  5. // @description:ja VRChatのWebページに機能を追加し、また使い勝手を改善します。
  6. // @namespace https://greasyfork.org/users/137
  7. // @version 2.2.0
  8. // @match https://www.vrchat.net/*
  9. // @match https://vrchat.net/*
  10. // @match https://www.vrchat.com/*
  11. // @match https://vrchat.com/*
  12. // @match https://api.vrchat.cloud/*
  13. // @require https://greasyfork.org/scripts/17895/code/polyfill.js?version=625392
  14. // @require https://greasyfork.org/scripts/19616/code/utilities.js?version=230651
  15. // @license MPL-2.0
  16. // @contributionURL https://pokemori.booth.pm/items/969835
  17. // @compatible Edge 非推奨 / Deprecated
  18. // @compatible Firefox
  19. // @compatible Opera
  20. // @compatible Chrome
  21. // @grant dummy
  22. // @run-at document-start
  23. // @icon 
  24. // @author 100の人
  25. // @homepageURL https://pokemori.booth.pm/items/969835
  26. // ==/UserScript==
  27.  
  28. 'use strict';
  29.  
  30. // L10N
  31. Gettext.setLocalizedTexts({
  32. /*eslint-disable quote-props, max-len */
  33. 'en': {
  34. 'エラーが発生しました': 'Error occurred',
  35. '$LENGTH$ 文字まで表示可能です。': 'This text is displayed up to $LENGTH$ characters.',
  36. },
  37. /*eslint-enable quote-props, max-len */
  38. });
  39.  
  40. Gettext.setLocale(navigator.language);
  41.  
  42.  
  43.  
  44. if (typeof content !== 'undefined') {
  45. // For Greasemonkey 4
  46. fetch = content.fetch.bind(content); //eslint-disable-line no-global-assign, no-native-reassign, no-undef
  47. }
  48.  
  49.  
  50.  
  51. /**
  52. * ページ上部にエラー内容を表示します。
  53. * @param {Error} exception
  54. * @returns {void}
  55. */
  56. function showError(exception)
  57. {
  58. console.error(exception);
  59. try {
  60. const errorMessage = _('エラーが発生しました') + ': ' + exception;
  61. const homeContent = document.getElementsByClassName('home-content')[0];
  62. if (homeContent) {
  63. homeContent.firstElementChild.firstElementChild.insertAdjacentHTML('afterbegin', h`<div class="row">
  64. <div class="alert alert-danger fade show" role="alert">${errorMessage}</div>
  65. </div>`);
  66. } else {
  67. alert(errorMessage);
  68. }
  69. } catch (e) {
  70. alert(_('エラーが発生しました') + ': ' + e);
  71. }
  72. }
  73.  
  74. /**
  75. * 一度に取得できる最大の要素数。
  76. * @constant {number}
  77. */
  78. const MAX_ITEMS_COUNT = 100;
  79.  
  80. /**
  81. * VRChatのWebページ上で一度に取得される要素数。
  82. * @constant {number}
  83. */
  84. const DEFAULT_ITEMS_COUNT = 25;
  85.  
  86. /**
  87. * Status Descriptionの最大文字数。
  88. * @constant {number}
  89. */
  90. const MAX_STATUS_DESCRIPTION_LENGTH = 32;
  91.  
  92. /**
  93. * 一つのブックマークグループの最大登録数。
  94. * @constant {number}
  95. */
  96. const MAX_FAVORITES_COUNT_PER_GROUP = 32;
  97.  
  98. /**
  99. * 各ブックマークタグの既定名 (ユーザー設定名を取得する方法は不明)。
  100. * @constant {OObject.<Object.<string[]>>} {@link FAVORITE_TYPES}の要素をキーに持つ連想配列。
  101. */
  102. const DEFAULT_FAVORITE_TAG_NAMES = {
  103. friend: {
  104. group_0: 'Group 1',
  105. group_1: 'Group 2',
  106. group_2: 'Group 3',
  107. },
  108. world: {
  109. worlds0: 'Playlist 1',
  110. // worlds1: 'Playlist 2',
  111. worlds2: 'Playlist 2',
  112. worlds3: 'Playlist 3',
  113. worlds4: 'Playlist 4',
  114. },
  115. };
  116.  
  117. /**
  118. * @type {?Object}
  119. * @access private
  120. */
  121. let userDetails;
  122.  
  123. /**
  124. * ログインしているユーザーの情報を取得します。
  125. * @see [User Info — VRChat API Documentation]{@link https://vrchatapi.github.io/#/UserAPI/CurrentUserDetails}
  126. * @returns {Promise.<?Object.<(string|string[]|boolean|number|Object)>>} ログインしていない状態であれば、nullを返します。
  127. */
  128. async function getUserDetails()
  129. {
  130. const details = await userDetails;
  131. if (details) {
  132. return details;
  133. }
  134.  
  135. const response = await fetch('/api/1/auth/user', { credentials: 'same-origin' });
  136. if (response.status === 401) {
  137. return null;
  138. }
  139.  
  140. if (!response.ok) {
  141. return Promise.reject(new Error(`${response.status} ${response.statusText}\n${await response.text()}`));
  142. }
  143.  
  144. userDetails = Object.assign({}, await response.json()); // For Greasemonkey 3
  145. return userDetails;
  146. }
  147.  
  148. /**
  149. * スクリプトで扱うブックマークの種類。
  150. * @constant {string[]}
  151. */
  152. const FAVORITE_TYPES = ['friend', 'world'];
  153.  
  154. /**
  155. * @type {Promise.<Object.<(string|string[])[]>[]>}
  156. * @access private
  157. */
  158. let favorites;
  159.  
  160. /**
  161. * ブックマークを全件取得します。
  162. * @see [List Favorites — VRChat API Documentation]{@link https://vrchatapi.github.io/#/FavoritesAPI/ListAllFavorites}
  163. * @returns {Promise.<Object.<(string|string[])[]>[]>} {@link FAVORITE_TYPES}の要素をキーに持つ連想配列。
  164. */
  165. function getFavorites()
  166. {
  167. return favorites || (favorites = async function () {
  168. const allFavorites = { };
  169. for (const type of FAVORITE_TYPES) {
  170. allFavorites[type] = [];
  171. }
  172. let offset = 0;
  173. while (true) {
  174. const favorites
  175. = await fetch(`/api/1/favorites/?n=${MAX_ITEMS_COUNT}&offset=${offset}`, {credentials: 'same-origin'})
  176. .then(async response => response.ok ? response.json() : Promise.reject(
  177. new Error(`${response.status} ${response.statusText}\n${await response.text()}`)
  178. ))
  179. .catch(showError);
  180.  
  181. for (const favorite of favorites) {
  182. if (!FAVORITE_TYPES.includes(favorite.type)) {
  183. continue;
  184. }
  185. allFavorites[favorite.type].push(favorite);
  186. }
  187.  
  188. if (favorites.length < MAX_ITEMS_COUNT) {
  189. break;
  190. }
  191.  
  192. offset += favorites.length;
  193. }
  194. return allFavorites;
  195. }());
  196. }
  197.  
  198. /**
  199. * 「Edit Profile」ページに、ステータス文変更フォームを挿入します。
  200. * @returns {Promise.<void>}
  201. */
  202. async function insertUpdateStatusForm()
  203. {
  204. if ('update-status' in document.forms) {
  205. return;
  206. }
  207.  
  208. const templateCard = document.getElementById('password-change-submit').closest('.card');
  209. const card = templateCard.cloneNode(true);
  210. card.getElementsByClassName('card-header')[0].textContent = 'Status';
  211. const form = card.getElementsByTagName('form')[0];
  212. form.name = 'update-status';
  213. form.oldPassword.closest('form > .row').remove();
  214.  
  215. const statusDescription = form.newPassword;
  216. statusDescription.id = 'status-description';
  217. statusDescription.type = 'text';
  218. statusDescription.name = 'status-description';
  219. statusDescription.pattern = `.{0,${MAX_STATUS_DESCRIPTION_LENGTH}}`;
  220. statusDescription.title = _('$LENGTH$ 文字まで表示可能です。').replace('$LENGTH$', MAX_STATUS_DESCRIPTION_LENGTH);
  221. statusDescription.disabled = true;
  222.  
  223. statusDescription.closest('form > .row').getElementsByClassName('fa')[0].classList.replace('fa-lock', 'fa-circle');
  224.  
  225. const submit = form.getElementsByClassName('btn')[0];
  226. submit.id = 'status-change-submit';
  227. submit.textContent = 'Update Status';
  228. submit.disabled = true;
  229. form.addEventListener('submit', function (event) {
  230. event.preventDefault();
  231. for (const control of event.target) {
  232. control.disabled = true;
  233. }
  234.  
  235. fetch(event.target.action, {
  236. method: 'PUT',
  237. headers: {'content-type': 'application/json'},
  238. credentials: 'same-origin',
  239. body: JSON.stringify({statusDescription: event.target['status-description'].value}),
  240. })
  241. .then(async function (response) {
  242. if (!response.ok) {
  243. return Promise.reject(
  244. new Error(`${response.status} ${response.statusText}\n${await response.text()}`)
  245. );
  246. }
  247. })
  248. .catch(showError)
  249. .then(function () {
  250. for (const control of event.target) {
  251. control.disabled = false;
  252. }
  253. });
  254. });
  255.  
  256. templateCard.parentElement.getElementsByClassName('card')[0].before(card, document.createElement('hr'));
  257.  
  258. form.action = '/api/1/users/' + (await getUserDetails()).id;
  259. statusDescription.value = (await getUserDetails()).statusDescription;
  260. for (const control of form) {
  261. control.disabled = false;
  262. }
  263. }
  264.  
  265. /**
  266. * ブックマーク登録/解除ボタンの登録数表示を更新します。
  267. * @param {string} type - 「user」「favorite」のいずれか。
  268. * @returns {Promise.<void>}
  269. */
  270. async function updateFavoriteCounts(type)
  271. {
  272. const counts = {};
  273. for (const favorite of (await getFavorites())[type]) {
  274. for (const tag of favorite.tags) {
  275. if (!(tag in counts)) {
  276. counts[tag] = 0;
  277. }
  278. counts[tag]++;
  279. }
  280. }
  281.  
  282. for (const button of document.getElementsByName('favorite-' + type)) {
  283. button.getElementsByClassName('count')[0].textContent = counts[button.value] || 0;
  284. }
  285. }
  286.  
  287. /**
  288. * ブックマーク登録/解除ボタンを追加します。
  289. * @param {string} type - {@link FAVOLITE_TYPES}のいずれかの要素。
  290. * @returns {Promise.<void>}
  291. */
  292. async function insertFavoriteButtons(type)
  293. {
  294. const homeContent = document.getElementsByClassName('home-content')[0];
  295. const sibling = type === 'friend'
  296. ? homeContent.getElementsByClassName('btn-group-vertical')[0]
  297. : homeContent.querySelector('[href*="/home/launch"]');
  298. if (!sibling) {
  299. return;
  300. }
  301. const parent = sibling.closest('[class*="col-"]');
  302.  
  303. const result = /[a-z]+_[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12}/.exec(location.pathname);
  304. if (!result) {
  305. return;
  306. }
  307. const id = result[0];
  308.  
  309. const buttons = document.getElementsByName('favorite-' + type);
  310. if (type === 'friend' && !(await getUserDetails()).friends.includes(id) || buttons[0]) {
  311. return;
  312. }
  313.  
  314. parent.insertAdjacentHTML('beforeend', '<div role="group" class="w-100 btn-group-lg btn-group-vertical mt-4">'
  315. + Object.keys(DEFAULT_FAVORITE_TAG_NAMES[type]).sort().map(tag => h`<button type="button"
  316. class="btn btn-secondary" name="favorite-${type}" value="${tag}" disabled="">
  317. <span aria-hidden="true" class="fa fa-star"></span>
  318. &#xA0;<span class="name">${DEFAULT_FAVORITE_TAG_NAMES[type][tag]}</span>
  319. &#xA0;<span class="count">‒</span>⁄${MAX_FAVORITES_COUNT_PER_GROUP}
  320. </button>`).join('')
  321. + '</div>');
  322.  
  323. await updateFavoriteCounts(type);
  324.  
  325. const tags = [].concat(...(await getFavorites())[type]
  326. .filter(favorite => favorite.favoriteId === id)
  327. .map(favorite => favorite.tags));
  328.  
  329. for (const button of buttons) {
  330. button.dataset.id = id;
  331. if (tags.includes(button.value)) {
  332. button.classList.remove('btn-secondary');
  333. button.classList.add('btn-primary');
  334. }
  335. if (button.getElementsByClassName('count')[0].textContent < MAX_FAVORITES_COUNT_PER_GROUP) {
  336. button.disabled = false;
  337. }
  338. }
  339.  
  340. parent.lastElementChild.addEventListener('click', async function (event) {
  341. if (event.target.name !== 'favorite-' + type) {
  342. return;
  343. }
  344.  
  345. const buttons = document.getElementsByName('favorite-' + type);
  346. for (const button of buttons) {
  347. button.disabled = true;
  348. }
  349.  
  350. const id = event.target.dataset.id;
  351. const newTags = event.target.classList.contains('btn-secondary') ? [event.target.value] : [];
  352.  
  353. const favorites = (await getFavorites())[type];
  354. for (let i = favorites.length - 1; i >= 0; i--) {
  355. if (favorites[i].favoriteId === id) {
  356. await fetch(
  357. '/api/1/favorites/' + favorites[i].id,
  358. {method: 'DELETE', credentials: 'same-origin'}
  359. );
  360.  
  361. for (const button of buttons) {
  362. if (favorites[i].tags.includes(button.value)) {
  363. button.classList.remove('btn-primary');
  364. button.classList.add('btn-secondary');
  365. }
  366. }
  367.  
  368. favorites.splice(i, 1);
  369. }
  370. }
  371.  
  372. if (newTags.length > 0) {
  373. await fetch('/api/1/favorites', {
  374. method: 'POST',
  375. headers: { 'content-type': 'application/json' },
  376. credentials: 'same-origin',
  377. body: JSON.stringify({type, favoriteId: id, tags: newTags}),
  378. })
  379. .then(async response => response.ok ? response.json() : Promise.reject(
  380. new Error(`${response.status} ${response.statusText}\n${await response.text()}`)
  381. ))
  382. .then(function (favorite) {
  383. favorites.push(favorite);
  384. for (const button of buttons) {
  385. if (favorite.tags.includes(button.value)) {
  386. button.classList.remove('btn-secondary');
  387. button.classList.add('btn-primary');
  388. }
  389. }
  390. })
  391. .catch(showError);
  392. }
  393.  
  394. await updateFavoriteCounts(type);
  395.  
  396. for (const button of buttons) {
  397. if (button.getElementsByClassName('count')[0].textContent < MAX_FAVORITES_COUNT_PER_GROUP) {
  398. button.disabled = false;
  399. }
  400. }
  401. });
  402. }
  403.  
  404. /**
  405. * フレンド数を表示します。
  406. * @returns {Promise.<void>}
  407. */
  408. async function showAllFriendCount()
  409. {
  410. const friendContainer = document.getElementsByClassName('friend-container')[0];
  411. if (!friendContainer) {
  412. return;
  413. }
  414.  
  415. const userDetails = await getUserDetails();
  416. if (!userDetails) {
  417. return;
  418. }
  419.  
  420. const heading = friendContainer.previousElementSibling;
  421. if (heading.textContent.includes('(')) {
  422. return;
  423. }
  424.  
  425. friendContainer.previousElementSibling.textContent += ` (${userDetails.friends.length})`;
  426. }
  427.  
  428. /**
  429. * フレンド一覧で次のページが確実に存在しない場合、次ページのボタンを無効化します。
  430. * また、オンラインのフレンド数を表示します。
  431. * @param {HTMLDivElement} group 「friend-group」クラスを持つ要素。
  432. * @returns {void}
  433. */
  434. function improveFriendList(group)
  435. {
  436. // フレンド一覧で次のページが確実に存在しない場合、次ページのボタンを無効化します
  437. const pager = group.querySelector('.friend-group > div:last-of-type');
  438. const count = group.querySelectorAll('.friend-group > div:not(:last-of-type)').length;
  439. const nextPageButton = pager.getElementsByClassName('fa-angle-down')[0].closest('button');
  440. if (!nextPageButton.disabled && count < MAX_ITEMS_COUNT) {
  441. pager.getElementsByClassName('fa-angle-down')[0].closest('button').disabled = true;
  442. }
  443.  
  444. // オンラインのフレンド数を表示します
  445. const heading = group.firstElementChild;
  446. if (heading.textContent.includes('Online')) {
  447. heading.textContent = `Online (${
  448. MAX_ITEMS_COUNT * (/[0-9]+/.exec(pager.getElementsByClassName('page')[0].textContent)[0] - 1)
  449. + count
  450. + (nextPageButton.disabled ? '' : '+')
  451. })`;
  452. }
  453. }
  454.  
  455. /**
  456. * ページ読み込み後に一度だけ実行する処理をすでに行っていれば `true`。
  457. * @type {boolean}
  458. * @access private
  459. */
  460. let headChildrenInserted = false;
  461.  
  462. new MutationObserver(async function (mutations) {
  463. if (document.head && !headChildrenInserted) {
  464. headChildrenInserted = true;
  465. document.head.insertAdjacentHTML('beforeend', `<style>
  466. /*====================================
  467. フレンドのユーザーページ
  468. */
  469. .btn[name^="favorite-"] {
  470. white-space: unset;
  471. }
  472.  
  473. /*====================================
  474. フレンド一覧
  475. */
  476.  
  477. /*------------------------------------
  478. 1行に2列まで表示されるように
  479. */
  480. .friend-group {
  481. display: flex;
  482. flex-wrap: wrap;
  483. }
  484.  
  485. .friend-group > :first-child,
  486. .friend-group > div:last-of-type {
  487. /* 見出し・ページャー */
  488. width: 100%;
  489. }
  490.  
  491. .friend-group > div:not(:last-of-type) {
  492. min-width: 50%;
  493. }
  494.  
  495. .friend-row a {
  496. display: flex;
  497. }
  498.  
  499. .friend-row a .friend-img {
  500. margin-right: unset;
  501. }
  502. </style>`);
  503.  
  504. // 一ページあたりに表示されるフレンド数を増やします
  505. // フレンド一覧を一ページの範囲内でアルファベット順に並べ替えます
  506. GreasemonkeyUtils.executeOnUnsafeContext(function (MAX_ITEMS_COUNT, DEFAULT_ITEMS_COUNT) {
  507. XMLHttpRequest.prototype.open = new Proxy(XMLHttpRequest.prototype.open, {
  508. apply(open, thisArgument, argumentList)
  509. {
  510. argumentList[1] = new URL(argumentList[1], location);
  511. switch (argumentList[1].pathname) {
  512. case '/api/1/auth/user/friends': {
  513. const params = argumentList[1].searchParams;
  514. params.set('n', MAX_ITEMS_COUNT);
  515. params.set('offset', params.get('offset') / DEFAULT_ITEMS_COUNT * MAX_ITEMS_COUNT);
  516. break;
  517. }
  518. }
  519. return Reflect.apply(open, thisArgument, argumentList);
  520. },
  521. });
  522.  
  523. const responseText = Object.getOwnPropertyDescriptor(XMLHttpRequest.prototype, 'responseText');
  524. responseText.get = new Proxy(responseText.get, {
  525. apply(get, thisArgument, argumentList)
  526. {
  527. let responseText = Reflect.apply(get, thisArgument, argumentList);
  528. switch (new URL(thisArgument.responseURL).pathname) {
  529. case '/api/1/auth/user/friends': {
  530. if (thisArgument.status !== 200) {
  531. break;
  532. }
  533. responseText = JSON.stringify(
  534. JSON.parse(responseText).sort((a, b) => a.displayName.localeCompare(b.displayName))
  535. );
  536. break;
  537. }
  538. }
  539. return responseText;
  540. },
  541. });
  542. Object.defineProperty(XMLHttpRequest.prototype, 'responseText', responseText);
  543.  
  544. History.prototype.pushState = new Proxy(History.prototype.pushState, {
  545. apply(pushState, thisArgument, argumentList)
  546. {
  547. Reflect.apply(pushState, thisArgument, argumentList);
  548. document.title = document.title.split(' | ').slice(-1)[0];
  549. },
  550. });
  551. }, [MAX_ITEMS_COUNT, DEFAULT_ITEMS_COUNT]);
  552.  
  553. addEventListener('popstate', function () {
  554. document.title = document.title.split(' | ').slice(-1)[0];
  555. });
  556. }
  557.  
  558. for (const mutation of mutations) {
  559. let parent = mutation.target;
  560. if (parent.id === 'home') {
  561. showAllFriendCount().catch(showError);
  562. break;
  563. }
  564.  
  565. if (/* URLを開いたとき */ parent.localName === 'head' && document.body
  566. || /* ページを移動したとき */ parent.id === 'app' || parent.classList.contains('home-content')) {
  567. const homeContent = document.getElementsByClassName('home-content')[0];
  568. if (!homeContent || homeContent.getElementsByClassName('fa-cog')[0]) {
  569. break;
  570. }
  571.  
  572. let promise;
  573. if (location.pathname === '/home/profile') {
  574. // 「Edit Profile」ページなら
  575. promise = insertUpdateStatusForm();
  576. } else if (location.pathname.startsWith('/home/user/')) {
  577. // ユーザーページ
  578. promise = insertFavoriteButtons('friend');
  579. if (!document.title.includes('|')) {
  580. const displayName = document.getElementsByTagName('h2')[0].textContent;
  581. const name = document.getElementsByTagName('h3')[0].firstChild.data;
  582. document.title = `${displayName} ${name} | ${document.title}`;
  583. }
  584. } else if (location.pathname.startsWith('/home/world/')) {
  585. // ワールドページ
  586. promise = insertFavoriteButtons('world');
  587. if (!document.title.includes('|')) {
  588. const heading = document.getElementsByTagName('h3')[0];
  589. const name = heading.firstChild.data;
  590. const author = heading.getElementsByTagName('small')[0].textContent;
  591. document.title = `${name} ${author} | ${document.title}`;
  592. }
  593. }
  594. if (promise) {
  595. promise.catch(showError);
  596. }
  597. }
  598.  
  599. if (parent.classList.contains('friend-container')) {
  600. showAllFriendCount().catch(showError);
  601. parent = mutation.addedNodes[0];
  602. }
  603.  
  604. if (parent.classList.contains('friend-group')) {
  605. let groups = document.getElementsByClassName('friend-group');
  606. const heading = groups[0].firstElementChild.textContent;
  607. if (groups.length === 1 || heading.includes('Online') && !heading.includes('(')) {
  608. groups = [parent];
  609. }
  610.  
  611. for (const group of groups) {
  612. improveFriendList(group);
  613. }
  614. break;
  615. }
  616. }
  617. }).observe(document, {childList: true, subtree: true});