- // ==UserScript==
- // @name VRChat Web Pages Extender
- // @name:ja VRChat Webページ拡張
- // @description Add features into VRChat Web Pages and improve user experience.
- // @description:ja VRChatのWebページに機能を追加し、また使い勝手を改善します。
- // @namespace https://greasyfork.org/users/137
- // @version 2.2.0
- // @match https://www.vrchat.net/*
- // @match https://vrchat.net/*
- // @match https://www.vrchat.com/*
- // @match https://vrchat.com/*
- // @match https://api.vrchat.cloud/*
- // @require https://greasyfork.org/scripts/17895/code/polyfill.js?version=625392
- // @require https://greasyfork.org/scripts/19616/code/utilities.js?version=230651
- // @license MPL-2.0
- // @contributionURL https://pokemori.booth.pm/items/969835
- // @compatible Edge 非推奨 / Deprecated
- // @compatible Firefox
- // @compatible Opera
- // @compatible Chrome
- // @grant dummy
- // @run-at document-start
- // @icon 
- // @author 100の人
- // @homepageURL https://pokemori.booth.pm/items/969835
- // ==/UserScript==
-
- 'use strict';
-
- // L10N
- Gettext.setLocalizedTexts({
- /*eslint-disable quote-props, max-len */
- 'en': {
- 'エラーが発生しました': 'Error occurred',
- '$LENGTH$ 文字まで表示可能です。': 'This text is displayed up to $LENGTH$ characters.',
- },
- /*eslint-enable quote-props, max-len */
- });
-
- Gettext.setLocale(navigator.language);
-
-
-
- if (typeof content !== 'undefined') {
- // For Greasemonkey 4
- fetch = content.fetch.bind(content); //eslint-disable-line no-global-assign, no-native-reassign, no-undef
- }
-
-
-
- /**
- * ページ上部にエラー内容を表示します。
- * @param {Error} exception
- * @returns {void}
- */
- function showError(exception)
- {
- console.error(exception);
- try {
- const errorMessage = _('エラーが発生しました') + ': ' + exception;
- const homeContent = document.getElementsByClassName('home-content')[0];
- if (homeContent) {
- homeContent.firstElementChild.firstElementChild.insertAdjacentHTML('afterbegin', h`<div class="row">
- <div class="alert alert-danger fade show" role="alert">${errorMessage}</div>
- </div>`);
- } else {
- alert(errorMessage);
- }
- } catch (e) {
- alert(_('エラーが発生しました') + ': ' + e);
- }
- }
-
- /**
- * 一度に取得できる最大の要素数。
- * @constant {number}
- */
- const MAX_ITEMS_COUNT = 100;
-
- /**
- * VRChatのWebページ上で一度に取得される要素数。
- * @constant {number}
- */
- const DEFAULT_ITEMS_COUNT = 25;
-
- /**
- * Status Descriptionの最大文字数。
- * @constant {number}
- */
- const MAX_STATUS_DESCRIPTION_LENGTH = 32;
-
- /**
- * 一つのブックマークグループの最大登録数。
- * @constant {number}
- */
- const MAX_FAVORITES_COUNT_PER_GROUP = 32;
-
- /**
- * 各ブックマークタグの既定名 (ユーザー設定名を取得する方法は不明)。
- * @constant {OObject.<Object.<string[]>>} {@link FAVORITE_TYPES}の要素をキーに持つ連想配列。
- */
- const DEFAULT_FAVORITE_TAG_NAMES = {
- friend: {
- group_0: 'Group 1',
- group_1: 'Group 2',
- group_2: 'Group 3',
- },
- world: {
- worlds0: 'Playlist 1',
- // worlds1: 'Playlist 2',
- worlds2: 'Playlist 2',
- worlds3: 'Playlist 3',
- worlds4: 'Playlist 4',
- },
- };
-
- /**
- * @type {?Object}
- * @access private
- */
- let userDetails;
-
- /**
- * ログインしているユーザーの情報を取得します。
- * @see [User Info — VRChat API Documentation]{@link https://vrchatapi.github.io/#/UserAPI/CurrentUserDetails}
- * @returns {Promise.<?Object.<(string|string[]|boolean|number|Object)>>} ログインしていない状態であれば、nullを返します。
- */
- async function getUserDetails()
- {
- const details = await userDetails;
- if (details) {
- return details;
- }
-
- const response = await fetch('/api/1/auth/user', { credentials: 'same-origin' });
- if (response.status === 401) {
- return null;
- }
-
- if (!response.ok) {
- return Promise.reject(new Error(`${response.status} ${response.statusText}\n${await response.text()}`));
- }
-
- userDetails = Object.assign({}, await response.json()); // For Greasemonkey 3
- return userDetails;
- }
-
- /**
- * スクリプトで扱うブックマークの種類。
- * @constant {string[]}
- */
- const FAVORITE_TYPES = ['friend', 'world'];
-
- /**
- * @type {Promise.<Object.<(string|string[])[]>[]>}
- * @access private
- */
- let favorites;
-
- /**
- * ブックマークを全件取得します。
- * @see [List Favorites — VRChat API Documentation]{@link https://vrchatapi.github.io/#/FavoritesAPI/ListAllFavorites}
- * @returns {Promise.<Object.<(string|string[])[]>[]>} {@link FAVORITE_TYPES}の要素をキーに持つ連想配列。
- */
- function getFavorites()
- {
- return favorites || (favorites = async function () {
- const allFavorites = { };
- for (const type of FAVORITE_TYPES) {
- allFavorites[type] = [];
- }
- let offset = 0;
- while (true) {
- const favorites
- = await fetch(`/api/1/favorites/?n=${MAX_ITEMS_COUNT}&offset=${offset}`, {credentials: 'same-origin'})
- .then(async response => response.ok ? response.json() : Promise.reject(
- new Error(`${response.status} ${response.statusText}\n${await response.text()}`)
- ))
- .catch(showError);
-
- for (const favorite of favorites) {
- if (!FAVORITE_TYPES.includes(favorite.type)) {
- continue;
- }
- allFavorites[favorite.type].push(favorite);
- }
-
- if (favorites.length < MAX_ITEMS_COUNT) {
- break;
- }
-
- offset += favorites.length;
- }
- return allFavorites;
- }());
- }
-
- /**
- * 「Edit Profile」ページに、ステータス文変更フォームを挿入します。
- * @returns {Promise.<void>}
- */
- async function insertUpdateStatusForm()
- {
- if ('update-status' in document.forms) {
- return;
- }
-
- const templateCard = document.getElementById('password-change-submit').closest('.card');
- const card = templateCard.cloneNode(true);
- card.getElementsByClassName('card-header')[0].textContent = 'Status';
- const form = card.getElementsByTagName('form')[0];
- form.name = 'update-status';
- form.oldPassword.closest('form > .row').remove();
-
- const statusDescription = form.newPassword;
- statusDescription.id = 'status-description';
- statusDescription.type = 'text';
- statusDescription.name = 'status-description';
- statusDescription.pattern = `.{0,${MAX_STATUS_DESCRIPTION_LENGTH}}`;
- statusDescription.title = _('$LENGTH$ 文字まで表示可能です。').replace('$LENGTH$', MAX_STATUS_DESCRIPTION_LENGTH);
- statusDescription.disabled = true;
-
- statusDescription.closest('form > .row').getElementsByClassName('fa')[0].classList.replace('fa-lock', 'fa-circle');
-
- const submit = form.getElementsByClassName('btn')[0];
- submit.id = 'status-change-submit';
- submit.textContent = 'Update Status';
- submit.disabled = true;
-
- form.addEventListener('submit', function (event) {
- event.preventDefault();
- for (const control of event.target) {
- control.disabled = true;
- }
-
- fetch(event.target.action, {
- method: 'PUT',
- headers: {'content-type': 'application/json'},
- credentials: 'same-origin',
- body: JSON.stringify({statusDescription: event.target['status-description'].value}),
- })
- .then(async function (response) {
- if (!response.ok) {
- return Promise.reject(
- new Error(`${response.status} ${response.statusText}\n${await response.text()}`)
- );
- }
- })
- .catch(showError)
- .then(function () {
- for (const control of event.target) {
- control.disabled = false;
- }
- });
- });
-
- templateCard.parentElement.getElementsByClassName('card')[0].before(card, document.createElement('hr'));
-
- form.action = '/api/1/users/' + (await getUserDetails()).id;
- statusDescription.value = (await getUserDetails()).statusDescription;
- for (const control of form) {
- control.disabled = false;
- }
- }
-
- /**
- * ブックマーク登録/解除ボタンの登録数表示を更新します。
- * @param {string} type - 「user」「favorite」のいずれか。
- * @returns {Promise.<void>}
- */
- async function updateFavoriteCounts(type)
- {
- const counts = {};
- for (const favorite of (await getFavorites())[type]) {
- for (const tag of favorite.tags) {
- if (!(tag in counts)) {
- counts[tag] = 0;
- }
- counts[tag]++;
- }
- }
-
- for (const button of document.getElementsByName('favorite-' + type)) {
- button.getElementsByClassName('count')[0].textContent = counts[button.value] || 0;
- }
- }
-
- /**
- * ブックマーク登録/解除ボタンを追加します。
- * @param {string} type - {@link FAVOLITE_TYPES}のいずれかの要素。
- * @returns {Promise.<void>}
- */
- async function insertFavoriteButtons(type)
- {
- const homeContent = document.getElementsByClassName('home-content')[0];
- const sibling = type === 'friend'
- ? homeContent.getElementsByClassName('btn-group-vertical')[0]
- : homeContent.querySelector('[href*="/home/launch"]');
- if (!sibling) {
- return;
- }
- const parent = sibling.closest('[class*="col-"]');
-
- 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);
- if (!result) {
- return;
- }
- const id = result[0];
-
- const buttons = document.getElementsByName('favorite-' + type);
- if (type === 'friend' && !(await getUserDetails()).friends.includes(id) || buttons[0]) {
- return;
- }
-
- parent.insertAdjacentHTML('beforeend', '<div role="group" class="w-100 btn-group-lg btn-group-vertical mt-4">'
- + Object.keys(DEFAULT_FAVORITE_TAG_NAMES[type]).sort().map(tag => h`<button type="button"
- class="btn btn-secondary" name="favorite-${type}" value="${tag}" disabled="">
- <span aria-hidden="true" class="fa fa-star"></span>
-  <span class="name">${DEFAULT_FAVORITE_TAG_NAMES[type][tag]}</span>
-  <span class="count">‒</span>⁄${MAX_FAVORITES_COUNT_PER_GROUP}
- </button>`).join('')
- + '</div>');
-
- await updateFavoriteCounts(type);
-
- const tags = [].concat(...(await getFavorites())[type]
- .filter(favorite => favorite.favoriteId === id)
- .map(favorite => favorite.tags));
-
- for (const button of buttons) {
- button.dataset.id = id;
- if (tags.includes(button.value)) {
- button.classList.remove('btn-secondary');
- button.classList.add('btn-primary');
- }
- if (button.getElementsByClassName('count')[0].textContent < MAX_FAVORITES_COUNT_PER_GROUP) {
- button.disabled = false;
- }
- }
-
- parent.lastElementChild.addEventListener('click', async function (event) {
- if (event.target.name !== 'favorite-' + type) {
- return;
- }
-
- const buttons = document.getElementsByName('favorite-' + type);
- for (const button of buttons) {
- button.disabled = true;
- }
-
- const id = event.target.dataset.id;
- const newTags = event.target.classList.contains('btn-secondary') ? [event.target.value] : [];
-
- const favorites = (await getFavorites())[type];
- for (let i = favorites.length - 1; i >= 0; i--) {
- if (favorites[i].favoriteId === id) {
- await fetch(
- '/api/1/favorites/' + favorites[i].id,
- {method: 'DELETE', credentials: 'same-origin'}
- );
-
- for (const button of buttons) {
- if (favorites[i].tags.includes(button.value)) {
- button.classList.remove('btn-primary');
- button.classList.add('btn-secondary');
- }
- }
-
- favorites.splice(i, 1);
- }
- }
-
- if (newTags.length > 0) {
- await fetch('/api/1/favorites', {
- method: 'POST',
- headers: { 'content-type': 'application/json' },
- credentials: 'same-origin',
- body: JSON.stringify({type, favoriteId: id, tags: newTags}),
- })
- .then(async response => response.ok ? response.json() : Promise.reject(
- new Error(`${response.status} ${response.statusText}\n${await response.text()}`)
- ))
- .then(function (favorite) {
- favorites.push(favorite);
- for (const button of buttons) {
- if (favorite.tags.includes(button.value)) {
- button.classList.remove('btn-secondary');
- button.classList.add('btn-primary');
- }
- }
- })
- .catch(showError);
- }
-
- await updateFavoriteCounts(type);
-
- for (const button of buttons) {
- if (button.getElementsByClassName('count')[0].textContent < MAX_FAVORITES_COUNT_PER_GROUP) {
- button.disabled = false;
- }
- }
- });
- }
-
- /**
- * フレンド数を表示します。
- * @returns {Promise.<void>}
- */
- async function showAllFriendCount()
- {
- const friendContainer = document.getElementsByClassName('friend-container')[0];
- if (!friendContainer) {
- return;
- }
-
- const userDetails = await getUserDetails();
- if (!userDetails) {
- return;
- }
-
- const heading = friendContainer.previousElementSibling;
- if (heading.textContent.includes('(')) {
- return;
- }
-
- friendContainer.previousElementSibling.textContent += ` (${userDetails.friends.length})`;
- }
-
- /**
- * フレンド一覧で次のページが確実に存在しない場合、次ページのボタンを無効化します。
- * また、オンラインのフレンド数を表示します。
- * @param {HTMLDivElement} group 「friend-group」クラスを持つ要素。
- * @returns {void}
- */
- function improveFriendList(group)
- {
- // フレンド一覧で次のページが確実に存在しない場合、次ページのボタンを無効化します
- const pager = group.querySelector('.friend-group > div:last-of-type');
- const count = group.querySelectorAll('.friend-group > div:not(:last-of-type)').length;
- const nextPageButton = pager.getElementsByClassName('fa-angle-down')[0].closest('button');
- if (!nextPageButton.disabled && count < MAX_ITEMS_COUNT) {
- pager.getElementsByClassName('fa-angle-down')[0].closest('button').disabled = true;
- }
-
- // オンラインのフレンド数を表示します
- const heading = group.firstElementChild;
- if (heading.textContent.includes('Online')) {
- heading.textContent = `Online (${
- MAX_ITEMS_COUNT * (/[0-9]+/.exec(pager.getElementsByClassName('page')[0].textContent)[0] - 1)
- + count
- + (nextPageButton.disabled ? '' : '+')
- })`;
- }
- }
-
- /**
- * ページ読み込み後に一度だけ実行する処理をすでに行っていれば `true`。
- * @type {boolean}
- * @access private
- */
- let headChildrenInserted = false;
-
- new MutationObserver(async function (mutations) {
- if (document.head && !headChildrenInserted) {
- headChildrenInserted = true;
- document.head.insertAdjacentHTML('beforeend', `<style>
- /*====================================
- フレンドのユーザーページ
- */
- .btn[name^="favorite-"] {
- white-space: unset;
- }
-
- /*====================================
- フレンド一覧
- */
-
- /*------------------------------------
- 1行に2列まで表示されるように
- */
- .friend-group {
- display: flex;
- flex-wrap: wrap;
- }
-
- .friend-group > :first-child,
- .friend-group > div:last-of-type {
- /* 見出し・ページャー */
- width: 100%;
- }
-
- .friend-group > div:not(:last-of-type) {
- min-width: 50%;
- }
-
- .friend-row a {
- display: flex;
- }
-
- .friend-row a .friend-img {
- margin-right: unset;
- }
- </style>`);
-
- // 一ページあたりに表示されるフレンド数を増やします
- // フレンド一覧を一ページの範囲内でアルファベット順に並べ替えます
- GreasemonkeyUtils.executeOnUnsafeContext(function (MAX_ITEMS_COUNT, DEFAULT_ITEMS_COUNT) {
- XMLHttpRequest.prototype.open = new Proxy(XMLHttpRequest.prototype.open, {
- apply(open, thisArgument, argumentList)
- {
- argumentList[1] = new URL(argumentList[1], location);
- switch (argumentList[1].pathname) {
- case '/api/1/auth/user/friends': {
- const params = argumentList[1].searchParams;
- params.set('n', MAX_ITEMS_COUNT);
- params.set('offset', params.get('offset') / DEFAULT_ITEMS_COUNT * MAX_ITEMS_COUNT);
- break;
- }
- }
- return Reflect.apply(open, thisArgument, argumentList);
- },
- });
-
- const responseText = Object.getOwnPropertyDescriptor(XMLHttpRequest.prototype, 'responseText');
- responseText.get = new Proxy(responseText.get, {
- apply(get, thisArgument, argumentList)
- {
- let responseText = Reflect.apply(get, thisArgument, argumentList);
- switch (new URL(thisArgument.responseURL).pathname) {
- case '/api/1/auth/user/friends': {
- if (thisArgument.status !== 200) {
- break;
- }
- responseText = JSON.stringify(
- JSON.parse(responseText).sort((a, b) => a.displayName.localeCompare(b.displayName))
- );
- break;
- }
- }
- return responseText;
- },
- });
- Object.defineProperty(XMLHttpRequest.prototype, 'responseText', responseText);
-
- History.prototype.pushState = new Proxy(History.prototype.pushState, {
- apply(pushState, thisArgument, argumentList)
- {
- Reflect.apply(pushState, thisArgument, argumentList);
- document.title = document.title.split(' | ').slice(-1)[0];
- },
- });
- }, [MAX_ITEMS_COUNT, DEFAULT_ITEMS_COUNT]);
-
- addEventListener('popstate', function () {
- document.title = document.title.split(' | ').slice(-1)[0];
- });
- }
-
- for (const mutation of mutations) {
- let parent = mutation.target;
- if (parent.id === 'home') {
- showAllFriendCount().catch(showError);
- break;
- }
-
- if (/* URLを開いたとき */ parent.localName === 'head' && document.body
- || /* ページを移動したとき */ parent.id === 'app' || parent.classList.contains('home-content')) {
- const homeContent = document.getElementsByClassName('home-content')[0];
- if (!homeContent || homeContent.getElementsByClassName('fa-cog')[0]) {
- break;
- }
-
- let promise;
- if (location.pathname === '/home/profile') {
- // 「Edit Profile」ページなら
- promise = insertUpdateStatusForm();
- } else if (location.pathname.startsWith('/home/user/')) {
- // ユーザーページ
- promise = insertFavoriteButtons('friend');
- if (!document.title.includes('|')) {
- const displayName = document.getElementsByTagName('h2')[0].textContent;
- const name = document.getElementsByTagName('h3')[0].firstChild.data;
- document.title = `${displayName} — ${name} | ${document.title}`;
- }
- } else if (location.pathname.startsWith('/home/world/')) {
- // ワールドページ
- promise = insertFavoriteButtons('world');
- if (!document.title.includes('|')) {
- const heading = document.getElementsByTagName('h3')[0];
- const name = heading.firstChild.data;
- const author = heading.getElementsByTagName('small')[0].textContent;
- document.title = `${name} ${author} | ${document.title}`;
- }
- }
- if (promise) {
- promise.catch(showError);
- }
- }
-
- if (parent.classList.contains('friend-container')) {
- showAllFriendCount().catch(showError);
- parent = mutation.addedNodes[0];
- }
-
- if (parent.classList.contains('friend-group')) {
- let groups = document.getElementsByClassName('friend-group');
- const heading = groups[0].firstElementChild.textContent;
- if (groups.length === 1 || heading.includes('Online') && !heading.includes('(')) {
- groups = [parent];
- }
-
- for (const group of groups) {
- improveFriendList(group);
- }
- break;
- }
- }
- }).observe(document, {childList: true, subtree: true});