您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
Greasy Fork is available in English.
Adds Status Description Update form to “Edit Profile” page on VRChat Web pages and you can modify Favorite on your friend’s user pages.
当前为
// ==UserScript== // @name VRChat Web Pages Extender // @name:ja VRChat Webページ拡張 // @description Adds Status Description Update form to “Edit Profile” page on VRChat Web pages and you can modify Favorite on your friend’s user pages. // @description:ja VRChatのWebページの「Edit Profile」へ、ステータス文の更新フォームを追加します。またフレンドのユーザーページから、Favoriteの変更ができるようにします。 // @namespace http://greasyfork.icu/users/137 // @version 2.0.0 // @match https://www.vrchat.net/* // @match https://vrchat.net/* // @match https://www.vrchat.com/* // @match https://vrchat.com/* // @require http://greasyfork.icu/scripts/17895/code/polyfill.js?version=189394 // @require http://greasyfork.icu/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 GM_xmlhttpRequest // @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', }, /*eslint-enable quote-props, max-len */ }); Gettext.setLocale(navigator.language); if (typeof content !== 'undefined') { // For Greasemonkey 4 fetch = content.fetch.bind(content); } /** * ページ上部にエラー内容を表示します。 * @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; /** * @see [User Info — VRChat API Documentation]{@link https://vrchatapi.github.io/#/UserAPI/CurrentUserDetails} * @type {Promise.<Object>} */ let userDetails; /** * @see [List Favorites — VRChat API Documentation]{@link https://vrchatapi.github.io/#/FavoritesAPI/ListAllFavorites} * @type {Promise.<Object[]>} */ let favoriteUsers; async function insertForm(homeContent) { if (location.pathname === '/home/profile') { if (!('update-status' in document.forms)) { homeContent.getElementsByTagName('h2')[0].insertAdjacentHTML('afterend', h`<div class="card row"> <h3>Update Status</h3> <div> <div class="center-panel"> <form class="form-horizontal" name="update-status"> <div class="form-group"> <div class="row"></div> <div class="row"> <div class="col-1"> <span aria-hidden="true" class="fa fa-circle fa-2x"></span> </div> <textarea class="col-md-10" name="status-description" disabled=""></textarea> </div> </div> <div class="form-group"> <div class="row"> <div class="col-4 offset-8"> <input class="btn btn-primary w-100" value="Update" type="submit" disabled="" /> </div> </div> </div> </form> </div> </div> </div>`); const form = document.forms['update-status']; form.action = '/api/1/users/' + (await userDetails).id; form['status-description'].value = (await userDetails).statusDescription; for (const control of Array.from(/* For Microsoft Edge */ form)) { control.disabled = false; } } } else if (location.pathname.startsWith('/home/user/')) { const favoriteAndBlockButtons = homeContent.getElementsByClassName('btn-group-vertical')[0]; if (favoriteAndBlockButtons) { const friendUserId = /\/user\/(usr_[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12})/ .exec(location.pathname)[1]; const buttons = document.getElementsByName('favorite-user'); if (!buttons[0] && (await userDetails).friends.includes(friendUserId)) { favoriteAndBlockButtons.insertAdjacentHTML('afterend', h` <div role="group" class="w-100 btn-group-lg btn-group-vertical mt-4"> <button type="button" class="btn btn-secondary" name="favorite-user" value="group_0" disabled=""> <span aria-hidden="true" class="fa fa-star"></span> Group 1 </button> <button type="button" class="btn btn-secondary" name="favorite-user" value="group_1" disabled=""> <span aria-hidden="true" class="fa fa-star"></span> Group 2 </button> <button type="button" class="btn btn-secondary" name="favorite-user" value="group_2" disabled=""> <span aria-hidden="true" class="fa fa-star"></span> Group 3 </button> </div> `); for (let i = 0, l = buttons.length; i < l; i++) { const label = buttons[i].childNodes[buttons[i].childNodes.length - 1]; label.data = label.data.replace('Group ' + (i + 1), (await userDetails).friendGroupNames[i]); } const tags = [].concat(...(await favoriteUsers) .filter(favorite => favorite.favoriteId === friendUserId) .map(favorite => favorite.tags)); for (const button of buttons) { button.dataset.id = friendUserId; if (tags.includes(button.value)) { button.classList.remove('btn-secondary'); button.classList.add('btn-primary'); } button.disabled = false; } } } } } new MutationObserver(function (mutations, observer) { observer.disconnect(); userDetails = async function () { const details = await fetch('/api/1/auth/user', {credentials: 'same-origin'}) .then(async response => response.ok ? response.json() : Promise.reject(new Error(`${response.status} ${response.statusText}\n${await response.text()}`))) .catch(showError); details.friendGroupNames.push(...['Group 1', 'Group 2', 'Group 3'].slice(details.friendGroupNames.length)); return details; }(); favoriteUsers = async function () { const users = []; 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); users.push(...favorites.filter(favorite => favorite.type === 'friend')); if (favorites.length < MAX_ITEMS_COUNT) { break; } offset++; } return users; }(); const homeContent = document.getElementsByClassName('home-content')[0]; insertForm(homeContent); new MutationObserver(function () { insertForm(homeContent).catch(showError); }).observe(homeContent, { childList: true }); homeContent.addEventListener('submit', function (event) { if (event.target.name === 'update-status') { event.preventDefault(); for (const control of Array.from(/* For Microsoft Edge */ 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 Array.from(/* For Microsoft Edge */ event.target)) { control.disabled = false; } }); } }); homeContent.addEventListener('click', async function (event) { if (event.target.name === 'favorite-user') { const buttons = document.getElementsByName('favorite-user'); for (const button of buttons) { button.disabled = true; } const friendUserId = event.target.dataset.id; const newTags = event.target.classList.contains('btn-secondary') ? [event.target.value] : []; const favorites = await favoriteUsers; for (let i = favorites.length - 1; i >= 0; i--) { if (favorites[i].favoriteId === friendUserId) { 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: 'friend', favoriteId: friendUserId, 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); } for (const button of buttons) { button.disabled = false; } } }); }).observe(document.getElementById('app'), { childList: true });