VRChat Web Pages Extender

Add features into VRChat Web Pages and improve user experience.

当前为 2018-09-01 提交的版本,查看 最新版本

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Greasemonkey 油猴子Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Userscripts ,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展后才能安装此脚本。

(我已经安装了用户脚本管理器,让我安装!)

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

(我已经安装了用户样式管理器,让我安装!)

// ==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>
						&#xA0;<span class="name">${DEFAULT_FRIEND_GROUP_NAMES[tag]}</span>
						&#xA0;<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});