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   https://gf.qytechs.cn/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     https://gf.qytechs.cn/scripts/17895/code/polyfill.js?version=189394
// @require     https://gf.qytechs.cn/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 });

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址