// ==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.1.0
// @match https://www.vrchat.net/*
// @match https://vrchat.net/*
// @match https://www.vrchat.com/*
// @match https://vrchat.com/*
// @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',
},
/*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;
/**
* VRChatのWebページ上で一度に取得される要素数。
* @constant {number}
*/
const DEFAULT_ITEMS_COUNT = 25;
/**
* 一つのブックマークグループの最大登録数。
* @constant {number}
*/
const MAX_FAVORITES_COUNT_PER_GROUP = 32;
/**
* フレンドの各ブックマークグループの既定名。
* @constant {string[]}
*/
const DEFAULT_FRIEND_GROUP_NAMES = {
group_0: 'Group 1',
group_1: 'Group 2',
group_2: 'Group 3',
};
/**
* @type {Promise.<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)>>}
*/
function getUserDetails()
{
return userDetails || (userDetails = async function () {
const details = Object.assign( // For Greasemonkey 3
{},
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.friendGroupTagsAndNames = {};
const tags = Object.keys(DEFAULT_FRIEND_GROUP_NAMES).sort();
for (let i = 0, l = tags.length; i < l; i++) {
details.friendGroupTagsAndNames[tags[i]]
= details.friendGroupNames[i] || DEFAULT_FRIEND_GROUP_NAMES[tags[i]];
}
return details;
}());
}
/**
* @type {Promise.<Object[]>}
* @access private
*/
let userFavorites;
/**
* ユーザーのブックマークのうち、フレンドを全件取得します。
* @see [List Favorites — VRChat API Documentation]{@link https://vrchatapi.github.io/#/FavoritesAPI/ListAllFavorites}
* @returns {Promise.<(string|string[])[]>}
*/
function getUserFavorites()
{
return userFavorites || (userFavorites = 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;
}());
}
/**
* 「Edit Profile」ページに、ステータス文変更フォームを挿入します。
* @returns {Promise.<void>}
*/
async function insertUpdateStatusForm()
{
if (!('update-status' in document.forms)) {
document.getElementsByClassName('home-content')[0].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 getUserDetails()).id;
form['status-description'].value = (await getUserDetails()).statusDescription;
for (const control of form) {
control.disabled = false;
}
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;
}
});
});
}
}
/**
* ユーザーページのブックマーク登録/解除ボタンの登録数表示を更新します。
* @returns {Promise.<void>}
*/
async function updateFavoriteCounts()
{
const counts = {};
for (const favorite of (await getUserFavorites())) {
for (const tag of favorite.tags) {
if (!(tag in counts)) {
counts[tag] = 0;
}
counts[tag]++;
}
}
for (const button of document.getElementsByName('favorite-user')) {
button.getElementsByClassName('count')[0].textContent = counts[button.value] || 0;
}
}
/**
* ユーザーページにブックマーク登録/解除ボタンを追加します。
* @returns {Promise.<void>}
*/
async function insertFavoriteButtons()
{
const homeContent = document.getElementsByClassName('home-content')[0];
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 ((await getUserDetails()).friends.includes(friendUserId) && !buttons[0]) {
favoriteAndBlockButtons
.insertAdjacentHTML('afterend', '<div role="group" class="w-100 btn-group-lg btn-group-vertical mt-4">'
+ Object.keys(DEFAULT_FRIEND_GROUP_NAMES).sort().map(tag => h`<button type="button"
class="btn btn-secondary" name="favorite-user" value="${tag}" disabled="">
<span aria-hidden="true" class="fa fa-star"></span>
 <span class="name">${DEFAULT_FRIEND_GROUP_NAMES[tag]}</span>
 <span class="count">‒</span>⁄${MAX_FAVORITES_COUNT_PER_GROUP}
</button>`).join('')
+ '</div>');
for (const button of buttons) {
button.getElementsByClassName('name')[0].textContent
= (await getUserDetails()).friendGroupTagsAndNames[button.value];
}
await updateFavoriteCounts();
const tags = [].concat(...(await getUserFavorites())
.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');
}
if (button.getElementsByClassName('count')[0].textContent < MAX_FAVORITES_COUNT_PER_GROUP) {
button.disabled = false;
}
}
favoriteAndBlockButtons.nextElementSibling.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 getUserFavorites();
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);
}
await updateFavoriteCounts();
for (const button of buttons) {
if (button.getElementsByClassName('count')[0].textContent < MAX_FAVORITES_COUNT_PER_GROUP) {
button.disabled = false;
}
}
}
});
}
}
}
/**
* ページ読み込み後に一度だけ実行する処理をすでに行っていれば `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-user"] {
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);
},
});
}, [MAX_ITEMS_COUNT, DEFAULT_ITEMS_COUNT]);
}
for (const mutation of mutations) {
let parent = mutation.target;
// フレンド数を表示します
if (parent.id === 'home') {
document.getElementsByClassName('friend-container')[0].previousElementSibling.textContent
+= ` (${(await getUserDetails()).friends.length})`;
break;
}
if (parent.id === 'app' || parent.classList.contains('home-content')) {
const homeContent = document.getElementsByClassName('home-content')[0];
if (homeContent && !homeContent.getElementsByClassName('fa-cog')[0]) {
let promise;
if (location.pathname === '/home/profile') {
// 「Edit Profile」ページなら
promise = insertUpdateStatusForm();
} else if (location.pathname.startsWith('/home/user/')) {
// ユーザーページなら
promise = insertFavoriteButtons();
}
if (promise) {
promise.catch(showError);
}
}
break;
}
if (parent.classList.contains('friend-container')) {
parent = mutation.addedNodes[0];
}
if (parent.classList.contains('friend-group')) {
const friends = Array.from(parent.querySelectorAll('.friend-group > div:not(:last-of-type)'));
if (!('sorted' in friends[0].dataset)) {
// フレンド一覧を一ページの範囲内でアルファベット順に並べ替えます
friends.sort((a, b) => a.querySelector('.friend-caption li').textContent
.localeCompare(b.querySelector('.friend-caption li').textContent));
friends[0].dataset.sorted = '';
parent.lastElementChild.before(...friends);
// フレンド一覧で次のページが確実に存在しない場合、次ページのボタンを無効化します
const pager = parent.querySelector('.friend-group > div:last-of-type');
pager.getElementsByClassName('fa-chevron-down')[0]
.classList[friends.length < MAX_ITEMS_COUNT ? 'add' : 'remove']('text-muted');
// フレンド一覧のページャーボタンについて、無効化されたボタンを区別しやすくします
for (const button of pager.getElementsByTagName('button')) {
button.disabled = button.getElementsByClassName('text-muted')[0];
}
// フレンド一覧のページ切り替え後にトップにスクロールします
if (!mutation.target.classList.contains('friend-container')) {
parent.scrollIntoView({block: 'start', inline: 'start', behavior: 'smooth'});
}
}
break;
}
}
}).observe(document, {childList: true, subtree: true});