Greasy Fork

Greasy Fork is available in English.

VRChat Web Pages Extender

Adds Status Description Update form to “Edit Profile” page on VRChat Web pages and you can modify Favorite on your friend’s user pages.

当前为 2018-08-28 提交的版本,查看 最新版本

// ==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.1
// @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       dummy
// @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>&nbsp;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>&nbsp;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>&nbsp;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 });