Merge branch 'develop' of misskey-dev into merge-upstream
This commit is contained in:
		
							
								
								
									
										
											BIN
										
									
								
								packages/frontend/assets/drop-and-fusion/bgm_1.mp3
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								packages/frontend/assets/drop-and-fusion/bgm_1.mp3
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								packages/frontend/assets/drop-and-fusion/bubble2.mp3
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								packages/frontend/assets/drop-and-fusion/bubble2.mp3
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								packages/frontend/assets/drop-and-fusion/click.mp3
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								packages/frontend/assets/drop-and-fusion/click.mp3
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								packages/frontend/assets/drop-and-fusion/gameover.mp3
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								packages/frontend/assets/drop-and-fusion/gameover.mp3
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								packages/frontend/assets/drop-and-fusion/hold.mp3
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								packages/frontend/assets/drop-and-fusion/hold.mp3
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								packages/frontend/assets/drop-and-fusion/poi1.mp3
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								packages/frontend/assets/drop-and-fusion/poi1.mp3
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								packages/frontend/assets/drop-and-fusion/poi2.mp3
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								packages/frontend/assets/drop-and-fusion/poi2.mp3
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| @@ -4,7 +4,7 @@ | ||||
| 	"type": "module", | ||||
| 	"scripts": { | ||||
| 		"watch": "vite", | ||||
| 		"dev": "vite --config vite.config.local-dev.ts", | ||||
| 		"dev": "vite --config vite.config.local-dev.ts --debug hmr", | ||||
| 		"build": "vite build", | ||||
| 		"storybook-dev": "nodemon --verbose --watch src --ext \"mdx,ts,vue\" --ignore \"*.stories.ts\" --exec \"pnpm build-storybook-pre && pnpm exec storybook dev -p 6006 --ci\"", | ||||
| 		"build-storybook-pre": "(tsc -p .storybook || echo done.) && node .storybook/generate.js && node .storybook/preload-locale.js && node .storybook/preload-theme.js", | ||||
| @@ -19,6 +19,8 @@ | ||||
| 	"dependencies": { | ||||
| 		"@discordapp/twemoji": "15.0.2", | ||||
| 		"@github/webauthn-json": "2.1.1", | ||||
| 		"@mcaptcha/vanilla-glue": "0.1.0-alpha-3", | ||||
| 		"@misskey-dev/browser-image-resizer": "2.2.1-misskey.10", | ||||
| 		"@rollup/plugin-json": "6.1.0", | ||||
| 		"@rollup/plugin-replace": "5.0.5", | ||||
| 		"@rollup/plugin-typescript": "11.1.5", | ||||
| @@ -26,12 +28,11 @@ | ||||
| 		"@syuilo/aiscript": "0.16.0", | ||||
| 		"@tabler/icons-webfont": "2.44.0", | ||||
| 		"@twemoji/parser": "15.0.0", | ||||
| 		"@vitejs/plugin-vue": "4.5.2", | ||||
| 		"@vue/compiler-sfc": "3.3.12", | ||||
| 		"@vitejs/plugin-vue": "5.0.2", | ||||
| 		"@vue/compiler-sfc": "3.4.3", | ||||
| 		"aiscript-vscode": "github:aiscript-dev/aiscript-vscode#v0.0.6", | ||||
| 		"astring": "1.8.6", | ||||
| 		"broadcast-channel": "7.0.0", | ||||
| 		"browser-image-resizer": "github:misskey-dev/browser-image-resizer#v2.2.1-misskey.3", | ||||
| 		"buraha": "0.0.1", | ||||
| 		"canvas-confetti": "1.6.1", | ||||
| 		"chart.js": "4.4.1", | ||||
| @@ -46,7 +47,6 @@ | ||||
| 		"escape-regexp": "0.0.1", | ||||
| 		"estree-walker": "3.0.3", | ||||
| 		"eventemitter3": "5.0.1", | ||||
| 		"gsap": "3.12.4", | ||||
| 		"idb-keyval": "6.2.1", | ||||
| 		"insert-text-at-cursor": "0.3.0", | ||||
| 		"is-file-animated": "1.0.2", | ||||
| @@ -59,6 +59,7 @@ | ||||
| 		"rollup": "4.9.1", | ||||
| 		"sanitize-html": "2.11.0", | ||||
| 		"sass": "1.69.5", | ||||
| 		"seedrandom": "^3.0.5", | ||||
| 		"shiki": "0.14.7", | ||||
| 		"strict-event-emitter-types": "2.0.0", | ||||
| 		"textarea-caret": "3.1.0", | ||||
| @@ -71,10 +72,12 @@ | ||||
| 		"uuid": "9.0.1", | ||||
| 		"v-code-diff": "1.7.2", | ||||
| 		"vite": "5.0.10", | ||||
| 		"vue": "3.3.12", | ||||
| 		"vue": "3.4.3", | ||||
| 		"vuedraggable": "next" | ||||
| 	}, | ||||
| 	"devDependencies": { | ||||
| 		"@misskey-dev/eslint-plugin": "^1.0.0", | ||||
| 		"@misskey-dev/summaly": "^5.0.3", | ||||
| 		"@storybook/addon-actions": "7.6.5", | ||||
| 		"@storybook/addon-essentials": "7.6.5", | ||||
| 		"@storybook/addon-interactions": "7.6.5", | ||||
| @@ -108,7 +111,7 @@ | ||||
| 		"@typescript-eslint/eslint-plugin": "6.14.0", | ||||
| 		"@typescript-eslint/parser": "6.14.0", | ||||
| 		"@vitest/coverage-v8": "0.34.6", | ||||
| 		"@vue/runtime-core": "3.3.12", | ||||
| 		"@vue/runtime-core": "3.4.3", | ||||
| 		"acorn": "8.11.2", | ||||
| 		"cross-env": "7.0.3", | ||||
| 		"cypress": "13.6.1", | ||||
| @@ -128,11 +131,10 @@ | ||||
| 		"start-server-and-test": "2.0.3", | ||||
| 		"storybook": "7.6.5", | ||||
| 		"storybook-addon-misskey-theme": "github:misskey-dev/storybook-addon-misskey-theme", | ||||
| 		"summaly": "github:misskey-dev/summaly", | ||||
| 		"vite-plugin-turbosnap": "1.0.3", | ||||
| 		"vitest": "0.34.6", | ||||
| 		"vitest-fetch-mock": "0.2.2", | ||||
| 		"vue-eslint-parser": "9.3.2", | ||||
| 		"vue-tsc": "1.8.25" | ||||
| 		"vue-tsc": "1.8.27" | ||||
| 	} | ||||
| } | ||||
|   | ||||
| @@ -11,7 +11,8 @@ import { miLocalStorage } from '@/local-storage.js'; | ||||
| import { MenuButton } from '@/types/menu.js'; | ||||
| import { del, get, set } from '@/scripts/idb-proxy.js'; | ||||
| import { apiUrl } from '@/config.js'; | ||||
| import { waiting, api, popup, popupMenu, success, alert } from '@/os.js'; | ||||
| import { waiting, popup, popupMenu, success, alert } from '@/os.js'; | ||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||
| import { unisonReload, reloadChannel } from '@/scripts/unison-reload.js'; | ||||
|  | ||||
| // TODO: 他のタブと永続化されたstateを同期 | ||||
| @@ -23,9 +24,14 @@ const accountData = miLocalStorage.getItem('account'); | ||||
| // TODO: 外部からはreadonlyに | ||||
| export const $i = accountData ? reactive(JSON.parse(accountData) as Account) : null; | ||||
|  | ||||
| export const iAmModerator = $i != null && ($i.isAdmin || $i.isModerator); | ||||
| export const iAmModerator = $i != null && ($i.isAdmin === true || $i.isModerator === true); | ||||
| export const iAmAdmin = $i != null && $i.isAdmin; | ||||
|  | ||||
| export function signinRequired() { | ||||
| 	if ($i == null) throw new Error('signin required'); | ||||
| 	return $i; | ||||
| } | ||||
|  | ||||
| export let notesCount = $i == null ? 0 : $i.notesCount; | ||||
| export function incNotesCount() { | ||||
| 	notesCount++; | ||||
| @@ -246,7 +252,7 @@ export async function openAccountMenu(opts: { | ||||
| 	} | ||||
|  | ||||
| 	const storedAccounts = await getAccounts().then(accounts => accounts.filter(x => x.id !== $i.id)); | ||||
| 	const accountsPromise = api('users/show', { userIds: storedAccounts.map(x => x.id) }); | ||||
| 	const accountsPromise = misskeyApi('users/show', { userIds: storedAccounts.map(x => x.id) }); | ||||
|  | ||||
| 	function createItem(account: Misskey.entities.UserDetailed) { | ||||
| 		return { | ||||
|   | ||||
| @@ -22,6 +22,7 @@ import { getAccountFromId } from '@/scripts/get-account-from-id.js'; | ||||
| import { deckStore } from '@/ui/deck/deck-store.js'; | ||||
| import { miLocalStorage } from '@/local-storage.js'; | ||||
| import { fetchCustomEmojis } from '@/custom-emojis.js'; | ||||
| import { setupRouter } from '@/global/router/definition.js'; | ||||
|  | ||||
| export async function common(createVue: () => App<Element>) { | ||||
| 	console.info(`Misskey v${version}`); | ||||
| @@ -241,6 +242,8 @@ export async function common(createVue: () => App<Element>) { | ||||
|  | ||||
| 	const app = createVue(); | ||||
|  | ||||
| 	setupRouter(app); | ||||
|  | ||||
| 	if (_DEV_) { | ||||
| 		app.config.performance = true; | ||||
| 	} | ||||
|   | ||||
| @@ -3,23 +3,23 @@ | ||||
|  * SPDX-License-Identifier: AGPL-3.0-only | ||||
|  */ | ||||
|  | ||||
| import { createApp, markRaw, defineAsyncComponent } from 'vue'; | ||||
| import { createApp, defineAsyncComponent, markRaw } from 'vue'; | ||||
| import { common } from './common.js'; | ||||
| import { ui } from '@/config.js'; | ||||
| import { i18n } from '@/i18n.js'; | ||||
| import { confirm, alert, post, popup, toast } from '@/os.js'; | ||||
| import { alert, confirm, popup, post, toast } from '@/os.js'; | ||||
| import { useStream } from '@/stream.js'; | ||||
| import * as sound from '@/scripts/sound.js'; | ||||
| import { $i, updateAccount, signout } from '@/account.js'; | ||||
| import { defaultStore, ColdDeviceStorage } from '@/store.js'; | ||||
| import { $i, signout, updateAccount } from '@/account.js'; | ||||
| import { ColdDeviceStorage, defaultStore } from '@/store.js'; | ||||
| import { makeHotkey } from '@/scripts/hotkey.js'; | ||||
| import { reactionPicker } from '@/scripts/reaction-picker.js'; | ||||
| import { miLocalStorage } from '@/local-storage.js'; | ||||
| import { claimAchievement, claimedAchievements } from '@/scripts/achievements.js'; | ||||
| import { mainRouter } from '@/router.js'; | ||||
| import { initializeSw } from '@/scripts/initialize-sw.js'; | ||||
| import { deckStore } from '@/ui/deck/deck-store.js'; | ||||
| import { emojiPicker } from '@/scripts/emoji-picker.js'; | ||||
| import { mainRouter } from '@/global/router/main.js'; | ||||
|  | ||||
| export async function mainBoot() { | ||||
| 	const { isClientUpdated } = await common(() => createApp( | ||||
| @@ -276,7 +276,7 @@ export async function mainBoot() { | ||||
|  | ||||
| 		main.on('unreadAntenna', () => { | ||||
| 			updateAccount({ hasUnreadAntenna: true }); | ||||
| 			sound.play('antenna'); | ||||
| 			sound.playMisskeySfx('antenna'); | ||||
| 		}); | ||||
|  | ||||
| 		stream.on('announcementCreated', (ev) => { | ||||
|   | ||||
| @@ -5,9 +5,9 @@ | ||||
|  | ||||
| import * as Misskey from 'misskey-js'; | ||||
| import { Cache } from '@/scripts/cache.js'; | ||||
| import { api } from '@/os.js'; | ||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||
|  | ||||
| export const clipsCache = new Cache<Misskey.entities.Clip[]>(1000 * 60 * 30, () => api('clips/list')); | ||||
| export const rolesCache = new Cache(1000 * 60 * 30, () => api('admin/roles/list')); | ||||
| export const userListsCache = new Cache<Misskey.entities.UserList[]>(1000 * 60 * 30, () => api('users/lists/list')); | ||||
| export const antennasCache = new Cache<Misskey.entities.Antenna[]>(1000 * 60 * 30, () => api('antennas/list')); | ||||
| export const clipsCache = new Cache<Misskey.entities.Clip[]>(1000 * 60 * 30, () => misskeyApi('clips/list')); | ||||
| export const rolesCache = new Cache(1000 * 60 * 30, () => misskeyApi('admin/roles/list')); | ||||
| export const userListsCache = new Cache<Misskey.entities.UserList[]>(1000 * 60 * 30, () => misskeyApi('users/lists/list')); | ||||
| export const antennasCache = new Cache<Misskey.entities.Antenna[]>(1000 * 60 * 30, () => misskeyApi('antennas/list')); | ||||
|   | ||||
| @@ -73,6 +73,7 @@ import MkButton from '@/components/MkButton.vue'; | ||||
| import MkSelect from '@/components/MkSelect.vue'; | ||||
| import * as os from '@/os.js'; | ||||
| import { i18n } from '@/i18n.js'; | ||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||
|  | ||||
| const props = defineProps<{ | ||||
| 	user: Misskey.entities.User; | ||||
| @@ -106,7 +107,7 @@ function muteUser() { | ||||
| } | ||||
|  | ||||
| function refreshUserInfo() { | ||||
| 	os.api('users/show', { userId: props.user.id }) | ||||
| 	misskeyApi('users/show', { userId: props.user.id }) | ||||
| 		.then((res) => { | ||||
| 			fullUserInfo.value = res; | ||||
| 		}); | ||||
| @@ -132,7 +133,7 @@ function send() { | ||||
| 		comment: comment.value, | ||||
| 		category: category.value, | ||||
| 	}, undefined).then(res => { | ||||
| 		os.api('users/show', { userId: props.user.id }) | ||||
| 		misskeyApi('users/show', { userId: props.user.id }) | ||||
| 			.then((res) => { | ||||
| 				fullUserInfo.value = res; | ||||
| 				uiWindow.value?.close(); | ||||
|   | ||||
| @@ -17,7 +17,7 @@ import * as Misskey from 'misskey-js'; | ||||
| import MkMention from './MkMention.vue'; | ||||
| import { i18n } from '@/i18n.js'; | ||||
| import { host as localHost } from '@/config.js'; | ||||
| import { api } from '@/os.js'; | ||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||
|  | ||||
| const user = ref<Misskey.entities.UserLite>(); | ||||
|  | ||||
| @@ -25,7 +25,7 @@ const props = defineProps<{ | ||||
| 	movedTo: string; // user id | ||||
| }>(); | ||||
|  | ||||
| api('users/show', { userId: props.movedTo }).then(u => user.value = u); | ||||
| misskeyApi('users/show', { userId: props.movedTo }).then(u => user.value = u); | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" module> | ||||
|   | ||||
| @@ -55,6 +55,7 @@ SPDX-License-Identifier: AGPL-3.0-only | ||||
| import * as Misskey from 'misskey-js'; | ||||
| import { onMounted, ref, computed } from 'vue'; | ||||
| import * as os from '@/os.js'; | ||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||
| import { i18n } from '@/i18n.js'; | ||||
| import { ACHIEVEMENT_TYPES, ACHIEVEMENT_BADGES, claimAchievement } from '@/scripts/achievements.js'; | ||||
|  | ||||
| @@ -71,7 +72,7 @@ const achievements = ref<Misskey.entities.UsersAchievementsResponse | null>(null | ||||
| const lockedAchievements = computed(() => ACHIEVEMENT_TYPES.filter(x => !(achievements.value ?? []).some(a => a.name === x))); | ||||
|  | ||||
| function fetch() { | ||||
| 	os.api('users/achievements', { userId: props.user.id }).then(res => { | ||||
| 	misskeyApi('users/achievements', { userId: props.user.id }).then(res => { | ||||
| 		achievements.value = []; | ||||
| 		for (const t of ACHIEVEMENT_TYPES) { | ||||
| 			const a = res.find(x => x.name === t); | ||||
|   | ||||
| @@ -28,6 +28,7 @@ SPDX-License-Identifier: AGPL-3.0-only | ||||
| import { onMounted, ref, shallowRef } from 'vue'; | ||||
| import * as Misskey from 'misskey-js'; | ||||
| import * as os from '@/os.js'; | ||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||
| import MkModal from '@/components/MkModal.vue'; | ||||
| import MkButton from '@/components/MkButton.vue'; | ||||
| import { i18n } from '@/i18n.js'; | ||||
| @@ -55,7 +56,7 @@ async function gotIt(): Promise<void> { | ||||
| 	} | ||||
|  | ||||
| 	modal.value.close(); | ||||
| 	os.api('i/read-announcement', { announcementId: props.announcement.id }); | ||||
| 	misskeyApi('i/read-announcement', { announcementId: props.announcement.id }); | ||||
| 	updateAccount({ | ||||
| 		unreadAnnouncements: $i!.unreadAnnouncements.filter(a => a.id !== props.announcement.id), | ||||
| 	}); | ||||
|   | ||||
| @@ -45,6 +45,7 @@ import contains from '@/scripts/contains.js'; | ||||
| import { char2twemojiFilePath, char2fluentEmojiFilePath } from '@/scripts/emoji-base.js'; | ||||
| import { acct } from '@/filters/user.js'; | ||||
| import * as os from '@/os.js'; | ||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||
| import { defaultStore } from '@/store.js'; | ||||
| import { emojilist, getEmojiName } from '@/scripts/emojilist.js'; | ||||
| import { i18n } from '@/i18n.js'; | ||||
| @@ -201,7 +202,7 @@ function exec() { | ||||
| 			users.value = JSON.parse(cache); | ||||
| 			fetching.value = false; | ||||
| 		} else { | ||||
| 			os.api('users/search-by-username-and-host', { | ||||
| 			misskeyApi('users/search-by-username-and-host', { | ||||
| 				username: props.q, | ||||
| 				limit: 10, | ||||
| 				detail: false, | ||||
| @@ -224,7 +225,7 @@ function exec() { | ||||
| 				hashtags.value = hashtags; | ||||
| 				fetching.value = false; | ||||
| 			} else { | ||||
| 				os.api('hashtags/search', { | ||||
| 				misskeyApi('hashtags/search', { | ||||
| 					query: props.q, | ||||
| 					limit: 30, | ||||
| 				}).then(searchedHashtags => { | ||||
|   | ||||
| @@ -15,7 +15,7 @@ SPDX-License-Identifier: AGPL-3.0-only | ||||
| <script lang="ts" setup> | ||||
| import { onMounted, ref } from 'vue'; | ||||
| import * as Misskey from 'misskey-js'; | ||||
| import * as os from '@/os.js'; | ||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||
|  | ||||
| const props = withDefaults(defineProps<{ | ||||
| 	userIds: string[]; | ||||
| @@ -27,7 +27,7 @@ const props = withDefaults(defineProps<{ | ||||
| const users = ref<Misskey.entities.UserLite[]>([]); | ||||
|  | ||||
| onMounted(async () => { | ||||
| 	users.value = await os.api('users/show', { | ||||
| 	users.value = await misskeyApi('users/show', { | ||||
| 		userIds: props.userIds, | ||||
| 	}) as unknown as Misskey.entities.UserLite[]; | ||||
| }); | ||||
|   | ||||
| @@ -131,6 +131,10 @@ function onMousedown(evt: MouseEvent): void { | ||||
| 	box-sizing: border-box; | ||||
| 	transition: background 0.1s ease; | ||||
|  | ||||
| 	&:hover { | ||||
| 		text-decoration: none; | ||||
| 	} | ||||
|  | ||||
| 	&:not(:disabled):hover { | ||||
| 		background: var(--buttonHoverBg); | ||||
| 	} | ||||
|   | ||||
| @@ -6,12 +6,16 @@ SPDX-License-Identifier: AGPL-3.0-only | ||||
| <template> | ||||
| <div> | ||||
| 	<span v-if="!available">{{ i18n.ts.waiting }}<MkEllipsis/></span> | ||||
| 	<div ref="captchaEl"></div> | ||||
| 	<div v-if="props.provider == 'mcaptcha'"> | ||||
| 		<div id="mcaptcha__widget-container" class="m-captcha-style"></div> | ||||
| 		<div ref="captchaEl"></div> | ||||
| 	</div> | ||||
| 	<div v-else ref="captchaEl"></div> | ||||
| </div> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts" setup> | ||||
| import { ref, shallowRef, computed, onMounted, onBeforeUnmount, watch } from 'vue'; | ||||
| import { ref, shallowRef, computed, onMounted, onBeforeUnmount, watch, onUnmounted } from 'vue'; | ||||
| import { defaultStore } from '@/store.js'; | ||||
| import { i18n } from '@/i18n.js'; | ||||
|  | ||||
| @@ -26,7 +30,7 @@ export type Captcha = { | ||||
| 	getResponse(id: string): string; | ||||
| }; | ||||
|  | ||||
| export type CaptchaProvider = 'hcaptcha' | 'recaptcha' | 'turnstile'; | ||||
| export type CaptchaProvider = 'hcaptcha' | 'recaptcha' | 'turnstile' | 'mcaptcha'; | ||||
|  | ||||
| type CaptchaContainer = { | ||||
| 	readonly [_ in CaptchaProvider]?: Captcha; | ||||
| @@ -39,6 +43,7 @@ declare global { | ||||
| const props = defineProps<{ | ||||
| 	provider: CaptchaProvider; | ||||
| 	sitekey: string | null; // null will show error on request | ||||
| 	instanceUrl?: string | null; | ||||
| 	modelValue?: string | null; | ||||
| }>(); | ||||
|  | ||||
| @@ -55,6 +60,7 @@ const variable = computed(() => { | ||||
| 		case 'hcaptcha': return 'hcaptcha'; | ||||
| 		case 'recaptcha': return 'grecaptcha'; | ||||
| 		case 'turnstile': return 'turnstile'; | ||||
| 		case 'mcaptcha': return 'mcaptcha'; | ||||
| 	} | ||||
| }); | ||||
|  | ||||
| @@ -65,6 +71,7 @@ const src = computed(() => { | ||||
| 		case 'hcaptcha': return 'https://js.hcaptcha.com/1/api.js?render=explicit&recaptchacompat=off'; | ||||
| 		case 'recaptcha': return 'https://www.recaptcha.net/recaptcha/api.js?render=explicit'; | ||||
| 		case 'turnstile': return 'https://challenges.cloudflare.com/turnstile/v0/api.js?render=explicit'; | ||||
| 		case 'mcaptcha': return null; | ||||
| 	} | ||||
| }); | ||||
|  | ||||
| @@ -72,9 +79,9 @@ const scriptId = computed(() => `script-${props.provider}`); | ||||
|  | ||||
| const captcha = computed<Captcha>(() => window[variable.value] || {} as unknown as Captcha); | ||||
|  | ||||
| if (loaded) { | ||||
| if (loaded || props.provider === 'mcaptcha') { | ||||
| 	available.value = true; | ||||
| } else { | ||||
| } else if (src.value !== null) { | ||||
| 	(document.getElementById(scriptId.value) ?? document.head.appendChild(Object.assign(document.createElement('script'), { | ||||
| 		async: true, | ||||
| 		id: scriptId.value, | ||||
| @@ -87,7 +94,7 @@ function reset() { | ||||
| 	if (captcha.value.reset) captcha.value.reset(); | ||||
| } | ||||
|  | ||||
| function requestRender() { | ||||
| async function requestRender() { | ||||
| 	if (captcha.value.render && captchaEl.value instanceof Element) { | ||||
| 		captcha.value.render(captchaEl.value, { | ||||
| 			sitekey: props.sitekey, | ||||
| @@ -96,6 +103,15 @@ function requestRender() { | ||||
| 			'expired-callback': callback, | ||||
| 			'error-callback': callback, | ||||
| 		}); | ||||
| 	} else if (props.provider === 'mcaptcha' && props.instanceUrl && props.sitekey) { | ||||
| 		const { default: Widget } = await import('@mcaptcha/vanilla-glue'); | ||||
| 		// @ts-expect-error avoid typecheck error | ||||
| 		new Widget({ | ||||
| 			siteKey: { | ||||
| 				instanceUrl: new URL(props.instanceUrl), | ||||
| 				key: props.sitekey, | ||||
| 			}, | ||||
| 		}); | ||||
| 	} else { | ||||
| 		window.setTimeout(requestRender, 1); | ||||
| 	} | ||||
| @@ -105,14 +121,27 @@ function callback(response?: string) { | ||||
| 	emit('update:modelValue', typeof response === 'string' ? response : null); | ||||
| } | ||||
|  | ||||
| function onReceivedMessage(message: MessageEvent) { | ||||
| 	if (message.data.token) { | ||||
| 		if (props.instanceUrl && new URL(message.origin).host === new URL(props.instanceUrl).host) { | ||||
| 			callback(<string>message.data.token); | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| onMounted(() => { | ||||
| 	if (available.value) { | ||||
| 		window.addEventListener('message', onReceivedMessage); | ||||
| 		requestRender(); | ||||
| 	} else { | ||||
| 		watch(available, requestRender); | ||||
| 	} | ||||
| }); | ||||
|  | ||||
| onUnmounted(() => { | ||||
| 	window.removeEventListener('message', onReceivedMessage); | ||||
| }); | ||||
|  | ||||
| onBeforeUnmount(() => { | ||||
| 	reset(); | ||||
| }); | ||||
|   | ||||
| @@ -26,7 +26,7 @@ SPDX-License-Identifier: AGPL-3.0-only | ||||
|  | ||||
| <script lang="ts" setup> | ||||
| import { ref } from 'vue'; | ||||
| import * as os from '@/os.js'; | ||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||
| import { i18n } from '@/i18n.js'; | ||||
|  | ||||
| const props = withDefaults(defineProps<{ | ||||
| @@ -44,12 +44,12 @@ async function onClick() { | ||||
|  | ||||
| 	try { | ||||
| 		if (isFollowing.value) { | ||||
| 			await os.api('channels/unfollow', { | ||||
| 			await misskeyApi('channels/unfollow', { | ||||
| 				channelId: props.channel.id, | ||||
| 			}); | ||||
| 			isFollowing.value = false; | ||||
| 		} else { | ||||
| 			await os.api('channels/follow', { | ||||
| 			await misskeyApi('channels/follow', { | ||||
| 				channelId: props.channel.id, | ||||
| 			}); | ||||
| 			isFollowing.value = true; | ||||
|   | ||||
| @@ -22,7 +22,7 @@ SPDX-License-Identifier: AGPL-3.0-only | ||||
| import { onMounted, ref, shallowRef, watch, PropType } from 'vue'; | ||||
| import { Chart } from 'chart.js'; | ||||
| import gradient from 'chartjs-plugin-gradient'; | ||||
| import * as os from '@/os.js'; | ||||
| import { misskeyApiGet } from '@/scripts/misskey-api.js'; | ||||
| import { defaultStore } from '@/store.js'; | ||||
| import { useChartTooltip } from '@/scripts/use-chart-tooltip.js'; | ||||
| import { chartVLine } from '@/scripts/chart-vline.js'; | ||||
| @@ -277,7 +277,7 @@ const exportData = () => { | ||||
| }; | ||||
|  | ||||
| const fetchFederationChart = async (): Promise<typeof chartData> => { | ||||
| 	const raw = await os.apiGet('charts/federation', { limit: props.limit, span: props.span }); | ||||
| 	const raw = await misskeyApiGet('charts/federation', { limit: props.limit, span: props.span }); | ||||
| 	return { | ||||
| 		series: [{ | ||||
| 			name: 'Received', | ||||
| @@ -327,7 +327,7 @@ const fetchFederationChart = async (): Promise<typeof chartData> => { | ||||
| }; | ||||
|  | ||||
| const fetchApRequestChart = async (): Promise<typeof chartData> => { | ||||
| 	const raw = await os.apiGet('charts/ap-request', { limit: props.limit, span: props.span }); | ||||
| 	const raw = await misskeyApiGet('charts/ap-request', { limit: props.limit, span: props.span }); | ||||
| 	return { | ||||
| 		series: [{ | ||||
| 			name: 'In', | ||||
| @@ -349,7 +349,7 @@ const fetchApRequestChart = async (): Promise<typeof chartData> => { | ||||
| }; | ||||
|  | ||||
| const fetchNotesChart = async (type: string): Promise<typeof chartData> => { | ||||
| 	const raw = await os.apiGet('charts/notes', { limit: props.limit, span: props.span }); | ||||
| 	const raw = await misskeyApiGet('charts/notes', { limit: props.limit, span: props.span }); | ||||
| 	return { | ||||
| 		series: [{ | ||||
| 			name: 'All', | ||||
| @@ -396,7 +396,7 @@ const fetchNotesChart = async (type: string): Promise<typeof chartData> => { | ||||
| }; | ||||
|  | ||||
| const fetchNotesTotalChart = async (): Promise<typeof chartData> => { | ||||
| 	const raw = await os.apiGet('charts/notes', { limit: props.limit, span: props.span }); | ||||
| 	const raw = await misskeyApiGet('charts/notes', { limit: props.limit, span: props.span }); | ||||
| 	return { | ||||
| 		series: [{ | ||||
| 			name: 'Combined', | ||||
| @@ -415,7 +415,7 @@ const fetchNotesTotalChart = async (): Promise<typeof chartData> => { | ||||
| }; | ||||
|  | ||||
| const fetchUsersChart = async (total: boolean): Promise<typeof chartData> => { | ||||
| 	const raw = await os.apiGet('charts/users', { limit: props.limit, span: props.span }); | ||||
| 	const raw = await misskeyApiGet('charts/users', { limit: props.limit, span: props.span }); | ||||
| 	return { | ||||
| 		series: [{ | ||||
| 			name: 'Combined', | ||||
| @@ -443,7 +443,7 @@ const fetchUsersChart = async (total: boolean): Promise<typeof chartData> => { | ||||
| }; | ||||
|  | ||||
| const fetchActiveUsersChart = async (): Promise<typeof chartData> => { | ||||
| 	const raw = await os.apiGet('charts/active-users', { limit: props.limit, span: props.span }); | ||||
| 	const raw = await misskeyApiGet('charts/active-users', { limit: props.limit, span: props.span }); | ||||
| 	return { | ||||
| 		series: [{ | ||||
| 			name: 'Read & Write', | ||||
| @@ -495,7 +495,7 @@ const fetchActiveUsersChart = async (): Promise<typeof chartData> => { | ||||
| }; | ||||
|  | ||||
| const fetchDriveChart = async (): Promise<typeof chartData> => { | ||||
| 	const raw = await os.apiGet('charts/drive', { limit: props.limit, span: props.span }); | ||||
| 	const raw = await misskeyApiGet('charts/drive', { limit: props.limit, span: props.span }); | ||||
| 	return { | ||||
| 		bytes: true, | ||||
| 		series: [{ | ||||
| @@ -531,7 +531,7 @@ const fetchDriveChart = async (): Promise<typeof chartData> => { | ||||
| }; | ||||
|  | ||||
| const fetchDriveFilesChart = async (): Promise<typeof chartData> => { | ||||
| 	const raw = await os.apiGet('charts/drive', { limit: props.limit, span: props.span }); | ||||
| 	const raw = await misskeyApiGet('charts/drive', { limit: props.limit, span: props.span }); | ||||
| 	return { | ||||
| 		series: [{ | ||||
| 			name: 'All', | ||||
| @@ -566,7 +566,7 @@ const fetchDriveFilesChart = async (): Promise<typeof chartData> => { | ||||
| }; | ||||
|  | ||||
| const fetchInstanceRequestsChart = async (): Promise<typeof chartData> => { | ||||
| 	const raw = await os.apiGet('charts/instance', { host: props.args.host, limit: props.limit, span: props.span }); | ||||
| 	const raw = await misskeyApiGet('charts/instance', { host: props.args.host, limit: props.limit, span: props.span }); | ||||
| 	return { | ||||
| 		series: [{ | ||||
| 			name: 'In', | ||||
| @@ -588,7 +588,7 @@ const fetchInstanceRequestsChart = async (): Promise<typeof chartData> => { | ||||
| }; | ||||
|  | ||||
| const fetchInstanceUsersChart = async (total: boolean): Promise<typeof chartData> => { | ||||
| 	const raw = await os.apiGet('charts/instance', { host: props.args.host, limit: props.limit, span: props.span }); | ||||
| 	const raw = await misskeyApiGet('charts/instance', { host: props.args.host, limit: props.limit, span: props.span }); | ||||
| 	return { | ||||
| 		series: [{ | ||||
| 			name: 'Users', | ||||
| @@ -603,7 +603,7 @@ const fetchInstanceUsersChart = async (total: boolean): Promise<typeof chartData | ||||
| }; | ||||
|  | ||||
| const fetchInstanceNotesChart = async (total: boolean): Promise<typeof chartData> => { | ||||
| 	const raw = await os.apiGet('charts/instance', { host: props.args.host, limit: props.limit, span: props.span }); | ||||
| 	const raw = await misskeyApiGet('charts/instance', { host: props.args.host, limit: props.limit, span: props.span }); | ||||
| 	return { | ||||
| 		series: [{ | ||||
| 			name: 'Notes', | ||||
| @@ -618,7 +618,7 @@ const fetchInstanceNotesChart = async (total: boolean): Promise<typeof chartData | ||||
| }; | ||||
|  | ||||
| const fetchInstanceFfChart = async (total: boolean): Promise<typeof chartData> => { | ||||
| 	const raw = await os.apiGet('charts/instance', { host: props.args.host, limit: props.limit, span: props.span }); | ||||
| 	const raw = await misskeyApiGet('charts/instance', { host: props.args.host, limit: props.limit, span: props.span }); | ||||
| 	return { | ||||
| 		series: [{ | ||||
| 			name: 'Following', | ||||
| @@ -641,7 +641,7 @@ const fetchInstanceFfChart = async (total: boolean): Promise<typeof chartData> = | ||||
| }; | ||||
|  | ||||
| const fetchInstanceDriveUsageChart = async (total: boolean): Promise<typeof chartData> => { | ||||
| 	const raw = await os.apiGet('charts/instance', { host: props.args.host, limit: props.limit, span: props.span }); | ||||
| 	const raw = await misskeyApiGet('charts/instance', { host: props.args.host, limit: props.limit, span: props.span }); | ||||
| 	return { | ||||
| 		bytes: true, | ||||
| 		series: [{ | ||||
| @@ -657,7 +657,7 @@ const fetchInstanceDriveUsageChart = async (total: boolean): Promise<typeof char | ||||
| }; | ||||
|  | ||||
| const fetchInstanceDriveFilesChart = async (total: boolean): Promise<typeof chartData> => { | ||||
| 	const raw = await os.apiGet('charts/instance', { host: props.args.host, limit: props.limit, span: props.span }); | ||||
| 	const raw = await misskeyApiGet('charts/instance', { host: props.args.host, limit: props.limit, span: props.span }); | ||||
| 	return { | ||||
| 		series: [{ | ||||
| 			name: 'Drive files', | ||||
| @@ -672,7 +672,7 @@ const fetchInstanceDriveFilesChart = async (total: boolean): Promise<typeof char | ||||
| }; | ||||
|  | ||||
| const fetchPerUserNotesChart = async (): Promise<typeof chartData> => { | ||||
| 	const raw = await os.apiGet('charts/user/notes', { userId: props.args.user.id, limit: props.limit, span: props.span }); | ||||
| 	const raw = await misskeyApiGet('charts/user/notes', { userId: props.args.user.id, limit: props.limit, span: props.span }); | ||||
| 	return { | ||||
| 		series: [...(props.args.withoutAll ? [] : [{ | ||||
| 			name: 'All', | ||||
| @@ -704,7 +704,7 @@ const fetchPerUserNotesChart = async (): Promise<typeof chartData> => { | ||||
| }; | ||||
|  | ||||
| const fetchPerUserPvChart = async (): Promise<typeof chartData> => { | ||||
| 	const raw = await os.apiGet('charts/user/pv', { userId: props.args.user.id, limit: props.limit, span: props.span }); | ||||
| 	const raw = await misskeyApiGet('charts/user/pv', { userId: props.args.user.id, limit: props.limit, span: props.span }); | ||||
| 	return { | ||||
| 		series: [{ | ||||
| 			name: 'Unique PV (user)', | ||||
| @@ -731,7 +731,7 @@ const fetchPerUserPvChart = async (): Promise<typeof chartData> => { | ||||
| }; | ||||
|  | ||||
| const fetchPerUserFollowingChart = async (): Promise<typeof chartData> => { | ||||
| 	const raw = await os.apiGet('charts/user/following', { userId: props.args.user.id, limit: props.limit, span: props.span }); | ||||
| 	const raw = await misskeyApiGet('charts/user/following', { userId: props.args.user.id, limit: props.limit, span: props.span }); | ||||
| 	return { | ||||
| 		series: [{ | ||||
| 			name: 'Local', | ||||
| @@ -746,7 +746,7 @@ const fetchPerUserFollowingChart = async (): Promise<typeof chartData> => { | ||||
| }; | ||||
|  | ||||
| const fetchPerUserFollowersChart = async (): Promise<typeof chartData> => { | ||||
| 	const raw = await os.apiGet('charts/user/following', { userId: props.args.user.id, limit: props.limit, span: props.span }); | ||||
| 	const raw = await misskeyApiGet('charts/user/following', { userId: props.args.user.id, limit: props.limit, span: props.span }); | ||||
| 	return { | ||||
| 		series: [{ | ||||
| 			name: 'Local', | ||||
| @@ -761,7 +761,7 @@ const fetchPerUserFollowersChart = async (): Promise<typeof chartData> => { | ||||
| }; | ||||
|  | ||||
| const fetchPerUserDriveChart = async (): Promise<typeof chartData> => { | ||||
| 	const raw = await os.apiGet('charts/user/drive', { userId: props.args.user.id, limit: props.limit, span: props.span }); | ||||
| 	const raw = await misskeyApiGet('charts/user/drive', { userId: props.args.user.id, limit: props.limit, span: props.span }); | ||||
| 	return { | ||||
| 		series: [{ | ||||
| 			name: 'Inc', | ||||
|   | ||||
| @@ -10,7 +10,7 @@ SPDX-License-Identifier: AGPL-3.0-only | ||||
| 		<div :class="$style.codeEditorScroller"> | ||||
| 			<textarea | ||||
| 				ref="inputEl" | ||||
| 				v-model="vModel" | ||||
| 				v-model="v" | ||||
| 				:class="[$style.textarea]" | ||||
| 				:placeholder="placeholder" | ||||
| 				:disabled="disabled" | ||||
| @@ -60,7 +60,6 @@ const emit = defineEmits<{ | ||||
| }>(); | ||||
|  | ||||
| const { modelValue } = toRefs(props); | ||||
| const vModel = ref<string>(modelValue.value ?? ''); | ||||
| const v = ref<string>(modelValue.value ?? ''); | ||||
| const focused = ref(false); | ||||
| const changed = ref(false); | ||||
| @@ -81,15 +80,14 @@ const onKeydown = (ev: KeyboardEvent) => { | ||||
|  | ||||
| 	if (ev.code === 'Enter') { | ||||
| 		const pos = inputEl.value?.selectionStart ?? 0; | ||||
| 		const posEnd = inputEl.value?.selectionEnd ?? vModel.value.length; | ||||
| 		const posEnd = inputEl.value?.selectionEnd ?? v.value.length; | ||||
| 		if (pos === posEnd) { | ||||
| 			const lines = vModel.value.slice(0, pos).split('\n'); | ||||
| 			const lines = v.value.slice(0, pos).split('\n'); | ||||
| 			const currentLine = lines[lines.length - 1]; | ||||
| 			const currentLineSpaces = currentLine.match(/^\s+/); | ||||
| 			const posDelta = currentLineSpaces ? currentLineSpaces[0].length : 0; | ||||
| 			ev.preventDefault(); | ||||
| 			vModel.value = vModel.value.slice(0, pos) + '\n' + (currentLineSpaces ? currentLineSpaces[0] : '') + vModel.value.slice(pos); | ||||
| 			v.value = vModel.value; | ||||
| 			v.value = v.value.slice(0, pos) + '\n' + (currentLineSpaces ? currentLineSpaces[0] : '') + v.value.slice(pos); | ||||
| 			nextTick(() => { | ||||
| 				inputEl.value?.setSelectionRange(pos + 1 + posDelta, pos + 1 + posDelta); | ||||
| 			}); | ||||
| @@ -99,9 +97,8 @@ const onKeydown = (ev: KeyboardEvent) => { | ||||
|  | ||||
| 	if (ev.key === 'Tab') { | ||||
| 		const pos = inputEl.value?.selectionStart ?? 0; | ||||
| 		const posEnd = inputEl.value?.selectionEnd ?? vModel.value.length; | ||||
| 		vModel.value = vModel.value.slice(0, pos) + '\t' + vModel.value.slice(posEnd); | ||||
| 		v.value = vModel.value; | ||||
| 		const posEnd = inputEl.value?.selectionEnd ?? v.value.length; | ||||
| 		v.value = v.value.slice(0, pos) + '\t' + v.value.slice(posEnd); | ||||
| 		nextTick(() => { | ||||
| 			inputEl.value?.setSelectionRange(pos + 1, pos + 1); | ||||
| 		}); | ||||
|   | ||||
| @@ -45,9 +45,9 @@ import bytes from '@/filters/bytes.js'; | ||||
| import * as os from '@/os.js'; | ||||
| import { i18n } from '@/i18n.js'; | ||||
| import { $i } from '@/account.js'; | ||||
| import { useRouter } from '@/router.js'; | ||||
| import { getDriveFileMenu } from '@/scripts/get-drive-file-menu.js'; | ||||
| import { deviceKind } from '@/scripts/device-kind.js'; | ||||
| import { useRouter } from '@/global/router/supplier.js'; | ||||
|  | ||||
| const router = useRouter(); | ||||
|  | ||||
|   | ||||
| @@ -35,6 +35,7 @@ SPDX-License-Identifier: AGPL-3.0-only | ||||
| import { computed, defineAsyncComponent, ref } from 'vue'; | ||||
| import * as Misskey from 'misskey-js'; | ||||
| import * as os from '@/os.js'; | ||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||
| import { i18n } from '@/i18n.js'; | ||||
| import { defaultStore } from '@/store.js'; | ||||
| import { claimAchievement } from '@/scripts/achievements.js'; | ||||
| @@ -144,7 +145,7 @@ function onDrop(ev: DragEvent) { | ||||
| 	if (driveFile != null && driveFile !== '') { | ||||
| 		const file = JSON.parse(driveFile); | ||||
| 		emit('removeFile', file.id); | ||||
| 		os.api('drive/files/update', { | ||||
| 		misskeyApi('drive/files/update', { | ||||
| 			fileId: file.id, | ||||
| 			folderId: props.folder.id, | ||||
| 		}); | ||||
| @@ -160,7 +161,7 @@ function onDrop(ev: DragEvent) { | ||||
| 		if (folder.id === props.folder.id) return; | ||||
|  | ||||
| 		emit('removeFolder', folder.id); | ||||
| 		os.api('drive/folders/update', { | ||||
| 		misskeyApi('drive/folders/update', { | ||||
| 			folderId: folder.id, | ||||
| 			parentId: props.folder.id, | ||||
| 		}).then(() => { | ||||
| @@ -214,7 +215,7 @@ function rename() { | ||||
| 		default: props.folder.name, | ||||
| 	}).then(({ canceled, result: name }) => { | ||||
| 		if (canceled) return; | ||||
| 		os.api('drive/folders/update', { | ||||
| 		misskeyApi('drive/folders/update', { | ||||
| 			folderId: props.folder.id, | ||||
| 			name: name, | ||||
| 		}); | ||||
| @@ -222,7 +223,7 @@ function rename() { | ||||
| } | ||||
|  | ||||
| function deleteFolder() { | ||||
| 	os.api('drive/folders/delete', { | ||||
| 	misskeyApi('drive/folders/delete', { | ||||
| 		folderId: props.folder.id, | ||||
| 	}).then(() => { | ||||
| 		if (defaultStore.state.uploadFolder === props.folder.id) { | ||||
|   | ||||
| @@ -20,7 +20,7 @@ SPDX-License-Identifier: AGPL-3.0-only | ||||
| <script lang="ts" setup> | ||||
| import { ref } from 'vue'; | ||||
| import * as Misskey from 'misskey-js'; | ||||
| import * as os from '@/os.js'; | ||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||
| import { i18n } from '@/i18n.js'; | ||||
|  | ||||
| const props = defineProps<{ | ||||
| @@ -112,7 +112,7 @@ function onDrop(ev: DragEvent) { | ||||
| 	if (driveFile != null && driveFile !== '') { | ||||
| 		const file = JSON.parse(driveFile); | ||||
| 		emit('removeFile', file.id); | ||||
| 		os.api('drive/files/update', { | ||||
| 		misskeyApi('drive/files/update', { | ||||
| 			fileId: file.id, | ||||
| 			folderId: props.folder ? props.folder.id : null, | ||||
| 		}); | ||||
| @@ -126,7 +126,7 @@ function onDrop(ev: DragEvent) { | ||||
| 		// 移動先が自分自身ならreject | ||||
| 		if (props.folder && folder.id === props.folder.id) return; | ||||
| 		emit('removeFolder', folder.id); | ||||
| 		os.api('drive/folders/update', { | ||||
| 		misskeyApi('drive/folders/update', { | ||||
| 			folderId: folder.id, | ||||
| 			parentId: props.folder ? props.folder.id : null, | ||||
| 		}); | ||||
|   | ||||
| @@ -102,6 +102,7 @@ import XNavFolder from '@/components/MkDrive.navFolder.vue'; | ||||
| import XFolder from '@/components/MkDrive.folder.vue'; | ||||
| import XFile from '@/components/MkDrive.file.vue'; | ||||
| import * as os from '@/os.js'; | ||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||
| import { useStream } from '@/stream.js'; | ||||
| import { defaultStore } from '@/store.js'; | ||||
| import { i18n } from '@/i18n.js'; | ||||
| @@ -254,7 +255,7 @@ function onDrop(ev: DragEvent): any { | ||||
| 		const file = JSON.parse(driveFile); | ||||
| 		if (files.value.some(f => f.id === file.id)) return; | ||||
| 		removeFile(file.id); | ||||
| 		os.api('drive/files/update', { | ||||
| 		misskeyApi('drive/files/update', { | ||||
| 			fileId: file.id, | ||||
| 			folderId: folder.value ? folder.value.id : null, | ||||
| 		}); | ||||
| @@ -270,7 +271,7 @@ function onDrop(ev: DragEvent): any { | ||||
| 		if (folder.value && droppedFolder.id === folder.value.id) return false; | ||||
| 		if (folders.value.some(f => f.id === droppedFolder.id)) return false; | ||||
| 		removeFolder(droppedFolder.id); | ||||
| 		os.api('drive/folders/update', { | ||||
| 		misskeyApi('drive/folders/update', { | ||||
| 			folderId: droppedFolder.id, | ||||
| 			parentId: folder.value ? folder.value.id : null, | ||||
| 		}).then(() => { | ||||
| @@ -307,7 +308,7 @@ function urlUpload() { | ||||
| 		placeholder: i18n.ts.uploadFromUrlDescription, | ||||
| 	}).then(({ canceled, result: url }) => { | ||||
| 		if (canceled || !url) return; | ||||
| 		os.api('drive/files/upload-from-url', { | ||||
| 		misskeyApi('drive/files/upload-from-url', { | ||||
| 			url: url, | ||||
| 			folderId: folder.value ? folder.value.id : undefined, | ||||
| 		}); | ||||
| @@ -325,7 +326,7 @@ function createFolder() { | ||||
| 		placeholder: i18n.ts.folderName, | ||||
| 	}).then(({ canceled, result: name }) => { | ||||
| 		if (canceled) return; | ||||
| 		os.api('drive/folders/create', { | ||||
| 		misskeyApi('drive/folders/create', { | ||||
| 			name: name, | ||||
| 			parentId: folder.value ? folder.value.id : undefined, | ||||
| 		}).then(createdFolder => { | ||||
| @@ -341,7 +342,7 @@ function renameFolder(folderToRename: Misskey.entities.DriveFolder) { | ||||
| 		default: folderToRename.name, | ||||
| 	}).then(({ canceled, result: name }) => { | ||||
| 		if (canceled) return; | ||||
| 		os.api('drive/folders/update', { | ||||
| 		misskeyApi('drive/folders/update', { | ||||
| 			folderId: folderToRename.id, | ||||
| 			name: name, | ||||
| 		}).then(updatedFolder => { | ||||
| @@ -352,7 +353,7 @@ function renameFolder(folderToRename: Misskey.entities.DriveFolder) { | ||||
| } | ||||
|  | ||||
| function deleteFolder(folderToDelete: Misskey.entities.DriveFolder) { | ||||
| 	os.api('drive/folders/delete', { | ||||
| 	misskeyApi('drive/folders/delete', { | ||||
| 		folderId: folderToDelete.id, | ||||
| 	}).then(() => { | ||||
| 		// 削除時に親フォルダに移動 | ||||
| @@ -436,7 +437,7 @@ function move(target?: Misskey.entities.DriveFolder) { | ||||
|  | ||||
| 	fetching.value = true; | ||||
|  | ||||
| 	os.api('drive/folders/show', { | ||||
| 	misskeyApi('drive/folders/show', { | ||||
| 		folderId: target, | ||||
| 	}).then(folderToMove => { | ||||
| 		folder.value = folderToMove; | ||||
| @@ -535,7 +536,7 @@ async function fetch() { | ||||
| 	const foldersMax = 30; | ||||
| 	const filesMax = 30; | ||||
|  | ||||
| 	const foldersPromise = os.api('drive/folders', { | ||||
| 	const foldersPromise = misskeyApi('drive/folders', { | ||||
| 		folderId: folder.value ? folder.value.id : null, | ||||
| 		limit: foldersMax + 1, | ||||
| 	}).then(fetchedFolders => { | ||||
| @@ -546,7 +547,7 @@ async function fetch() { | ||||
| 		return fetchedFolders; | ||||
| 	}); | ||||
|  | ||||
| 	const filesPromise = os.api('drive/files', { | ||||
| 	const filesPromise = misskeyApi('drive/files', { | ||||
| 		folderId: folder.value ? folder.value.id : null, | ||||
| 		type: props.type, | ||||
| 		limit: filesMax + 1, | ||||
| @@ -571,7 +572,7 @@ function fetchMoreFolders() { | ||||
|  | ||||
| 	const max = 30; | ||||
|  | ||||
| 	os.api('drive/folders', { | ||||
| 	misskeyApi('drive/folders', { | ||||
| 		folderId: folder.value ? folder.value.id : null, | ||||
| 		type: props.type, | ||||
| 		untilId: folders.value[folders.value.length - 1].id, | ||||
| @@ -594,7 +595,7 @@ function fetchMoreFiles() { | ||||
| 	const max = 30; | ||||
|  | ||||
| 	// ファイル一覧取得 | ||||
| 	os.api('drive/files', { | ||||
| 	misskeyApi('drive/files', { | ||||
| 		folderId: folder.value ? folder.value.id : null, | ||||
| 		type: props.type, | ||||
| 		untilId: files.value[files.value.length - 1].id, | ||||
|   | ||||
| @@ -10,11 +10,11 @@ SPDX-License-Identifier: AGPL-3.0-only | ||||
| <script lang="ts" setup> | ||||
| import { ref } from 'vue'; | ||||
| import * as Misskey from 'misskey-js'; | ||||
| import * as os from '@/os.js'; | ||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||
|  | ||||
| const meta = ref<Misskey.entities.MetaResponse>(); | ||||
|  | ||||
| os.api('meta', { detail: true }).then(gotMeta => { | ||||
| misskeyApi('meta', { detail: true }).then(gotMeta => { | ||||
| 	meta.value = gotMeta; | ||||
| }); | ||||
| </script> | ||||
|   | ||||
| @@ -38,11 +38,12 @@ SPDX-License-Identifier: AGPL-3.0-only | ||||
| import { onBeforeUnmount, onMounted, ref } from 'vue'; | ||||
| import * as Misskey from 'misskey-js'; | ||||
| import * as os from '@/os.js'; | ||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||
| import { useStream } from '@/stream.js'; | ||||
| import { i18n } from '@/i18n.js'; | ||||
| import { claimAchievement } from '@/scripts/achievements.js'; | ||||
| import { $i } from '@/account.js'; | ||||
| import { defaultStore } from "@/store.js"; | ||||
| import { defaultStore } from '@/store.js'; | ||||
|  | ||||
| const props = withDefaults(defineProps<{ | ||||
| 	user: Misskey.entities.UserDetailed, | ||||
| @@ -63,7 +64,7 @@ const wait = ref(false); | ||||
| const connection = useStream().useChannel('main'); | ||||
|  | ||||
| if (props.user.isFollowing == null) { | ||||
| 	os.api('users/show', { | ||||
| 	misskeyApi('users/show', { | ||||
| 		userId: props.user.id, | ||||
| 	}) | ||||
| 		.then(onFollowChange); | ||||
| @@ -88,17 +89,17 @@ async function onClick() { | ||||
|  | ||||
| 			if (canceled) return; | ||||
|  | ||||
| 			await os.api('following/delete', { | ||||
| 			await misskeyApi('following/delete', { | ||||
| 				userId: props.user.id, | ||||
| 			}); | ||||
| 		} else { | ||||
| 			if (hasPendingFollowRequestFromYou.value) { | ||||
| 				await os.api('following/requests/cancel', { | ||||
| 				await misskeyApi('following/requests/cancel', { | ||||
| 					userId: props.user.id, | ||||
| 				}); | ||||
| 				hasPendingFollowRequestFromYou.value = false; | ||||
| 			} else { | ||||
| 				await os.api('following/create', { | ||||
| 				await misskeyApi('following/create', { | ||||
| 					userId: props.user.id, | ||||
| 					withReplies: defaultStore.state.defaultWithReplies, | ||||
| 				}); | ||||
|   | ||||
| @@ -20,7 +20,7 @@ SPDX-License-Identifier: AGPL-3.0-only | ||||
| 	</template> | ||||
|  | ||||
| 	<MkSpacer :marginMin="20" :marginMax="32"> | ||||
| 		<div class="_gaps_m"> | ||||
| 		<div v-if="Object.keys(form).filter(item => !form[item].hidden).length > 0" class="_gaps_m"> | ||||
| 			<template v-for="item in Object.keys(form).filter(item => !form[item].hidden)"> | ||||
| 				<MkInput v-if="form[item].type === 'number'" v-model="values[item]" type="number" :step="form[item].step || 1"> | ||||
| 					<template #label><span v-text="form[item].label || item"></span><span v-if="form[item].required === false"> ({{ i18n.ts.optional }})</span></template> | ||||
| @@ -55,6 +55,10 @@ SPDX-License-Identifier: AGPL-3.0-only | ||||
| 				</MkButton> | ||||
| 			</template> | ||||
| 		</div> | ||||
| 		<div v-else class="_fullinfo"> | ||||
| 			<img :src="infoImageUrl" class="_ghost"/> | ||||
| 			<div>{{ i18n.ts.nothing }}</div> | ||||
| 		</div> | ||||
| 	</MkSpacer> | ||||
| </MkModalWindow> | ||||
| </template> | ||||
| @@ -70,6 +74,7 @@ import MkButton from './MkButton.vue'; | ||||
| import MkRadios from './MkRadios.vue'; | ||||
| import MkModalWindow from '@/components/MkModalWindow.vue'; | ||||
| import { i18n } from '@/i18n.js'; | ||||
| import { infoImageUrl } from '@/instance.js'; | ||||
|  | ||||
| const props = defineProps<{ | ||||
| 	title: string; | ||||
|   | ||||
| @@ -15,7 +15,7 @@ SPDX-License-Identifier: AGPL-3.0-only | ||||
| <script lang="ts" setup> | ||||
| import { onMounted, nextTick, watch, shallowRef, ref } from 'vue'; | ||||
| import { Chart } from 'chart.js'; | ||||
| import * as os from '@/os.js'; | ||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||
| import { defaultStore } from '@/store.js'; | ||||
| import { useChartTooltip } from '@/scripts/use-chart-tooltip.js'; | ||||
| import { alpha } from '@/scripts/color.js'; | ||||
| @@ -72,19 +72,19 @@ async function renderChart() { | ||||
| 	let values; | ||||
|  | ||||
| 	if (props.src === 'active-users') { | ||||
| 		const raw = await os.api('charts/active-users', { limit: chartLimit, span: 'day' }); | ||||
| 		const raw = await misskeyApi('charts/active-users', { limit: chartLimit, span: 'day' }); | ||||
| 		values = raw.readWrite; | ||||
| 	} else if (props.src === 'notes') { | ||||
| 		const raw = await os.api('charts/notes', { limit: chartLimit, span: 'day' }); | ||||
| 		const raw = await misskeyApi('charts/notes', { limit: chartLimit, span: 'day' }); | ||||
| 		values = raw.local.inc; | ||||
| 	} else if (props.src === 'ap-requests-inbox-received') { | ||||
| 		const raw = await os.api('charts/ap-request', { limit: chartLimit, span: 'day' }); | ||||
| 		const raw = await misskeyApi('charts/ap-request', { limit: chartLimit, span: 'day' }); | ||||
| 		values = raw.inboxReceived; | ||||
| 	} else if (props.src === 'ap-requests-deliver-succeeded') { | ||||
| 		const raw = await os.api('charts/ap-request', { limit: chartLimit, span: 'day' }); | ||||
| 		const raw = await misskeyApi('charts/ap-request', { limit: chartLimit, span: 'day' }); | ||||
| 		values = raw.deliverSucceeded; | ||||
| 	} else if (props.src === 'ap-requests-deliver-failed') { | ||||
| 		const raw = await os.api('charts/ap-request', { limit: chartLimit, span: 'day' }); | ||||
| 		const raw = await misskeyApi('charts/ap-request', { limit: chartLimit, span: 'day' }); | ||||
| 		values = raw.deliverFailed; | ||||
| 	} | ||||
|  | ||||
|   | ||||
| @@ -18,7 +18,7 @@ SPDX-License-Identifier: AGPL-3.0-only | ||||
| import { ref } from 'vue'; | ||||
| import * as Misskey from 'misskey-js'; | ||||
| import MkMiniChart from '@/components/MkMiniChart.vue'; | ||||
| import * as os from '@/os.js'; | ||||
| import { misskeyApiGet } from '@/scripts/misskey-api.js'; | ||||
| import { getProxiedImageUrlNullable } from '@/scripts/media-proxy.js'; | ||||
|  | ||||
| const props = defineProps<{ | ||||
| @@ -27,7 +27,7 @@ const props = defineProps<{ | ||||
|  | ||||
| const chartValues = ref<number[] | null>(null); | ||||
|  | ||||
| os.apiGet('charts/instance', { host: props.instance.host, limit: 16 + 1, span: 'day' }).then(res => { | ||||
| misskeyApiGet('charts/instance', { host: props.instance.host, limit: 16 + 1, span: 'day' }).then(res => { | ||||
| 	// 今日のぶんの値はまだ途中の値であり、それも含めると大抵の場合前日よりも下降しているようなグラフになってしまうため今日は弾く | ||||
| 	res['requests.received'].splice(0, 1); | ||||
| 	chartValues.value = res['requests.received']; | ||||
|   | ||||
| @@ -90,6 +90,7 @@ import MkSelect from '@/components/MkSelect.vue'; | ||||
| import MkChart from '@/components/MkChart.vue'; | ||||
| import { useChartTooltip } from '@/scripts/use-chart-tooltip.js'; | ||||
| import * as os from '@/os.js'; | ||||
| import { misskeyApiGet } from '@/scripts/misskey-api.js'; | ||||
| import { i18n } from '@/i18n.js'; | ||||
| import MkHeatmap from '@/components/MkHeatmap.vue'; | ||||
| import MkFoldableSection from '@/components/MkFoldableSection.vue'; | ||||
| @@ -162,7 +163,7 @@ function createDoughnut(chartEl, tooltip, data) { | ||||
| } | ||||
|  | ||||
| onMounted(() => { | ||||
| 	os.apiGet('federation/stats', { limit: 30 }).then(fedStats => { | ||||
| 	misskeyApiGet('federation/stats', { limit: 30 }).then(fedStats => { | ||||
| 		createDoughnut(subDoughnutEl.value, externalTooltipHandler1, fedStats.topSubInstances.map(x => ({ | ||||
| 			name: x.host, | ||||
| 			color: x.themeColor, | ||||
|   | ||||
| @@ -164,6 +164,7 @@ import { checkWordMute } from '@/scripts/check-word-mute.js'; | ||||
| import { userPage } from '@/filters/user.js'; | ||||
| import * as os from '@/os.js'; | ||||
| import * as sound from '@/scripts/sound.js'; | ||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||
| import { defaultStore, noteViewInterruptors } from '@/store.js'; | ||||
| import { reactionPicker } from '@/scripts/reaction-picker.js'; | ||||
| import { extractUrlFromMfm } from '@/scripts/extract-url-from-mfm.js'; | ||||
| @@ -270,7 +271,7 @@ const keymap = { | ||||
| }; | ||||
|  | ||||
| provide('react', (reaction: string) => { | ||||
| 	os.api('notes/reactions/create', { | ||||
| 	misskeyApi('notes/reactions/create', { | ||||
| 		noteId: appearNote.value.id, | ||||
| 		reaction: reaction, | ||||
| 	}); | ||||
| @@ -291,7 +292,7 @@ if (props.mock) { | ||||
|  | ||||
| if (!props.mock) { | ||||
| 	useTooltip(renoteButton, async (showing) => { | ||||
| 		const renotes = await os.api('notes/renotes', { | ||||
| 		const renotes = await misskeyApi('notes/renotes', { | ||||
| 			noteId: appearNote.value.id, | ||||
| 			limit: 11, | ||||
| 		}); | ||||
| @@ -337,13 +338,13 @@ function react(viaKeyboard = false): void { | ||||
| 	pleaseLogin(); | ||||
| 	showMovedDialog(); | ||||
| 	if (appearNote.value.reactionAcceptance === 'likeOnly') { | ||||
| 		sound.play('reaction'); | ||||
| 		sound.playMisskeySfx('reaction'); | ||||
|  | ||||
| 		if (props.mock) { | ||||
| 			return; | ||||
| 		} | ||||
|  | ||||
| 		os.api('notes/reactions/create', { | ||||
| 		misskeyApi('notes/reactions/create', { | ||||
| 			noteId: appearNote.value.id, | ||||
| 			reaction: '❤️', | ||||
| 		}); | ||||
| @@ -357,14 +358,14 @@ function react(viaKeyboard = false): void { | ||||
| 	} else { | ||||
| 		blur(); | ||||
| 		reactionPicker.show(reactButton.value, reaction => { | ||||
| 			sound.play('reaction'); | ||||
| 			sound.playMisskeySfx('reaction'); | ||||
|  | ||||
| 			if (props.mock) { | ||||
| 				emit('reaction', reaction); | ||||
| 				return; | ||||
| 			} | ||||
|  | ||||
| 			os.api('notes/reactions/create', { | ||||
| 			misskeyApi('notes/reactions/create', { | ||||
| 				noteId: appearNote.value.id, | ||||
| 				reaction: reaction, | ||||
| 			}); | ||||
| @@ -386,7 +387,7 @@ function undoReact(note): void { | ||||
| 		return; | ||||
| 	} | ||||
|  | ||||
| 	os.api('notes/reactions/delete', { | ||||
| 	misskeyApi('notes/reactions/delete', { | ||||
| 		noteId: note.id, | ||||
| 	}); | ||||
| } | ||||
| @@ -446,7 +447,7 @@ function showRenoteMenu(viaKeyboard = false): void { | ||||
| 			icon: 'ti ti-trash', | ||||
| 			danger: true, | ||||
| 			action: () => { | ||||
| 				os.api('notes/delete', { | ||||
| 				misskeyApi('notes/delete', { | ||||
| 					noteId: note.value.id, | ||||
| 				}); | ||||
| 				isDeleted.value = true; | ||||
| @@ -492,7 +493,7 @@ function focusAfter() { | ||||
| } | ||||
|  | ||||
| function readPromo() { | ||||
| 	os.api('promo/read', { | ||||
| 	misskeyApi('promo/read', { | ||||
| 		noteId: appearNote.value.id, | ||||
| 	}); | ||||
| 	isDeleted.value = true; | ||||
|   | ||||
| @@ -210,6 +210,7 @@ import { checkWordMute } from '@/scripts/check-word-mute.js'; | ||||
| import { userPage } from '@/filters/user.js'; | ||||
| import { notePage } from '@/filters/note.js'; | ||||
| import * as os from '@/os.js'; | ||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||
| import * as sound from '@/scripts/sound.js'; | ||||
| import { defaultStore, noteViewInterruptors } from '@/store.js'; | ||||
| import { reactionPicker } from '@/scripts/reaction-picker.js'; | ||||
| @@ -292,7 +293,7 @@ const keymap = { | ||||
| }; | ||||
|  | ||||
| provide('react', (reaction: string) => { | ||||
| 	os.api('notes/reactions/create', { | ||||
| 	misskeyApi('notes/reactions/create', { | ||||
| 		noteId: appearNote.value.id, | ||||
| 		reaction: reaction, | ||||
| 	}); | ||||
| @@ -326,7 +327,7 @@ useNoteCapture({ | ||||
| }); | ||||
|  | ||||
| useTooltip(renoteButton, async (showing) => { | ||||
| 	const renotes = await os.api('notes/renotes', { | ||||
| 	const renotes = await misskeyApi('notes/renotes', { | ||||
| 		noteId: appearNote.value.id, | ||||
| 		limit: 11, | ||||
| 	}); | ||||
| @@ -369,9 +370,9 @@ function react(viaKeyboard = false): void { | ||||
| 	pleaseLogin(); | ||||
| 	showMovedDialog(); | ||||
| 	if (appearNote.value.reactionAcceptance === 'likeOnly') { | ||||
| 		sound.play('reaction'); | ||||
| 		sound.playMisskeySfx('reaction'); | ||||
|  | ||||
| 		os.api('notes/reactions/create', { | ||||
| 		misskeyApi('notes/reactions/create', { | ||||
| 			noteId: appearNote.value.id, | ||||
| 			reaction: '❤️', | ||||
| 		}); | ||||
| @@ -385,9 +386,9 @@ function react(viaKeyboard = false): void { | ||||
| 	} else { | ||||
| 		blur(); | ||||
| 		reactionPicker.show(reactButton.value, reaction => { | ||||
| 			sound.play('reaction'); | ||||
| 			sound.playMisskeySfx('reaction'); | ||||
|  | ||||
| 			os.api('notes/reactions/create', { | ||||
| 			misskeyApi('notes/reactions/create', { | ||||
| 				noteId: appearNote.value.id, | ||||
| 				reaction: reaction, | ||||
| 			}); | ||||
| @@ -403,7 +404,7 @@ function react(viaKeyboard = false): void { | ||||
| function undoReact(note): void { | ||||
| 	const oldReaction = note.myReaction; | ||||
| 	if (!oldReaction) return; | ||||
| 	os.api('notes/reactions/delete', { | ||||
| 	misskeyApi('notes/reactions/delete', { | ||||
| 		noteId: note.id, | ||||
| 	}); | ||||
| } | ||||
| @@ -446,7 +447,7 @@ function showRenoteMenu(viaKeyboard = false): void { | ||||
| 		icon: 'ti ti-trash', | ||||
| 		danger: true, | ||||
| 		action: () => { | ||||
| 			os.api('notes/delete', { | ||||
| 			misskeyApi('notes/delete', { | ||||
| 				noteId: note.value.id, | ||||
| 			}); | ||||
| 			isDeleted.value = true; | ||||
| @@ -468,7 +469,7 @@ const repliesLoaded = ref(false); | ||||
|  | ||||
| function loadReplies() { | ||||
| 	repliesLoaded.value = true; | ||||
| 	os.api('notes/children', { | ||||
| 	misskeyApi('notes/children', { | ||||
| 		noteId: appearNote.value.id, | ||||
| 		limit: 30, | ||||
| 	}).then(res => { | ||||
| @@ -480,7 +481,7 @@ const conversationLoaded = ref(false); | ||||
|  | ||||
| function loadConversation() { | ||||
| 	conversationLoaded.value = true; | ||||
| 	os.api('notes/conversation', { | ||||
| 	misskeyApi('notes/conversation', { | ||||
| 		noteId: appearNote.value.replyId, | ||||
| 	}).then(res => { | ||||
| 		conversation.value = res.reverse(); | ||||
|   | ||||
| @@ -46,7 +46,7 @@ import MkNoteHeader from '@/components/MkNoteHeader.vue'; | ||||
| import MkSubNoteContent from '@/components/MkSubNoteContent.vue'; | ||||
| import MkCwButton from '@/components/MkCwButton.vue'; | ||||
| import { notePage } from '@/filters/note.js'; | ||||
| import * as os from '@/os.js'; | ||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||
| import { i18n } from '@/i18n.js'; | ||||
| import { $i } from '@/account.js'; | ||||
| import { userPage } from '@/filters/user.js'; | ||||
| @@ -68,7 +68,7 @@ const showContent = ref(false); | ||||
| const replies = ref<Misskey.entities.Note[]>([]); | ||||
|  | ||||
| if (props.detail) { | ||||
| 	os.api('notes/children', { | ||||
| 	misskeyApi('notes/children', { | ||||
| 		noteId: props.note.id, | ||||
| 		limit: 5, | ||||
| 	}).then(res => { | ||||
|   | ||||
| @@ -145,7 +145,7 @@ import { getNoteSummary } from '@/scripts/get-note-summary.js'; | ||||
| import { notePage } from '@/filters/note.js'; | ||||
| import { userPage } from '@/filters/user.js'; | ||||
| import { i18n } from '@/i18n.js'; | ||||
| import * as os from '@/os.js'; | ||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||
| import { $i } from '@/account.js'; | ||||
| import { infoImageUrl } from '@/instance.js'; | ||||
|  | ||||
| @@ -162,12 +162,12 @@ const followRequestDone = ref(false); | ||||
|  | ||||
| const acceptFollowRequest = () => { | ||||
| 	followRequestDone.value = true; | ||||
| 	os.api('following/requests/accept', { userId: props.notification.user.id }); | ||||
| 	misskeyApi('following/requests/accept', { userId: props.notification.user.id }); | ||||
| }; | ||||
|  | ||||
| const rejectFollowRequest = () => { | ||||
| 	followRequestDone.value = true; | ||||
| 	os.api('following/requests/reject', { userId: props.notification.user.id }); | ||||
| 	misskeyApi('following/requests/reject', { userId: props.notification.user.id }); | ||||
| }; | ||||
| </script> | ||||
|  | ||||
|   | ||||
| @@ -9,7 +9,6 @@ SPDX-License-Identifier: AGPL-3.0-only | ||||
|  | ||||
| <script lang="ts" setup> | ||||
| import { reactive, watch } from 'vue'; | ||||
| import gsap from 'gsap'; | ||||
| import number from '@/filters/number.js'; | ||||
|  | ||||
| const props = defineProps<{ | ||||
| @@ -20,8 +19,24 @@ const tweened = reactive({ | ||||
| 	number: 0, | ||||
| }); | ||||
|  | ||||
| watch(() => props.value, (n) => { | ||||
| 	gsap.to(tweened, { duration: 1, number: Number(n) || 0 }); | ||||
| watch(() => props.value, (to, from) => { | ||||
| 	// requestAnimationFrameを利用して、500msでfromからtoまでを1次関数的に変化させる | ||||
| 	let start: number | null = null; | ||||
|  | ||||
| 	function step(timestamp: number) { | ||||
| 		if (start === null) { | ||||
| 			start = timestamp; | ||||
| 		} | ||||
| 		const elapsed = timestamp - start; | ||||
| 		tweened.number = (from ?? 0) + (to - (from ?? 0)) * elapsed / 500; | ||||
| 		if (elapsed < 500) { | ||||
| 			window.requestAnimationFrame(step); | ||||
| 		} else { | ||||
| 			tweened.number = to; | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	window.requestAnimationFrame(step); | ||||
| }, { | ||||
| 	immediate: true, | ||||
| }); | ||||
|   | ||||
| @@ -23,26 +23,26 @@ SPDX-License-Identifier: AGPL-3.0-only | ||||
| 	</template> | ||||
|  | ||||
| 	<div ref="contents" :class="$style.root" style="container-type: inline-size;"> | ||||
| 		<RouterView :key="reloadCount" :router="router"/> | ||||
| 		<RouterView :key="reloadCount" :router="windowRouter"/> | ||||
| 	</div> | ||||
| </MkWindow> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts" setup> | ||||
| import { ComputedRef, onMounted, onUnmounted, provide, shallowRef, ref, computed } from 'vue'; | ||||
| import { computed, ComputedRef, onMounted, onUnmounted, provide, ref, shallowRef } from 'vue'; | ||||
| import RouterView from '@/components/global/RouterView.vue'; | ||||
| import MkWindow from '@/components/MkWindow.vue'; | ||||
| import { popout as _popout } from '@/scripts/popout.js'; | ||||
| import copyToClipboard from '@/scripts/copy-to-clipboard.js'; | ||||
| import { url } from '@/config.js'; | ||||
| import { mainRouter, routes, page } from '@/router.js'; | ||||
| import { $i } from '@/account.js'; | ||||
| import { Router, useScrollPositionManager } from '@/nirax.js'; | ||||
| import { useScrollPositionManager } from '@/nirax.js'; | ||||
| import { i18n } from '@/i18n.js'; | ||||
| import { PageMetadata, provideMetadataReceiver } from '@/scripts/page-metadata.js'; | ||||
| import { openingWindowsCount } from '@/os.js'; | ||||
| import { claimAchievement } from '@/scripts/achievements.js'; | ||||
| import { getScrollContainer } from '@/scripts/scroll.js'; | ||||
| import { useRouterFactory } from '@/global/router/supplier.js'; | ||||
| import { mainRouter } from '@/global/router/main.js'; | ||||
|  | ||||
| const props = defineProps<{ | ||||
| 	initialPath: string; | ||||
| @@ -52,14 +52,15 @@ defineEmits<{ | ||||
| 	(ev: 'closed'): void; | ||||
| }>(); | ||||
|  | ||||
| const router = new Router(routes, props.initialPath, !!$i, page(() => import('@/pages/not-found.vue'))); | ||||
| const routerFactory = useRouterFactory(); | ||||
| const windowRouter = routerFactory(props.initialPath); | ||||
|  | ||||
| const contents = shallowRef<HTMLElement>(); | ||||
| const pageMetadata = ref<null | ComputedRef<PageMetadata>>(); | ||||
| const windowEl = shallowRef<InstanceType<typeof MkWindow>>(); | ||||
| const history = ref<{ path: string; key: any; }[]>([{ | ||||
| 	path: router.getCurrentPath(), | ||||
| 	key: router.getCurrentKey(), | ||||
| 	path: windowRouter.getCurrentPath(), | ||||
| 	key: windowRouter.getCurrentKey(), | ||||
| }]); | ||||
| const buttonsLeft = computed(() => { | ||||
| 	const buttons = []; | ||||
| @@ -88,11 +89,11 @@ const buttonsRight = computed(() => { | ||||
| }); | ||||
| const reloadCount = ref(0); | ||||
|  | ||||
| router.addListener('push', ctx => { | ||||
| windowRouter.addListener('push', ctx => { | ||||
| 	history.value.push({ path: ctx.path, key: ctx.key }); | ||||
| }); | ||||
|  | ||||
| provide('router', router); | ||||
| provide('router', windowRouter); | ||||
| provideMetadataReceiver((info) => { | ||||
| 	pageMetadata.value = info; | ||||
| }); | ||||
| @@ -112,20 +113,20 @@ const contextmenu = computed(() => ([{ | ||||
| 	icon: 'ti ti-external-link', | ||||
| 	text: i18n.ts.openInNewTab, | ||||
| 	action: () => { | ||||
| 		window.open(url + router.getCurrentPath(), '_blank', 'noopener'); | ||||
| 		window.open(url + windowRouter.getCurrentPath(), '_blank', 'noopener'); | ||||
| 		windowEl.value.close(); | ||||
| 	}, | ||||
| }, { | ||||
| 	icon: 'ti ti-link', | ||||
| 	text: i18n.ts.copyLink, | ||||
| 	action: () => { | ||||
| 		copyToClipboard(url + router.getCurrentPath()); | ||||
| 		copyToClipboard(url + windowRouter.getCurrentPath()); | ||||
| 	}, | ||||
| }])); | ||||
|  | ||||
| function back() { | ||||
| 	history.value.pop(); | ||||
| 	router.replace(history.value.at(-1)!.path, history.value.at(-1)!.key); | ||||
| 	windowRouter.replace(history.value.at(-1)!.path, history.value.at(-1)!.key); | ||||
| } | ||||
|  | ||||
| function reload() { | ||||
| @@ -137,16 +138,16 @@ function close() { | ||||
| } | ||||
|  | ||||
| function expand() { | ||||
| 	mainRouter.push(router.getCurrentPath(), 'forcePage'); | ||||
| 	mainRouter.push(windowRouter.getCurrentPath(), 'forcePage'); | ||||
| 	windowEl.value.close(); | ||||
| } | ||||
|  | ||||
| function popout() { | ||||
| 	_popout(router.getCurrentPath(), windowEl.value.$el); | ||||
| 	_popout(windowRouter.getCurrentPath(), windowEl.value.$el); | ||||
| 	windowEl.value.close(); | ||||
| } | ||||
|  | ||||
| useScrollPositionManager(() => getScrollContainer(contents.value), router); | ||||
| useScrollPositionManager(() => getScrollContainer(contents.value), windowRouter); | ||||
|  | ||||
| onMounted(() => { | ||||
| 	openingWindowsCount.value++; | ||||
|   | ||||
| @@ -46,6 +46,7 @@ SPDX-License-Identifier: AGPL-3.0-only | ||||
| import { computed, ComputedRef, isRef, nextTick, onActivated, onBeforeMount, onBeforeUnmount, onDeactivated, ref, shallowRef, watch } from 'vue'; | ||||
| import * as Misskey from 'misskey-js'; | ||||
| import * as os from '@/os.js'; | ||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||
| import { onScrollTop, isTopVisible, getBodyScrollHeight, getScrollContainer, onScrollBottom, scrollToBottom, scroll, isBottomVisible } from '@/scripts/scroll.js'; | ||||
| import { useDocumentVisibility } from '@/scripts/use-document-visibility.js'; | ||||
| import { defaultStore } from '@/store.js'; | ||||
| @@ -173,7 +174,7 @@ async function init(): Promise<void> { | ||||
| 	queue.value = []; | ||||
| 	fetching.value = true; | ||||
| 	const params = props.pagination.params ? isRef(props.pagination.params) ? props.pagination.params.value : props.pagination.params : {}; | ||||
| 	await os.api(props.pagination.endpoint, { | ||||
| 	await misskeyApi(props.pagination.endpoint, { | ||||
| 		...params, | ||||
| 		limit: props.pagination.limit ?? 10, | ||||
| 		allowPartial: true, | ||||
| @@ -210,7 +211,7 @@ const fetchMore = async (): Promise<void> => { | ||||
| 	if (!more.value || fetching.value || moreFetching.value || items.value.length === 0) return; | ||||
| 	moreFetching.value = true; | ||||
| 	const params = props.pagination.params ? isRef(props.pagination.params) ? props.pagination.params.value : props.pagination.params : {}; | ||||
| 	await os.api(props.pagination.endpoint, { | ||||
| 	await misskeyApi(props.pagination.endpoint, { | ||||
| 		...params, | ||||
| 		limit: SECOND_FETCH_LIMIT, | ||||
| 		...(props.pagination.offsetMode ? { | ||||
| @@ -274,7 +275,7 @@ const fetchMoreAhead = async (): Promise<void> => { | ||||
| 	if (!more.value || fetching.value || moreFetching.value || items.value.length === 0) return; | ||||
| 	moreFetching.value = true; | ||||
| 	const params = props.pagination.params ? isRef(props.pagination.params) ? props.pagination.params.value : props.pagination.params : {}; | ||||
| 	await os.api(props.pagination.endpoint, { | ||||
| 	await misskeyApi(props.pagination.endpoint, { | ||||
| 		...params, | ||||
| 		limit: SECOND_FETCH_LIMIT, | ||||
| 		...(props.pagination.offsetMode ? { | ||||
|   | ||||
| @@ -41,7 +41,9 @@ import MkInput from '@/components/MkInput.vue'; | ||||
| import MkButton from '@/components/MkButton.vue'; | ||||
| import MkModalWindow from '@/components/MkModalWindow.vue'; | ||||
| import { i18n } from '@/i18n.js'; | ||||
| import { $i } from '@/account.js'; | ||||
| import { signinRequired } from '@/account.js'; | ||||
|  | ||||
| const $i = signinRequired(); | ||||
|  | ||||
| const emit = defineEmits<{ | ||||
| 	(ev: 'done', v: { password: string; token: string | null; }): void; | ||||
|   | ||||
| @@ -32,11 +32,13 @@ import * as Misskey from 'misskey-js'; | ||||
| import { sum } from '@/scripts/array.js'; | ||||
| import { pleaseLogin } from '@/scripts/please-login.js'; | ||||
| import * as os from '@/os.js'; | ||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||
| import { i18n } from '@/i18n.js'; | ||||
| import { useInterval } from '@/scripts/use-interval.js'; | ||||
| import { WithNonNullable } from '@/type.js'; | ||||
|  | ||||
| const props = defineProps<{ | ||||
| 	note: Misskey.entities.Note; | ||||
| 	note: WithNonNullable<Misskey.entities.Note, 'poll'>; | ||||
| 	readOnly?: boolean; | ||||
| }>(); | ||||
|  | ||||
| @@ -83,7 +85,7 @@ const vote = async (id) => { | ||||
| 	}); | ||||
| 	if (canceled) return; | ||||
|  | ||||
| 	await os.api('notes/polls/vote', { | ||||
| 	await misskeyApi('notes/polls/vote', { | ||||
| 		noteId: props.note.id, | ||||
| 		choice: id, | ||||
| 	}); | ||||
|   | ||||
| @@ -116,12 +116,13 @@ import { extractMentions } from '@/scripts/extract-mentions.js'; | ||||
| import { formatTimeString } from '@/scripts/format-time-string.js'; | ||||
| import { Autocomplete } from '@/scripts/autocomplete.js'; | ||||
| import * as os from '@/os.js'; | ||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||
| import { selectFiles } from '@/scripts/select-file.js'; | ||||
| import { defaultStore, notePostInterruptors, postFormActions } from '@/store.js'; | ||||
| import MkInfo from '@/components/MkInfo.vue'; | ||||
| import { i18n } from '@/i18n.js'; | ||||
| import { instance } from '@/instance.js'; | ||||
| import { $i, notesCount, incNotesCount, getAccounts, openAccountMenu as openAccountMenu_ } from '@/account.js'; | ||||
| import { signinRequired, notesCount, incNotesCount, getAccounts, openAccountMenu as openAccountMenu_ } from '@/account.js'; | ||||
| import { uploadFile } from '@/scripts/upload.js'; | ||||
| import { deepClone } from '@/scripts/clone.js'; | ||||
| import MkRippleEffect from '@/components/MkRippleEffect.vue'; | ||||
| @@ -130,6 +131,8 @@ import { claimAchievement } from '@/scripts/achievements.js'; | ||||
| import { emojiPicker } from '@/scripts/emoji-picker.js'; | ||||
| import { mfmFunctionPicker } from '@/scripts/mfm-function-picker.js'; | ||||
|  | ||||
| const $i = signinRequired(); | ||||
|  | ||||
| const modal = inject('modal'); | ||||
|  | ||||
| const props = withDefaults(defineProps<{ | ||||
| @@ -312,7 +315,7 @@ if (props.reply && props.reply.text != null) { | ||||
| 	} | ||||
| } | ||||
|  | ||||
| if ($i?.isSilenced && visibility.value === 'public') { | ||||
| if ($i.isSilenced && visibility.value === 'public') { | ||||
| 	visibility.value = 'home'; | ||||
| } | ||||
|  | ||||
| @@ -333,7 +336,7 @@ if (props.reply && ['home', 'followers', 'specified'].includes(props.reply.visib | ||||
|  | ||||
| 	if (visibility.value === 'specified') { | ||||
| 		if (props.reply.visibleUserIds) { | ||||
| 			os.api('users/show', { | ||||
| 			misskeyApi('users/show', { | ||||
| 				userIds: props.reply.visibleUserIds.filter(uid => uid !== $i.id && uid !== props.reply.userId), | ||||
| 			}).then(users => { | ||||
| 				users.forEach(pushVisibleUser); | ||||
| @@ -341,7 +344,7 @@ if (props.reply && ['home', 'followers', 'specified'].includes(props.reply.visib | ||||
| 		} | ||||
|  | ||||
| 		if (props.reply.userId !== $i.id) { | ||||
| 			os.api('users/show', { userId: props.reply.userId }).then(user => { | ||||
| 			misskeyApi('users/show', { userId: props.reply.userId }).then(user => { | ||||
| 				pushVisibleUser(user); | ||||
| 			}); | ||||
| 		} | ||||
| @@ -388,7 +391,7 @@ function addMissingMention() { | ||||
|  | ||||
| 	for (const x of extractMentions(ast)) { | ||||
| 		if (!visibleUsers.value.some(u => (u.username === x.username) && (u.host === x.host))) { | ||||
| 			os.api('users/show', { username: x.username, host: x.host }).then(user => { | ||||
| 			misskeyApi('users/show', { username: x.username, host: x.host }).then(user => { | ||||
| 				visibleUsers.value.push(user); | ||||
| 			}); | ||||
| 		} | ||||
| @@ -465,7 +468,7 @@ function setVisibility() { | ||||
|  | ||||
| 	os.popup(defineAsyncComponent(() => import('@/components/MkVisibilityPicker.vue')), { | ||||
| 		currentVisibility: visibility.value, | ||||
| 		isSilenced: $i?.isSilenced, | ||||
| 		isSilenced: $i.isSilenced, | ||||
| 		localOnly: localOnly.value, | ||||
| 		src: visibilityButton.value, | ||||
| 	}, { | ||||
| @@ -757,7 +760,17 @@ async function post(ev?: MouseEvent) { | ||||
|  | ||||
| 	if (withHashtags.value && hashtags.value && hashtags.value.trim() !== '') { | ||||
| 		const hashtags_ = hashtags.value.trim().split(' ').map(x => x.startsWith('#') ? x : '#' + x).join(' '); | ||||
| 		postData.text = postData.text ? `${postData.text} ${hashtags_}` : hashtags_; | ||||
| 		if (!postData.text) { | ||||
| 			postData.text = hashtags_; | ||||
| 		} else { | ||||
| 			const postTextLines = postData.text.split('\n'); | ||||
| 			if (postTextLines[postTextLines.length - 1].trim() === '') { | ||||
| 				postTextLines[postTextLines.length - 1] += hashtags_; | ||||
| 			} else { | ||||
| 				postTextLines[postTextLines.length - 1] += ' ' + hashtags_; | ||||
| 			} | ||||
| 			postData.text = postTextLines.join('\n'); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	// plugin | ||||
| @@ -779,7 +792,7 @@ async function post(ev?: MouseEvent) { | ||||
| 	} | ||||
|  | ||||
| 	posting.value = true; | ||||
| 	os.api('notes/create', postData, token).then(() => { | ||||
| 	misskeyApi('notes/create', postData, token).then(() => { | ||||
| 		if (props.freezeAfterPosted) { | ||||
| 			posted.value = true; | ||||
| 		} else { | ||||
|   | ||||
| @@ -24,6 +24,7 @@ import { defineAsyncComponent, inject } from 'vue'; | ||||
| import * as Misskey from 'misskey-js'; | ||||
| import MkDriveFileThumbnail from '@/components/MkDriveFileThumbnail.vue'; | ||||
| import * as os from '@/os.js'; | ||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||
| import { i18n } from '@/i18n.js'; | ||||
|  | ||||
| const Sortable = defineAsyncComponent(() => import('vuedraggable').then(x => x.default)); | ||||
| @@ -61,7 +62,7 @@ function toggleSensitive(file) { | ||||
| 		return; | ||||
| 	} | ||||
|  | ||||
| 	os.api('drive/files/update', { | ||||
| 	misskeyApi('drive/files/update', { | ||||
| 		fileId: file.id, | ||||
| 		isSensitive: !file.isSensitive, | ||||
| 	}).then(() => { | ||||
| @@ -78,7 +79,7 @@ async function rename(file) { | ||||
| 		allowEmpty: false, | ||||
| 	}); | ||||
| 	if (canceled) return; | ||||
| 	os.api('drive/files/update', { | ||||
| 	misskeyApi('drive/files/update', { | ||||
| 		fileId: file.id, | ||||
| 		name: result, | ||||
| 	}).then(() => { | ||||
| @@ -96,7 +97,7 @@ async function describe(file) { | ||||
| 	}, { | ||||
| 		done: caption => { | ||||
| 			let comment = caption.length === 0 ? null : caption; | ||||
| 			os.api('drive/files/update', { | ||||
| 			misskeyApi('drive/files/update', { | ||||
| 				fileId: file.id, | ||||
| 				comment: comment, | ||||
| 			}).then(() => { | ||||
|   | ||||
| @@ -45,7 +45,8 @@ import { ref } from 'vue'; | ||||
| import { $i, getAccounts } from '@/account.js'; | ||||
| import MkButton from '@/components/MkButton.vue'; | ||||
| import { instance } from '@/instance.js'; | ||||
| import { api, apiWithDialog, promiseDialog } from '@/os.js'; | ||||
| import { apiWithDialog, promiseDialog } from '@/os.js'; | ||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||
| import { i18n } from '@/i18n.js'; | ||||
|  | ||||
| defineProps<{ | ||||
| @@ -82,7 +83,7 @@ function subscribe() { | ||||
| 			pushSubscription.value = subscription; | ||||
|  | ||||
| 			// Register | ||||
| 			pushRegistrationInServer.value = await api('sw/register', { | ||||
| 			pushRegistrationInServer.value = await misskeyApi('sw/register', { | ||||
| 				endpoint: subscription.endpoint, | ||||
| 				auth: encode(subscription.getKey('auth')), | ||||
| 				publickey: encode(subscription.getKey('p256dh')), | ||||
| @@ -159,7 +160,7 @@ if (navigator.serviceWorker == null) { | ||||
| 			supported.value = true; | ||||
|  | ||||
| 			if (pushSubscription.value) { | ||||
| 				const res = await api('sw/show-registration', { | ||||
| 				const res = await misskeyApi('sw/show-registration', { | ||||
| 					endpoint: pushSubscription.value.endpoint, | ||||
| 				}); | ||||
|  | ||||
|   | ||||
| @@ -43,6 +43,7 @@ const props = withDefaults(defineProps<{ | ||||
|  | ||||
| const emit = defineEmits<{ | ||||
| 	(ev: 'update:modelValue', value: number): void; | ||||
| 	(ev: 'dragEnded', value: number): void; | ||||
| }>(); | ||||
|  | ||||
| const containerEl = shallowRef<HTMLElement>(); | ||||
| @@ -143,6 +144,7 @@ const onMousedown = (ev: MouseEvent | TouchEvent) => { | ||||
| 		// 値が変わってたら通知 | ||||
| 		if (beforeValue !== finalValue.value) { | ||||
| 			emit('update:modelValue', finalValue.value); | ||||
| 			emit('dragEnded', finalValue.value); | ||||
| 		} | ||||
| 	}; | ||||
|  | ||||
|   | ||||
| @@ -22,6 +22,7 @@ import * as Misskey from 'misskey-js'; | ||||
| import XDetails from '@/components/MkReactionsViewer.details.vue'; | ||||
| import MkReactionIcon from '@/components/MkReactionIcon.vue'; | ||||
| import * as os from '@/os.js'; | ||||
| import { misskeyApi, misskeyApiGet } from '@/scripts/misskey-api.js'; | ||||
| import { useTooltip } from '@/scripts/use-tooltip.js'; | ||||
| import { $i } from '@/account.js'; | ||||
| import MkReactionEffect from '@/components/MkReactionEffect.vue'; | ||||
| @@ -61,7 +62,7 @@ async function toggleReaction() { | ||||
| 		if (confirm.canceled) return; | ||||
|  | ||||
| 		if (oldReaction !== props.reaction) { | ||||
| 			sound.play('reaction'); | ||||
| 			sound.playMisskeySfx('reaction'); | ||||
| 		} | ||||
|  | ||||
| 		if (mock) { | ||||
| @@ -69,25 +70,25 @@ async function toggleReaction() { | ||||
| 			return; | ||||
| 		} | ||||
|  | ||||
| 		os.api('notes/reactions/delete', { | ||||
| 		misskeyApi('notes/reactions/delete', { | ||||
| 			noteId: props.note.id, | ||||
| 		}).then(() => { | ||||
| 			if (oldReaction !== props.reaction) { | ||||
| 				os.api('notes/reactions/create', { | ||||
| 				misskeyApi('notes/reactions/create', { | ||||
| 					noteId: props.note.id, | ||||
| 					reaction: props.reaction, | ||||
| 				}); | ||||
| 			} | ||||
| 		}); | ||||
| 	} else { | ||||
| 		sound.play('reaction'); | ||||
| 		sound.playMisskeySfx('reaction'); | ||||
|  | ||||
| 		if (mock) { | ||||
| 			emit('reactionToggled', props.reaction, (props.count + 1)); | ||||
| 			return; | ||||
| 		} | ||||
|  | ||||
| 		os.api('notes/reactions/create', { | ||||
| 		misskeyApi('notes/reactions/create', { | ||||
| 			noteId: props.note.id, | ||||
| 			reaction: props.reaction, | ||||
| 		}); | ||||
| @@ -117,7 +118,7 @@ onMounted(() => { | ||||
|  | ||||
| if (!mock) { | ||||
| 	useTooltip(buttonEl, async (showing) => { | ||||
| 		const reactions = await os.apiGet('notes/reactions', { | ||||
| 		const reactions = await misskeyApiGet('notes/reactions', { | ||||
| 			noteId: props.note.id, | ||||
| 			type: props.reaction, | ||||
| 			limit: 10, | ||||
|   | ||||
| @@ -15,7 +15,7 @@ SPDX-License-Identifier: AGPL-3.0-only | ||||
| <script lang="ts" setup> | ||||
| import { onMounted, nextTick, shallowRef, ref } from 'vue'; | ||||
| import { Chart } from 'chart.js'; | ||||
| import * as os from '@/os.js'; | ||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||
| import { defaultStore } from '@/store.js'; | ||||
| import { useChartTooltip } from '@/scripts/use-chart-tooltip.js'; | ||||
| import { alpha } from '@/scripts/color.js'; | ||||
| @@ -43,7 +43,7 @@ async function renderChart() { | ||||
|  | ||||
| 	const maxDays = wide ? 10 : narrow ? 5 : 7; | ||||
|  | ||||
| 	let raw = await os.api('retention', { }); | ||||
| 	let raw = await misskeyApi('retention', { }); | ||||
|  | ||||
| 	raw = raw.slice(0, maxDays + 1); | ||||
|  | ||||
|   | ||||
| @@ -16,7 +16,7 @@ import { useChartTooltip } from '@/scripts/use-chart-tooltip.js'; | ||||
| import { chartVLine } from '@/scripts/chart-vline.js'; | ||||
| import { alpha } from '@/scripts/color.js'; | ||||
| import { initChart } from '@/scripts/init-chart.js'; | ||||
| import * as os from '@/os.js'; | ||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||
|  | ||||
| initChart(); | ||||
|  | ||||
| @@ -40,7 +40,7 @@ const getDate = (ymd: string) => { | ||||
| }; | ||||
|  | ||||
| onMounted(async () => { | ||||
| 	let raw = await os.api('retention', { }); | ||||
| 	let raw = await misskeyApi('retention', { }); | ||||
|  | ||||
| 	const vLineColor = defaultStore.state.darkMode ? 'rgba(255, 255, 255, 0.2)' : 'rgba(0, 0, 0, 0.2)'; | ||||
|  | ||||
|   | ||||
| @@ -77,7 +77,14 @@ const emit = defineEmits<{ | ||||
| 	(ev: 'end'): void; | ||||
| }>(); | ||||
|  | ||||
| const particles = []; | ||||
| const particles: { | ||||
| 	size: number; | ||||
| 	xA: number; | ||||
| 	yA: number; | ||||
| 	xB: number; | ||||
| 	yB: number; | ||||
| 	color: string; | ||||
| }[] = []; | ||||
| const origin = 64; | ||||
| const colors = ['#FF1493', '#00FFFF', '#FFE202']; | ||||
| const zIndex = os.claimZIndex('high'); | ||||
|   | ||||
| @@ -59,6 +59,7 @@ import MkInput from '@/components/MkInput.vue'; | ||||
| import MkInfo from '@/components/MkInfo.vue'; | ||||
| import { host as configHost } from '@/config.js'; | ||||
| import * as os from '@/os.js'; | ||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||
| import { login } from '@/account.js'; | ||||
| import { i18n } from '@/i18n.js'; | ||||
|  | ||||
| @@ -95,7 +96,7 @@ const props = defineProps({ | ||||
| }); | ||||
|  | ||||
| function onUsernameChange(): void { | ||||
| 	os.api('users/show', { | ||||
| 	misskeyApi('users/show', { | ||||
| 		username: username.value, | ||||
| 	}).then(userResponse => { | ||||
| 		user.value = userResponse; | ||||
| @@ -120,7 +121,7 @@ async function queryKey(): Promise<void> { | ||||
| 			credentialRequest.value = null; | ||||
| 			queryingKey.value = false; | ||||
| 			signing.value = true; | ||||
| 			return os.api('signin', { | ||||
| 			return misskeyApi('signin', { | ||||
| 				username: username.value, | ||||
| 				password: password.value, | ||||
| 				credential: credential.toJSON(), | ||||
| @@ -142,7 +143,7 @@ function onSubmit(): void { | ||||
| 	signing.value = true; | ||||
| 	if (!totpLogin.value && user.value && user.value.twoFactorEnabled) { | ||||
| 		if (webAuthnSupported() && user.value.securityKeys) { | ||||
| 			os.api('signin', { | ||||
| 			misskeyApi('signin', { | ||||
| 				username: username.value, | ||||
| 				password: password.value, | ||||
| 			}).then(res => { | ||||
| @@ -159,7 +160,7 @@ function onSubmit(): void { | ||||
| 			signing.value = false; | ||||
| 		} | ||||
| 	} else { | ||||
| 		os.api('signin', { | ||||
| 		misskeyApi('signin', { | ||||
| 			username: username.value, | ||||
| 			password: password.value, | ||||
| 			token: user.value?.twoFactorEnabled ? token.value : undefined, | ||||
|   | ||||
| @@ -63,6 +63,7 @@ SPDX-License-Identifier: AGPL-3.0-only | ||||
| 				</template> | ||||
| 			</MkInput> | ||||
| 			<MkCaptcha v-if="instance.enableHcaptcha" ref="hcaptcha" v-model="hCaptchaResponse" :class="$style.captcha" provider="hcaptcha" :sitekey="instance.hcaptchaSiteKey"/> | ||||
| 			<MkCaptcha v-if="instance.enableMcaptcha" ref="mcaptcha" v-model="mCaptchaResponse" :class="$style.captcha" provider="mcaptcha" :sitekey="instance.mcaptchaSiteKey" :instanceUrl="instance.mcaptchaInstanceUrl"/> | ||||
| 			<MkCaptcha v-if="instance.enableRecaptcha" ref="recaptcha" v-model="reCaptchaResponse" :class="$style.captcha" provider="recaptcha" :sitekey="instance.recaptchaSiteKey"/> | ||||
| 			<MkCaptcha v-if="instance.enableTurnstile" ref="turnstile" v-model="turnstileResponse" :class="$style.captcha" provider="turnstile" :sitekey="instance.turnstileSiteKey"/> | ||||
| 			<MkButton type="submit" :disabled="shouldDisableSubmitting" large gradate rounded data-cy-signup-submit style="margin: 0 auto;"> | ||||
| @@ -84,6 +85,7 @@ import MkInput from './MkInput.vue'; | ||||
| import MkCaptcha, { type Captcha } from '@/components/MkCaptcha.vue'; | ||||
| import * as config from '@/config.js'; | ||||
| import * as os from '@/os.js'; | ||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||
| import { login } from '@/account.js'; | ||||
| import { instance } from '@/instance.js'; | ||||
| import { i18n } from '@/i18n.js'; | ||||
| @@ -116,6 +118,7 @@ const passwordStrength = ref<'' | 'low' | 'medium' | 'high'>(''); | ||||
| const passwordRetypeState = ref<null | 'match' | 'not-match'>(null); | ||||
| const submitting = ref<boolean>(false); | ||||
| const hCaptchaResponse = ref<string | null>(null); | ||||
| const mCaptchaResponse = ref<string | null>(null); | ||||
| const reCaptchaResponse = ref<string | null>(null); | ||||
| const turnstileResponse = ref<string | null>(null); | ||||
| const usernameAbortController = ref<null | AbortController>(null); | ||||
| @@ -124,6 +127,7 @@ const emailAbortController = ref<null | AbortController>(null); | ||||
| const shouldDisableSubmitting = computed((): boolean => { | ||||
| 	return submitting.value || | ||||
| 		instance.enableHcaptcha && !hCaptchaResponse.value || | ||||
| 		instance.enableMcaptcha && !mCaptchaResponse.value || | ||||
| 		instance.enableRecaptcha && !reCaptchaResponse.value || | ||||
| 		instance.enableTurnstile && !turnstileResponse.value || | ||||
| 		instance.emailRequiredForSignup && emailState.value !== 'ok' || | ||||
| @@ -180,7 +184,7 @@ function onChangeUsername(): void { | ||||
| 	usernameState.value = 'wait'; | ||||
| 	usernameAbortController.value = new AbortController(); | ||||
|  | ||||
| 	os.api('username/available', { | ||||
| 	misskeyApi('username/available', { | ||||
| 		username: username.value, | ||||
| 	}, undefined, usernameAbortController.value.signal).then(result => { | ||||
| 		usernameState.value = result.available ? 'ok' : 'unavailable'; | ||||
| @@ -203,7 +207,7 @@ function onChangeEmail(): void { | ||||
| 	emailState.value = 'wait'; | ||||
| 	emailAbortController.value = new AbortController(); | ||||
|  | ||||
| 	os.api('email-address/available', { | ||||
| 	misskeyApi('email-address/available', { | ||||
| 		emailAddress: email.value, | ||||
| 	}, undefined, emailAbortController.value.signal).then(result => { | ||||
| 		emailState.value = result.available ? 'ok' : | ||||
| @@ -245,12 +249,13 @@ async function onSubmit(): Promise<void> { | ||||
| 	submitting.value = true; | ||||
|  | ||||
| 	try { | ||||
| 		await os.api('signup', { | ||||
| 		await misskeyApi('signup', { | ||||
| 			username: username.value, | ||||
| 			password: password.value, | ||||
| 			emailAddress: email.value, | ||||
| 			invitationCode: invitationCode.value, | ||||
| 			'hcaptcha-response': hCaptchaResponse.value, | ||||
| 			'm-captcha-response': mCaptchaResponse.value, | ||||
| 			'g-recaptcha-response': reCaptchaResponse.value, | ||||
| 			'turnstile-response': turnstileResponse.value, | ||||
| 		}); | ||||
| @@ -262,7 +267,7 @@ async function onSubmit(): Promise<void> { | ||||
| 			}); | ||||
| 			emit('signupEmailPending'); | ||||
| 		} else { | ||||
| 			const res = await os.api('signin', { | ||||
| 			const res = await misskeyApi('signin', { | ||||
| 				username: username.value, | ||||
| 				password: password.value, | ||||
| 			}); | ||||
|   | ||||
| @@ -11,13 +11,13 @@ SPDX-License-Identifier: AGPL-3.0-only | ||||
| 		:pagination="paginationQuery" | ||||
| 		:noGap="!defaultStore.state.showGapBetweenNotesInTimeline" | ||||
| 		@queue="emit('queue', $event)" | ||||
| 		@status="prComponent.setDisabled($event)" | ||||
| 		@status="prComponent?.setDisabled($event)" | ||||
| 	/> | ||||
| </MkPullToRefresh> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts" setup> | ||||
| import { computed, watch, onUnmounted, provide, ref } from 'vue'; | ||||
| import { computed, watch, onUnmounted, provide, ref, shallowRef } from 'vue'; | ||||
| import { Connection } from 'misskey-js/built/streaming.js'; | ||||
| import MkNotes from '@/components/MkNotes.vue'; | ||||
| import MkPullToRefresh from '@/components/MkPullToRefresh.vue'; | ||||
| @@ -62,12 +62,14 @@ type TimelineQueryType = { | ||||
|   roleId?: string | ||||
| } | ||||
|  | ||||
| const prComponent = ref<InstanceType<typeof MkPullToRefresh>>(); | ||||
| const tlComponent = ref<InstanceType<typeof MkNotes>>(); | ||||
| const prComponent = shallowRef<InstanceType<typeof MkPullToRefresh>>(); | ||||
| const tlComponent = shallowRef<InstanceType<typeof MkNotes>>(); | ||||
|  | ||||
| let tlNotesCount = 0; | ||||
|  | ||||
| const prepend = note => { | ||||
| function prepend(note) { | ||||
| 	if (tlComponent.value == null) return; | ||||
|  | ||||
| 	tlNotesCount++; | ||||
|  | ||||
| 	if (instance.notesPerOneAd > 0 && tlNotesCount % instance.notesPerOneAd === 0) { | ||||
| @@ -79,9 +81,9 @@ const prepend = note => { | ||||
| 	emit('note'); | ||||
|  | ||||
| 	if (props.sound) { | ||||
| 		sound.play($i && (note.userId === $i.id) ? 'noteMy' : 'note'); | ||||
| 		sound.playMisskeySfx($i && (note.userId === $i.id) ? 'noteMy' : 'note'); | ||||
| 	} | ||||
| }; | ||||
| } | ||||
|  | ||||
| let connection: Connection; | ||||
| let connection2: Connection; | ||||
| @@ -262,6 +264,8 @@ onUnmounted(() => { | ||||
|  | ||||
| function reloadTimeline() { | ||||
| 	return new Promise<void>((res) => { | ||||
| 		if (tlComponent.value == null) return; | ||||
|  | ||||
| 		tlNotesCount = 0; | ||||
|  | ||||
| 		tlComponent.value.pagingComponent?.reload().then(() => { | ||||
|   | ||||
| @@ -11,7 +11,7 @@ SPDX-License-Identifier: AGPL-3.0-only | ||||
| 	:withOkButton="true" | ||||
| 	:okButtonDisabled="false" | ||||
| 	:canClose="false" | ||||
| 	@close="dialog.close()" | ||||
| 	@close="dialog?.close()" | ||||
| 	@closed="$emit('closed')" | ||||
| 	@ok="ok()" | ||||
| > | ||||
| @@ -33,7 +33,13 @@ SPDX-License-Identifier: AGPL-3.0-only | ||||
| 				<MkButton inline @click="enableAll">{{ i18n.ts.enableAll }}</MkButton> | ||||
| 			</div> | ||||
| 			<div class="_gaps_s"> | ||||
| 				<MkSwitch v-for="kind in Object.keys(permissions)" :key="kind" v-model="permissions[kind]">{{ i18n.t(`_permissions.${kind}`) }}</MkSwitch> | ||||
| 				<MkSwitch v-for="kind in Object.keys(permissionSwitches)" :key="kind" v-model="permissionSwitches[kind]">{{ i18n.t(`_permissions.${kind}`) }}</MkSwitch> | ||||
| 			</div> | ||||
| 			<div v-if="iAmAdmin" :class="$style.adminPermissions"> | ||||
| 				<div :class="$style.adminPermissionsHeader"><b>{{ i18n.ts.adminPermission }}</b></div> | ||||
| 				<div class="_gaps_s"> | ||||
| 					<MkSwitch v-for="kind in Object.keys(permissionSwitchesForAdmin)" :key="kind" v-model="permissionSwitchesForAdmin[kind]">{{ i18n.t(`_permissions.${kind}`) }}</MkSwitch> | ||||
| 				</div> | ||||
| 			</div> | ||||
| 		</div> | ||||
| 	</MkSpacer> | ||||
| @@ -49,6 +55,7 @@ import MkButton from './MkButton.vue'; | ||||
| import MkInfo from './MkInfo.vue'; | ||||
| import MkModalWindow from '@/components/MkModalWindow.vue'; | ||||
| import { i18n } from '@/i18n.js'; | ||||
| import { iAmAdmin } from '@/account.js'; | ||||
|  | ||||
| const props = withDefaults(defineProps<{ | ||||
| 	title?: string | null; | ||||
| @@ -68,37 +75,76 @@ const emit = defineEmits<{ | ||||
| }>(); | ||||
|  | ||||
| const defaultPermissions = Misskey.permissions.filter(p => !p.startsWith('read:admin') && !p.startsWith('write:admin')); | ||||
| const adminPermissions = Misskey.permissions.filter(p => p.startsWith('read:admin') || p.startsWith('write:admin')); | ||||
|  | ||||
| const dialog = shallowRef<InstanceType<typeof MkModalWindow>>(); | ||||
| const name = ref(props.initialName); | ||||
| const permissions = ref(<Record<(typeof Misskey.permissions)[number], boolean>>{}); | ||||
| const permissionSwitches = ref(<Record<(typeof Misskey.permissions)[number], boolean>>{}); | ||||
| const permissionSwitchesForAdmin = ref(<Record<(typeof Misskey.permissions)[number], boolean>>{}); | ||||
|  | ||||
| if (props.initialPermissions) { | ||||
| 	for (const kind of props.initialPermissions) { | ||||
| 		permissions.value[kind] = true; | ||||
| 		permissionSwitches.value[kind] = true; | ||||
| 	} | ||||
| } else { | ||||
| 	for (const kind of defaultPermissions) { | ||||
| 		permissions.value[kind] = false; | ||||
| 		permissionSwitches.value[kind] = false; | ||||
| 	} | ||||
|  | ||||
| 	if (iAmAdmin) { | ||||
| 		for (const kind of adminPermissions) { | ||||
| 			permissionSwitchesForAdmin.value[kind] = false; | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| function ok(): void { | ||||
| 	emit('done', { | ||||
| 		name: name.value, | ||||
| 		permissions: Object.keys(permissions.value).filter(p => permissions.value[p]), | ||||
| 		permissions: [ | ||||
| 			...Object.keys(permissionSwitches.value).filter(p => permissionSwitches.value[p]), | ||||
| 			...(iAmAdmin ? Object.keys(permissionSwitchesForAdmin.value).filter(p => permissionSwitchesForAdmin.value[p]) : []), | ||||
| 		], | ||||
| 	}); | ||||
| 	dialog.value.close(); | ||||
| 	dialog.value?.close(); | ||||
| } | ||||
|  | ||||
| function disableAll(): void { | ||||
| 	for (const p in permissions.value) { | ||||
| 		permissions.value[p] = false; | ||||
| 	for (const p in permissionSwitches.value) { | ||||
| 		permissionSwitches.value[p] = false; | ||||
| 	} | ||||
| 	if (iAmAdmin) { | ||||
| 		for (const p in permissionSwitchesForAdmin.value) { | ||||
| 			permissionSwitchesForAdmin.value[p] = false; | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| function enableAll(): void { | ||||
| 	for (const p in permissions.value) { | ||||
| 		permissions.value[p] = true; | ||||
| 	for (const p in permissionSwitches.value) { | ||||
| 		permissionSwitches.value[p] = true; | ||||
| 	} | ||||
| 	if (iAmAdmin) { | ||||
| 		for (const p in permissionSwitchesForAdmin.value) { | ||||
| 			permissionSwitchesForAdmin.value[p] = true; | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| </script> | ||||
|  | ||||
| <style module lang="scss"> | ||||
| .adminPermissions { | ||||
| 	margin: 8px -6px 0; | ||||
| 	padding: 24px 6px 6px; | ||||
| 	border: 2px solid var(--error); | ||||
| 	border-radius: calc(var(--radius) / 2); | ||||
| } | ||||
|  | ||||
| .adminPermissionsHeader { | ||||
| 	margin: -34px 0 6px 12px; | ||||
| 	padding: 0 4px; | ||||
| 	width: fit-content; | ||||
| 	color: var(--error); | ||||
| 	background: var(--panel); | ||||
| } | ||||
| </style> | ||||
|   | ||||
| @@ -13,8 +13,10 @@ SPDX-License-Identifier: AGPL-3.0-only | ||||
| > | ||||
| 	<div v-show="showing" ref="el" :class="$style.root" class="_acrylic _shadow" :style="{ zIndex, maxWidth: maxWidth + 'px' }"> | ||||
| 		<slot> | ||||
| 			<Mfm v-if="asMfm" :text="text"/> | ||||
| 			<span v-else>{{ text }}</span> | ||||
| 			<template v-if="text"> | ||||
| 				<Mfm v-if="asMfm" :text="text"/> | ||||
| 				<span v-else>{{ text }}</span> | ||||
| 			</template> | ||||
| 		</slot> | ||||
| 	</div> | ||||
| </Transition> | ||||
| @@ -53,6 +55,7 @@ const el = shallowRef<HTMLElement>(); | ||||
| const zIndex = os.claimZIndex('high'); | ||||
|  | ||||
| function setPosition() { | ||||
| 	if (!el.value || !props.targetElement) return; | ||||
| 	const data = calcPopupPosition(el.value, { | ||||
| 		anchorElement: props.targetElement, | ||||
| 		direction: props.direction, | ||||
|   | ||||
| @@ -4,12 +4,12 @@ SPDX-License-Identifier: AGPL-3.0-only | ||||
| --> | ||||
|  | ||||
| <template> | ||||
| <MkModal ref="modal" :zPriority="'middle'" @click="$refs.modal.close()" @closed="$emit('closed')"> | ||||
| <MkModal ref="modal" :zPriority="'middle'" @click="modal?.close()" @closed="$emit('closed')"> | ||||
| 	<div :class="$style.root"> | ||||
| 		<div :class="$style.title"><MkSparkle>{{ i18n.ts.misskeyUpdated }}</MkSparkle></div> | ||||
| 		<div :class="$style.version">✨{{ version }}🚀</div> | ||||
| 		<MkButton full @click="whatIsNew">{{ i18n.ts.whatIsNew }}</MkButton> | ||||
| 		<MkButton :class="$style.gotIt" primary full @click="$refs.modal.close()">{{ i18n.ts.gotIt }}</MkButton> | ||||
| 		<MkButton :class="$style.gotIt" primary full @click="modal?.close()">{{ i18n.ts.gotIt }}</MkButton> | ||||
| 	</div> | ||||
| </MkModal> | ||||
| </template> | ||||
| @@ -25,10 +25,10 @@ import { confetti } from '@/scripts/confetti.js'; | ||||
|  | ||||
| const modal = shallowRef<InstanceType<typeof MkModal>>(); | ||||
|  | ||||
| const whatIsNew = () => { | ||||
| 	modal.value.close(); | ||||
| function whatIsNew() { | ||||
| 	modal.value?.close(); | ||||
| 	window.open('https://go.misskey.io/changelog', '_blank', 'noopener'); | ||||
| }; | ||||
| } | ||||
|  | ||||
| onMounted(() => { | ||||
| 	confetti({ | ||||
|   | ||||
| @@ -84,7 +84,7 @@ SPDX-License-Identifier: AGPL-3.0-only | ||||
|  | ||||
| <script lang="ts" setup> | ||||
| import { defineAsyncComponent, onUnmounted, ref } from 'vue'; | ||||
| import type { summaly } from 'summaly'; | ||||
| import type { summaly } from '@misskey-dev/summaly'; | ||||
| import { url as local } from '@/config.js'; | ||||
| import { i18n } from '@/i18n.js'; | ||||
| import * as os from '@/os.js'; | ||||
|   | ||||
| @@ -72,6 +72,7 @@ import MkModalWindow from '@/components/MkModalWindow.vue'; | ||||
| import MkButton from '@/components/MkButton.vue'; | ||||
| import MkInput from '@/components/MkInput.vue'; | ||||
| import * as os from '@/os.js'; | ||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||
| import { i18n } from '@/i18n.js'; | ||||
| import MkTextarea from '@/components/MkTextarea.vue'; | ||||
| import MkSwitch from '@/components/MkSwitch.vue'; | ||||
| @@ -152,7 +153,7 @@ async function del(): Promise<void> { | ||||
| 	}); | ||||
| 	if (canceled) return; | ||||
|  | ||||
| 	os.api('admin/announcements/delete', { | ||||
| 	misskeyApi('admin/announcements/delete', { | ||||
| 		id: props.announcement.id, | ||||
| 	}).then(() => { | ||||
| 		emit('done', { | ||||
|   | ||||
| @@ -18,7 +18,7 @@ SPDX-License-Identifier: AGPL-3.0-only | ||||
| import * as Misskey from 'misskey-js'; | ||||
| import { onMounted, ref } from 'vue'; | ||||
| import MkMiniChart from '@/components/MkMiniChart.vue'; | ||||
| import * as os from '@/os.js'; | ||||
| import { misskeyApiGet } from '@/scripts/misskey-api.js'; | ||||
| import { acct } from '@/filters/user.js'; | ||||
|  | ||||
| const props = withDefaults(defineProps<{ | ||||
| @@ -32,7 +32,7 @@ const chartValues = ref<number[] | null>(null); | ||||
|  | ||||
| onMounted(() => { | ||||
| 	if (props.withChart) { | ||||
| 		os.apiGet('charts/user/notes', { userId: props.user.id, limit: 16 + 1, span: 'day' }).then(res => { | ||||
| 		misskeyApiGet('charts/user/notes', { userId: props.user.id, limit: 16 + 1, span: 'day' }).then(res => { | ||||
| 			// 今日のぶんの値はまだ途中の値であり、それも含めると大抵の場合前日よりも下降しているようなグラフになってしまうため今日は弾く | ||||
| 			res.inc.splice(0, 1); | ||||
| 			chartValues.value = res.inc; | ||||
|   | ||||
| @@ -60,6 +60,7 @@ import * as Misskey from 'misskey-js'; | ||||
| import MkFollowButton from '@/components/MkFollowButton.vue'; | ||||
| import { userPage } from '@/filters/user.js'; | ||||
| import * as os from '@/os.js'; | ||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||
| import { getUserMenu } from '@/scripts/get-user-menu.js'; | ||||
| import number from '@/filters/number.js'; | ||||
| import { i18n } from '@/i18n.js'; | ||||
| @@ -97,7 +98,7 @@ onMounted(() => { | ||||
| 			Misskey.acct.parse(props.q.substring(1)) : | ||||
| 			{ userId: props.q }; | ||||
|  | ||||
| 		os.api('users/show', query).then(res => { | ||||
| 		misskeyApi('users/show', query).then(res => { | ||||
| 			if (!props.showing) return; | ||||
| 			user.value = res; | ||||
| 		}); | ||||
|   | ||||
| @@ -62,7 +62,7 @@ import * as Misskey from 'misskey-js'; | ||||
| import MkInput from '@/components/MkInput.vue'; | ||||
| import FormSplit from '@/components/form/split.vue'; | ||||
| import MkModalWindow from '@/components/MkModalWindow.vue'; | ||||
| import * as os from '@/os.js'; | ||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||
| import { defaultStore } from '@/store.js'; | ||||
| import { i18n } from '@/i18n.js'; | ||||
| import { $i } from '@/account.js'; | ||||
| @@ -90,7 +90,7 @@ const search = () => { | ||||
| 		users.value = []; | ||||
| 		return; | ||||
| 	} | ||||
| 	os.api('users/search-by-username-and-host', { | ||||
| 	misskeyApi('users/search-by-username-and-host', { | ||||
| 		username: username.value, | ||||
| 		host: host.value, | ||||
| 		limit: 10, | ||||
| @@ -118,7 +118,7 @@ const cancel = () => { | ||||
| }; | ||||
|  | ||||
| onMounted(() => { | ||||
| 	os.api('users/show', { | ||||
| 	misskeyApi('users/show', { | ||||
| 		userIds: defaultStore.state.recentlyUsedUsers, | ||||
| 	}).then(users => { | ||||
| 		if (props.includeSelf && users.find(x => $i ? x.id === $i.id : true) == null) { | ||||
|   | ||||
| @@ -49,7 +49,7 @@ import { i18n } from '@/i18n.js'; | ||||
| import MkSwitch from '@/components/MkSwitch.vue'; | ||||
| import MkInfo from '@/components/MkInfo.vue'; | ||||
| import MkFolder from '@/components/MkFolder.vue'; | ||||
| import * as os from '@/os.js'; | ||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||
|  | ||||
| const isLocked = ref(false); | ||||
| const hideOnlineStatus = ref(false); | ||||
| @@ -57,7 +57,7 @@ const noCrawle = ref(false); | ||||
| const preventAiLearning = ref(true); | ||||
|  | ||||
| watch([isLocked, hideOnlineStatus, noCrawle, preventAiLearning], () => { | ||||
| 	os.api('i/update', { | ||||
| 	misskeyApi('i/update', { | ||||
| 		isLocked: !!isLocked.value, | ||||
| 		hideOnlineStatus: !!hideOnlineStatus.value, | ||||
| 		noCrawle: !!noCrawle.value, | ||||
|   | ||||
| @@ -29,7 +29,7 @@ import * as Misskey from 'misskey-js'; | ||||
| import { ref } from 'vue'; | ||||
| import MkButton from '@/components/MkButton.vue'; | ||||
| import { i18n } from '@/i18n.js'; | ||||
| import * as os from '@/os.js'; | ||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||
|  | ||||
| const props = defineProps<{ | ||||
| 	user: Misskey.entities.UserDetailed; | ||||
| @@ -39,7 +39,7 @@ const isFollowing = ref(false); | ||||
|  | ||||
| async function follow() { | ||||
| 	isFollowing.value = true; | ||||
| 	os.api('following/create', { | ||||
| 	misskeyApi('following/create', { | ||||
| 		userId: props.user.id, | ||||
| 	}); | ||||
| } | ||||
|   | ||||
| @@ -17,7 +17,7 @@ import { onMounted, shallowRef, ref } from 'vue'; | ||||
| import { Chart } from 'chart.js'; | ||||
| import gradient from 'chartjs-plugin-gradient'; | ||||
| import tinycolor from 'tinycolor2'; | ||||
| import * as os from '@/os.js'; | ||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||
| import { defaultStore } from '@/store.js'; | ||||
| import { useChartTooltip } from '@/scripts/use-chart-tooltip.js'; | ||||
| import { chartVLine } from '@/scripts/chart-vline.js'; | ||||
| @@ -53,7 +53,7 @@ async function renderChart() { | ||||
| 		})); | ||||
| 	}; | ||||
|  | ||||
| 	const raw = await os.api('charts/active-users', { limit: chartLimit, span: 'day' }); | ||||
| 	const raw = await misskeyApi('charts/active-users', { limit: chartLimit, span: 'day' }); | ||||
|  | ||||
| 	const vLineColor = defaultStore.state.darkMode ? 'rgba(255, 255, 255, 0.2)' : 'rgba(0, 0, 0, 0.2)'; | ||||
|  | ||||
|   | ||||
| @@ -60,6 +60,7 @@ import MkTimeline from '@/components/MkTimeline.vue'; | ||||
| import MkInfo from '@/components/MkInfo.vue'; | ||||
| import { instanceName } from '@/config.js'; | ||||
| import * as os from '@/os.js'; | ||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||
| import { i18n } from '@/i18n.js'; | ||||
| import { instance } from '@/instance.js'; | ||||
| import MkNumber from '@/components/MkNumber.vue'; | ||||
| @@ -68,11 +69,11 @@ import XActiveUsersChart from '@/components/MkVisitorDashboard.ActiveUsersChart. | ||||
| const meta = ref<Misskey.entities.MetaResponse | null>(null); | ||||
| const stats = ref<Misskey.entities.StatsResponse | null>(null); | ||||
|  | ||||
| os.api('meta', { detail: true }).then(_meta => { | ||||
| misskeyApi('meta', { detail: true }).then(_meta => { | ||||
| 	meta.value = _meta; | ||||
| }); | ||||
|  | ||||
| os.api('stats', {}).then((res) => { | ||||
| misskeyApi('stats', {}).then((res) => { | ||||
| 	stats.value = res; | ||||
| }); | ||||
|  | ||||
|   | ||||
| @@ -143,6 +143,7 @@ function top() { | ||||
| } | ||||
|  | ||||
| function maximize() { | ||||
| 	if (rootEl.value == null) return; | ||||
| 	maximized.value = true; | ||||
| 	unResizedTop = rootEl.value.style.top; | ||||
| 	unResizedLeft = rootEl.value.style.left; | ||||
| @@ -155,6 +156,7 @@ function maximize() { | ||||
| } | ||||
|  | ||||
| function unMaximize() { | ||||
| 	if (rootEl.value == null) return; | ||||
| 	maximized.value = false; | ||||
| 	rootEl.value.style.top = unResizedTop; | ||||
| 	rootEl.value.style.left = unResizedLeft; | ||||
| @@ -163,6 +165,7 @@ function unMaximize() { | ||||
| } | ||||
|  | ||||
| function minimize() { | ||||
| 	if (rootEl.value == null) return; | ||||
| 	minimized.value = true; | ||||
| 	unResizedWidth = rootEl.value.style.width; | ||||
| 	unResizedHeight = rootEl.value.style.height; | ||||
| @@ -171,8 +174,8 @@ function minimize() { | ||||
| } | ||||
|  | ||||
| function unMinimize() { | ||||
| 	if (rootEl.value == null) return; | ||||
| 	const main = rootEl.value; | ||||
| 	if (main == null) return; | ||||
|  | ||||
| 	minimized.value = false; | ||||
| 	rootEl.value.style.width = unResizedWidth; | ||||
|   | ||||
| @@ -15,7 +15,7 @@ import * as os from '@/os.js'; | ||||
| import copyToClipboard from '@/scripts/copy-to-clipboard.js'; | ||||
| import { url } from '@/config.js'; | ||||
| import { i18n } from '@/i18n.js'; | ||||
| import { useRouter } from '@/router.js'; | ||||
| import { useRouter } from '@/global/router/supplier.js'; | ||||
|  | ||||
| const props = withDefaults(defineProps<{ | ||||
| 	to: string; | ||||
|   | ||||
| @@ -91,7 +91,7 @@ function onClick(ev: MouseEvent) { | ||||
| 			icon: 'ti ti-plus', | ||||
| 			action: () => { | ||||
| 				react(`:${props.name}:`); | ||||
| 				sound.play('reaction'); | ||||
| 				sound.playMisskeySfx('reaction'); | ||||
| 			}, | ||||
| 		}] : [])], ev.currentTarget ?? ev.target); | ||||
| 	} | ||||
|   | ||||
| @@ -5,15 +5,14 @@ SPDX-License-Identifier: AGPL-3.0-only | ||||
|  | ||||
| <template> | ||||
| <img v-if="!useOsNativeEmojis" :class="$style.root" :src="url" :alt="props.emoji" decoding="async" @pointerenter="computeTitle" @click="onClick"/> | ||||
| <span v-else-if="useOsNativeEmojis" :alt="props.emoji" @pointerenter="computeTitle" @click="onClick">{{ props.emoji }}</span> | ||||
| <span v-else>{{ emoji }}</span> | ||||
| <span v-else :alt="props.emoji" @pointerenter="computeTitle" @click="onClick">{{ colorizedNativeEmoji }}</span> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts" setup> | ||||
| import { computed, inject } from 'vue'; | ||||
| import { char2twemojiFilePath, char2fluentEmojiFilePath } from '@/scripts/emoji-base.js'; | ||||
| import { defaultStore } from '@/store.js'; | ||||
| import { getEmojiName } from '@/scripts/emojilist.js'; | ||||
| import { colorizeEmoji, getEmojiName } from '@/scripts/emojilist.js'; | ||||
| import * as os from '@/os.js'; | ||||
| import copyToClipboard from '@/scripts/copy-to-clipboard.js'; | ||||
| import * as sound from '@/scripts/sound.js'; | ||||
| @@ -30,9 +29,8 @@ const react = inject<((name: string) => void) | null>('react', null); | ||||
| const char2path = defaultStore.state.emojiStyle === 'twemoji' ? char2twemojiFilePath : char2fluentEmojiFilePath; | ||||
|  | ||||
| const useOsNativeEmojis = computed(() => defaultStore.state.emojiStyle === 'native'); | ||||
| const url = computed(() => { | ||||
| 	return char2path(props.emoji); | ||||
| }); | ||||
| const url = computed(() => char2path(props.emoji)); | ||||
| const colorizedNativeEmoji = computed(() => colorizeEmoji(props.emoji)); | ||||
|  | ||||
| // Searching from an array with 2000 items for every emoji felt like too energy-consuming, so I decided to do it lazily on pointerenter | ||||
| function computeTitle(event: PointerEvent): void { | ||||
| @@ -57,7 +55,7 @@ function onClick(ev: MouseEvent) { | ||||
| 			icon: 'ti ti-plus', | ||||
| 			action: () => { | ||||
| 				react(props.emoji); | ||||
| 				sound.play('reaction'); | ||||
| 				sound.playMisskeySfx('reaction'); | ||||
| 			}, | ||||
| 		}] : [])], ev.currentTarget ?? ev.target); | ||||
| 	} | ||||
|   | ||||
| @@ -16,12 +16,12 @@ SPDX-License-Identifier: AGPL-3.0-only | ||||
| </template> | ||||
|  | ||||
| <script lang="ts" setup> | ||||
| import { inject, onBeforeUnmount, provide, shallowRef, ref } from 'vue'; | ||||
| import { Resolved, Router } from '@/nirax.js'; | ||||
| import { inject, onBeforeUnmount, provide, ref, shallowRef } from 'vue'; | ||||
| import { IRouter, Resolved } from '@/nirax.js'; | ||||
| import { defaultStore } from '@/store.js'; | ||||
|  | ||||
| const props = defineProps<{ | ||||
| 	router?: Router; | ||||
| 	router?: IRouter; | ||||
| }>(); | ||||
|  | ||||
| const router = props.router ?? inject('router'); | ||||
|   | ||||
| @@ -16,7 +16,7 @@ import * as Misskey from 'misskey-js'; | ||||
| import { NoteBlock } from './block.type.js'; | ||||
| import MkNote from '@/components/MkNote.vue'; | ||||
| import MkNoteDetailed from '@/components/MkNoteDetailed.vue'; | ||||
| import * as os from '@/os.js'; | ||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||
|  | ||||
| const props = defineProps<{ | ||||
| 	block: NoteBlock, | ||||
| @@ -26,7 +26,7 @@ const props = defineProps<{ | ||||
| const note = ref<Misskey.entities.Note | null>(null); | ||||
|  | ||||
| onMounted(() => { | ||||
| 	os.api('notes/show', { noteId: props.block.note }) | ||||
| 	misskeyApi('notes/show', { noteId: props.block.note }) | ||||
| 		.then(result => { | ||||
| 			note.value = result; | ||||
| 		}); | ||||
|   | ||||
| @@ -5,7 +5,7 @@ | ||||
|  | ||||
| import { shallowRef, computed, markRaw, watch } from 'vue'; | ||||
| import * as Misskey from 'misskey-js'; | ||||
| import { api, apiGet } from '@/os.js'; | ||||
| import { misskeyApi, misskeyApiGet } from '@/scripts/misskey-api.js'; | ||||
| import { useStream } from '@/stream.js'; | ||||
| import { get, set, exist } from '@/scripts/idb-proxy.js'; | ||||
|  | ||||
| @@ -52,12 +52,12 @@ export async function fetchCustomEmojis(force = false) { | ||||
|  | ||||
| 	let res; | ||||
| 	if (force) { | ||||
| 		res = await api('emojis', {}); | ||||
| 		res = await misskeyApi('emojis', {}); | ||||
| 	} else { | ||||
| 		const emojiCacheExist = await exist('emojis'); | ||||
| 		const lastFetchedAt = await get('lastEmojisFetchedAt'); | ||||
| 		if (lastFetchedAt && (now - lastFetchedAt) < 1000 * 60 * 60 && emojiCacheExist) return; | ||||
| 		res = await apiGet('emojis', {}); | ||||
| 		res = await misskeyApiGet('emojis', {}); | ||||
| 	} | ||||
|  | ||||
| 	customEmojis.value = res.emojis; | ||||
|   | ||||
							
								
								
									
										575
									
								
								packages/frontend/src/global/router/definition.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										575
									
								
								packages/frontend/src/global/router/definition.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,575 @@ | ||||
| /* | ||||
|  * SPDX-FileCopyrightText: syuilo and other misskey contributors | ||||
|  * SPDX-License-Identifier: AGPL-3.0-only | ||||
|  */ | ||||
|  | ||||
| import { App, AsyncComponentLoader, defineAsyncComponent, provide } from 'vue'; | ||||
| import { IRouter, Router } from '@/nirax.js'; | ||||
| import { $i, iAmModerator } from '@/account.js'; | ||||
| import MkLoading from '@/pages/_loading_.vue'; | ||||
| import MkError from '@/pages/_error_.vue'; | ||||
| import { setMainRouter } from '@/global/router/main.js'; | ||||
|  | ||||
| const page = (loader: AsyncComponentLoader<any>) => defineAsyncComponent({ | ||||
| 	loader: loader, | ||||
| 	loadingComponent: MkLoading, | ||||
| 	errorComponent: MkError, | ||||
| }); | ||||
| const routes = [{ | ||||
| 	path: '/@:initUser/pages/:initPageName/view-source', | ||||
| 	component: page(() => import('@/pages/page-editor/page-editor.vue')), | ||||
| }, { | ||||
| 	path: '/@:username/pages/:pageName', | ||||
| 	component: page(() => import('@/pages/page.vue')), | ||||
| }, { | ||||
| 	path: '/@:acct/following', | ||||
| 	component: page(() => import('@/pages/user/following.vue')), | ||||
| }, { | ||||
| 	path: '/@:acct/followers', | ||||
| 	component: page(() => import('@/pages/user/followers.vue')), | ||||
| }, { | ||||
| 	name: 'user', | ||||
| 	path: '/@:acct/:page?', | ||||
| 	component: page(() => import('@/pages/user/index.vue')), | ||||
| }, { | ||||
| 	name: 'note', | ||||
| 	path: '/notes/:noteId', | ||||
| 	component: page(() => import('@/pages/note.vue')), | ||||
| }, { | ||||
| 	name: 'list', | ||||
| 	path: '/list/:listId', | ||||
| 	component: page(() => import('@/pages/list.vue')), | ||||
| }, { | ||||
| 	path: '/clips/:clipId', | ||||
| 	component: page(() => import('@/pages/clip.vue')), | ||||
| }, { | ||||
| 	path: '/instance-info/:host', | ||||
| 	component: page(() => import('@/pages/instance-info.vue')), | ||||
| }, { | ||||
| 	name: 'settings', | ||||
| 	path: '/settings', | ||||
| 	component: page(() => import('@/pages/settings/index.vue')), | ||||
| 	loginRequired: true, | ||||
| 	children: [{ | ||||
| 		path: '/profile', | ||||
| 		name: 'profile', | ||||
| 		component: page(() => import('@/pages/settings/profile.vue')), | ||||
| 	}, { | ||||
| 		path: '/avatar-decoration', | ||||
| 		name: 'avatarDecoration', | ||||
| 		component: page(() => import('@/pages/settings/avatar-decoration.vue')), | ||||
| 	}, { | ||||
| 		path: '/roles', | ||||
| 		name: 'roles', | ||||
| 		component: page(() => import('@/pages/settings/roles.vue')), | ||||
| 	}, { | ||||
| 		path: '/privacy', | ||||
| 		name: 'privacy', | ||||
| 		component: page(() => import('@/pages/settings/privacy.vue')), | ||||
| 	}, { | ||||
| 		path: '/emoji-picker', | ||||
| 		name: 'emojiPicker', | ||||
| 		component: page(() => import('@/pages/settings/emoji-picker.vue')), | ||||
| 	}, { | ||||
| 		path: '/drive', | ||||
| 		name: 'drive', | ||||
| 		component: page(() => import('@/pages/settings/drive.vue')), | ||||
| 	}, { | ||||
| 		path: '/drive/cleaner', | ||||
| 		name: 'drive', | ||||
| 		component: page(() => import('@/pages/settings/drive-cleaner.vue')), | ||||
| 	}, { | ||||
| 		path: '/notifications', | ||||
| 		name: 'notifications', | ||||
| 		component: page(() => import('@/pages/settings/notifications.vue')), | ||||
| 	}, { | ||||
| 		path: '/email', | ||||
| 		name: 'email', | ||||
| 		component: page(() => import('@/pages/settings/email.vue')), | ||||
| 	}, { | ||||
| 		path: '/security', | ||||
| 		name: 'security', | ||||
| 		component: page(() => import('@/pages/settings/security.vue')), | ||||
| 	}, { | ||||
| 		path: '/general', | ||||
| 		name: 'general', | ||||
| 		component: page(() => import('@/pages/settings/general.vue')), | ||||
| 	}, { | ||||
| 		path: '/theme/install', | ||||
| 		name: 'theme', | ||||
| 		component: page(() => import('@/pages/settings/theme.install.vue')), | ||||
| 	}, { | ||||
| 		path: '/theme/manage', | ||||
| 		name: 'theme', | ||||
| 		component: page(() => import('@/pages/settings/theme.manage.vue')), | ||||
| 	}, { | ||||
| 		path: '/theme', | ||||
| 		name: 'theme', | ||||
| 		component: page(() => import('@/pages/settings/theme.vue')), | ||||
| 	}, { | ||||
| 		path: '/navbar', | ||||
| 		name: 'navbar', | ||||
| 		component: page(() => import('@/pages/settings/navbar.vue')), | ||||
| 	}, { | ||||
| 		path: '/statusbar', | ||||
| 		name: 'statusbar', | ||||
| 		component: page(() => import('@/pages/settings/statusbar.vue')), | ||||
| 	}, { | ||||
| 		path: '/sounds', | ||||
| 		name: 'sounds', | ||||
| 		component: page(() => import('@/pages/settings/sounds.vue')), | ||||
| 	}, { | ||||
| 		path: '/plugin/install', | ||||
| 		name: 'plugin', | ||||
| 		component: page(() => import('@/pages/settings/plugin.install.vue')), | ||||
| 	}, { | ||||
| 		path: '/plugin', | ||||
| 		name: 'plugin', | ||||
| 		component: page(() => import('@/pages/settings/plugin.vue')), | ||||
| 	}, { | ||||
| 		path: '/import-export', | ||||
| 		name: 'import-export', | ||||
| 		component: page(() => import('@/pages/settings/import-export.vue')), | ||||
| 	}, { | ||||
| 		path: '/mute-block', | ||||
| 		name: 'mute-block', | ||||
| 		component: page(() => import('@/pages/settings/mute-block.vue')), | ||||
| 	}, { | ||||
| 		path: '/api', | ||||
| 		name: 'api', | ||||
| 		component: page(() => import('@/pages/settings/api.vue')), | ||||
| 	}, { | ||||
| 		path: '/apps', | ||||
| 		name: 'api', | ||||
| 		component: page(() => import('@/pages/settings/apps.vue')), | ||||
| 	}, { | ||||
| 		path: '/webhook/edit/:webhookId', | ||||
| 		name: 'webhook', | ||||
| 		component: page(() => import('@/pages/settings/webhook.edit.vue')), | ||||
| 	}, { | ||||
| 		path: '/webhook/new', | ||||
| 		name: 'webhook', | ||||
| 		component: page(() => import('@/pages/settings/webhook.new.vue')), | ||||
| 	}, { | ||||
| 		path: '/webhook', | ||||
| 		name: 'webhook', | ||||
| 		component: page(() => import('@/pages/settings/webhook.vue')), | ||||
| 	}, { | ||||
| 		path: '/deck', | ||||
| 		name: 'deck', | ||||
| 		component: page(() => import('@/pages/settings/deck.vue')), | ||||
| 	}, { | ||||
| 		path: '/preferences-backups', | ||||
| 		name: 'preferences-backups', | ||||
| 		component: page(() => import('@/pages/settings/preferences-backups.vue')), | ||||
| 	}, { | ||||
| 		path: '/migration', | ||||
| 		name: 'migration', | ||||
| 		component: page(() => import('@/pages/settings/migration.vue')), | ||||
| 	}, { | ||||
| 		path: '/custom-css', | ||||
| 		name: 'general', | ||||
| 		component: page(() => import('@/pages/settings/custom-css.vue')), | ||||
| 	}, { | ||||
| 		path: '/accounts', | ||||
| 		name: 'profile', | ||||
| 		component: page(() => import('@/pages/settings/accounts.vue')), | ||||
| 	}, { | ||||
| 		path: '/account-stats', | ||||
| 		name: 'other', | ||||
| 		component: page(() => import('@/pages/settings/account-stats.vue')), | ||||
| 	}, { | ||||
| 		path: '/other', | ||||
| 		name: 'other', | ||||
| 		component: page(() => import('@/pages/settings/other.vue')), | ||||
| 	}, { | ||||
| 		path: '/', | ||||
| 		component: page(() => import('@/pages/_empty_.vue')), | ||||
| 	}], | ||||
| }, { | ||||
| 	path: '/reset-password/:token?', | ||||
| 	component: page(() => import('@/pages/reset-password.vue')), | ||||
| }, { | ||||
| 	path: '/signup-complete/:code', | ||||
| 	component: page(() => import('@/pages/signup-complete.vue')), | ||||
| }, { | ||||
| 	path: '/announcements', | ||||
| 	component: page(() => import('@/pages/announcements.vue')), | ||||
| }, { | ||||
| 	path: '/about', | ||||
| 	component: page(() => import('@/pages/about.vue')), | ||||
| 	hash: 'initialTab', | ||||
| }, { | ||||
| 	path: '/about-misskey', | ||||
| 	component: page(() => import('@/pages/about-misskey.vue')), | ||||
| }, { | ||||
| 	path: '/invite', | ||||
| 	name: 'invite', | ||||
| 	component: page(() => import('@/pages/invite.vue')), | ||||
| }, { | ||||
| 	path: '/ads', | ||||
| 	component: page(() => import('@/pages/ads.vue')), | ||||
| }, { | ||||
| 	path: '/theme-editor', | ||||
| 	component: page(() => import('@/pages/theme-editor.vue')), | ||||
| 	loginRequired: true, | ||||
| }, { | ||||
| 	path: '/roles/:role', | ||||
| 	component: page(() => import('@/pages/role.vue')), | ||||
| }, { | ||||
| 	path: '/user-tags/:tag', | ||||
| 	component: page(() => import('@/pages/user-tag.vue')), | ||||
| }, { | ||||
| 	path: '/explore', | ||||
| 	component: page(() => import('@/pages/explore.vue')), | ||||
| 	hash: 'initialTab', | ||||
| }, { | ||||
| 	path: '/search', | ||||
| 	component: page(() => import('@/pages/search.vue')), | ||||
| 	query: { | ||||
| 		q: 'query', | ||||
| 		channel: 'channel', | ||||
| 		type: 'type', | ||||
| 		origin: 'origin', | ||||
| 	}, | ||||
| }, { | ||||
| 	path: '/authorize-follow', | ||||
| 	component: page(() => import('@/pages/follow.vue')), | ||||
| 	loginRequired: true, | ||||
| }, { | ||||
| 	path: '/share', | ||||
| 	component: page(() => import('@/pages/share.vue')), | ||||
| 	loginRequired: true, | ||||
| }, { | ||||
| 	path: '/api-console', | ||||
| 	component: page(() => import('@/pages/api-console.vue')), | ||||
| 	loginRequired: true, | ||||
| }, { | ||||
| 	path: '/scratchpad', | ||||
| 	component: page(() => import('@/pages/scratchpad.vue')), | ||||
| }, { | ||||
| 	path: '/auth/:token', | ||||
| 	component: page(() => import('@/pages/auth.vue')), | ||||
| }, { | ||||
| 	path: '/miauth/:session', | ||||
| 	component: page(() => import('@/pages/miauth.vue')), | ||||
| 	query: { | ||||
| 		callback: 'callback', | ||||
| 		name: 'name', | ||||
| 		icon: 'icon', | ||||
| 		permission: 'permission', | ||||
| 	}, | ||||
| }, { | ||||
| 	path: '/oauth/authorize', | ||||
| 	component: page(() => import('@/pages/oauth.vue')), | ||||
| }, { | ||||
| 	path: '/tags/:tag', | ||||
| 	component: page(() => import('@/pages/tag.vue')), | ||||
| }, { | ||||
| 	path: '/pages/new', | ||||
| 	component: page(() => import('@/pages/page-editor/page-editor.vue')), | ||||
| 	loginRequired: true, | ||||
| }, { | ||||
| 	path: '/pages/edit/:initPageId', | ||||
| 	component: page(() => import('@/pages/page-editor/page-editor.vue')), | ||||
| 	loginRequired: true, | ||||
| }, { | ||||
| 	path: '/pages', | ||||
| 	component: page(() => import('@/pages/pages.vue')), | ||||
| }, { | ||||
| 	path: '/play/:id/edit', | ||||
| 	component: page(() => import('@/pages/flash/flash-edit.vue')), | ||||
| 	loginRequired: true, | ||||
| }, { | ||||
| 	path: '/play/new', | ||||
| 	component: page(() => import('@/pages/flash/flash-edit.vue')), | ||||
| 	loginRequired: true, | ||||
| }, { | ||||
| 	path: '/play/:id', | ||||
| 	component: page(() => import('@/pages/flash/flash.vue')), | ||||
| }, { | ||||
| 	path: '/play', | ||||
| 	component: page(() => import('@/pages/flash/flash-index.vue')), | ||||
| }, { | ||||
| 	path: '/gallery/:postId/edit', | ||||
| 	component: page(() => import('@/pages/gallery/edit.vue')), | ||||
| 	loginRequired: true, | ||||
| }, { | ||||
| 	path: '/gallery/new', | ||||
| 	component: page(() => import('@/pages/gallery/edit.vue')), | ||||
| 	loginRequired: true, | ||||
| }, { | ||||
| 	path: '/gallery/:postId', | ||||
| 	component: page(() => import('@/pages/gallery/post.vue')), | ||||
| }, { | ||||
| 	path: '/gallery', | ||||
| 	component: page(() => import('@/pages/gallery/index.vue')), | ||||
| }, { | ||||
| 	path: '/channels/:channelId/edit', | ||||
| 	component: page(() => import('@/pages/channel-editor.vue')), | ||||
| 	loginRequired: true, | ||||
| }, { | ||||
| 	path: '/channels/new', | ||||
| 	component: page(() => import('@/pages/channel-editor.vue')), | ||||
| 	loginRequired: true, | ||||
| }, { | ||||
| 	path: '/channels/:channelId', | ||||
| 	component: page(() => import('@/pages/channel.vue')), | ||||
| }, { | ||||
| 	path: '/channels', | ||||
| 	component: page(() => import('@/pages/channels.vue')), | ||||
| }, { | ||||
| 	path: '/custom-emojis-manager', | ||||
| 	component: page(() => import('@/pages/custom-emojis-manager.vue')), | ||||
| }, { | ||||
| 	path: '/avatar-decorations', | ||||
| 	name: 'avatarDecorations', | ||||
| 	component: page(() => import('@/pages/avatar-decorations.vue')), | ||||
| }, { | ||||
| 	path: '/registry/keys/:domain/:path(*)?', | ||||
| 	component: page(() => import('@/pages/registry.keys.vue')), | ||||
| }, { | ||||
| 	path: '/registry/value/:domain/:path(*)?', | ||||
| 	component: page(() => import('@/pages/registry.value.vue')), | ||||
| }, { | ||||
| 	path: '/registry', | ||||
| 	component: page(() => import('@/pages/registry.vue')), | ||||
| }, { | ||||
| 	path: '/install-extentions', | ||||
| 	component: page(() => import('@/pages/install-extentions.vue')), | ||||
| 	loginRequired: true, | ||||
| }, { | ||||
| 	path: '/admin/user/:userId', | ||||
| 	component: iAmModerator ? page(() => import('@/pages/admin-user.vue')) : page(() => import('@/pages/not-found.vue')), | ||||
| }, { | ||||
| 	path: '/admin/file/:fileId', | ||||
| 	component: iAmModerator ? page(() => import('@/pages/admin-file.vue')) : page(() => import('@/pages/not-found.vue')), | ||||
| }, { | ||||
| 	path: '/admin', | ||||
| 	component: iAmModerator ? page(() => import('@/pages/admin/index.vue')) : page(() => import('@/pages/not-found.vue')), | ||||
| 	children: [{ | ||||
| 		path: '/overview', | ||||
| 		name: 'overview', | ||||
| 		component: page(() => import('@/pages/admin/overview.vue')), | ||||
| 	}, { | ||||
| 		path: '/users', | ||||
| 		name: 'users', | ||||
| 		component: page(() => import('@/pages/admin/users.vue')), | ||||
| 	}, { | ||||
| 		path: '/emojis', | ||||
| 		name: 'emojis', | ||||
| 		component: page(() => import('@/pages/custom-emojis-manager.vue')), | ||||
| 	}, { | ||||
| 		path: '/avatar-decorations', | ||||
| 		name: 'avatarDecorations', | ||||
| 		component: page(() => import('@/pages/avatar-decorations.vue')), | ||||
| 	}, { | ||||
| 		path: '/queue', | ||||
| 		name: 'queue', | ||||
| 		component: page(() => import('@/pages/admin/queue.vue')), | ||||
| 	}, { | ||||
| 		path: '/files', | ||||
| 		name: 'files', | ||||
| 		component: page(() => import('@/pages/admin/files.vue')), | ||||
| 	}, { | ||||
| 		path: '/federation', | ||||
| 		name: 'federation', | ||||
| 		component: page(() => import('@/pages/admin/federation.vue')), | ||||
| 	}, { | ||||
| 		path: '/announcements', | ||||
| 		name: 'announcements', | ||||
| 		component: page(() => import('@/pages/admin/announcements.vue')), | ||||
| 	}, { | ||||
| 		path: '/ads', | ||||
| 		name: 'ads', | ||||
| 		component: page(() => import('@/pages/admin/ads.vue')), | ||||
| 	}, { | ||||
| 		path: '/roles/:id/edit', | ||||
| 		name: 'roles', | ||||
| 		component: page(() => import('@/pages/admin/roles.edit.vue')), | ||||
| 	}, { | ||||
| 		path: '/roles/new', | ||||
| 		name: 'roles', | ||||
| 		component: page(() => import('@/pages/admin/roles.edit.vue')), | ||||
| 	}, { | ||||
| 		path: '/roles/:id', | ||||
| 		name: 'roles', | ||||
| 		component: page(() => import('@/pages/admin/roles.role.vue')), | ||||
| 	}, { | ||||
| 		path: '/roles', | ||||
| 		name: 'roles', | ||||
| 		component: page(() => import('@/pages/admin/roles.vue')), | ||||
| 	}, { | ||||
| 		path: '/database', | ||||
| 		name: 'database', | ||||
| 		component: page(() => import('@/pages/admin/database.vue')), | ||||
| 	}, { | ||||
| 		path: '/abuses', | ||||
| 		name: 'abuses', | ||||
| 		component: page(() => import('@/pages/admin/abuses.vue')), | ||||
| 	}, { | ||||
| 		path: '/modlog', | ||||
| 		name: 'modlog', | ||||
| 		component: page(() => import('@/pages/admin/modlog.vue')), | ||||
| 	}, { | ||||
| 		path: '/settings', | ||||
| 		name: 'settings', | ||||
| 		component: page(() => import('@/pages/admin/settings.vue')), | ||||
| 	}, { | ||||
| 		path: '/branding', | ||||
| 		name: 'branding', | ||||
| 		component: page(() => import('@/pages/admin/branding.vue')), | ||||
| 	}, { | ||||
| 		path: '/moderation', | ||||
| 		name: 'moderation', | ||||
| 		component: page(() => import('@/pages/admin/moderation.vue')), | ||||
| 	}, { | ||||
| 		path: '/email-settings', | ||||
| 		name: 'email-settings', | ||||
| 		component: page(() => import('@/pages/admin/email-settings.vue')), | ||||
| 	}, { | ||||
| 		path: '/object-storage', | ||||
| 		name: 'object-storage', | ||||
| 		component: page(() => import('@/pages/admin/object-storage.vue')), | ||||
| 	}, { | ||||
| 		path: '/security', | ||||
| 		name: 'security', | ||||
| 		component: page(() => import('@/pages/admin/security.vue')), | ||||
| 	}, { | ||||
| 		path: '/relays', | ||||
| 		name: 'relays', | ||||
| 		component: page(() => import('@/pages/admin/relays.vue')), | ||||
| 	}, { | ||||
| 		path: '/instance-block', | ||||
| 		name: 'instance-block', | ||||
| 		component: page(() => import('@/pages/admin/instance-block.vue')), | ||||
| 	}, { | ||||
| 		path: '/proxy-account', | ||||
| 		name: 'proxy-account', | ||||
| 		component: page(() => import('@/pages/admin/proxy-account.vue')), | ||||
| 	}, { | ||||
| 		path: '/external-services', | ||||
| 		name: 'external-services', | ||||
| 		component: page(() => import('@/pages/admin/external-services.vue')), | ||||
| 	}, { | ||||
| 		path: '/other-settings', | ||||
| 		name: 'other-settings', | ||||
| 		component: page(() => import('@/pages/admin/other-settings.vue')), | ||||
| 	}, { | ||||
| 		path: '/server-rules', | ||||
| 		name: 'server-rules', | ||||
| 		component: page(() => import('@/pages/admin/server-rules.vue')), | ||||
| 	}, { | ||||
| 		path: '/invites', | ||||
| 		name: 'invites', | ||||
| 		component: page(() => import('@/pages/admin/invites.vue')), | ||||
| 	}, { | ||||
| 		path: '/', | ||||
| 		component: page(() => import('@/pages/_empty_.vue')), | ||||
| 	}], | ||||
| }, { | ||||
| 	path: '/my/notifications', | ||||
| 	component: page(() => import('@/pages/notifications.vue')), | ||||
| 	loginRequired: true, | ||||
| }, { | ||||
| 	path: '/my/favorites', | ||||
| 	component: page(() => import('@/pages/favorites.vue')), | ||||
| 	loginRequired: true, | ||||
| }, { | ||||
| 	path: '/my/achievements', | ||||
| 	component: page(() => import('@/pages/achievements.vue')), | ||||
| 	loginRequired: true, | ||||
| }, { | ||||
| 	path: '/my/drive/folder/:folder', | ||||
| 	component: page(() => import('@/pages/drive.vue')), | ||||
| 	loginRequired: true, | ||||
| }, { | ||||
| 	path: '/my/drive', | ||||
| 	component: page(() => import('@/pages/drive.vue')), | ||||
| 	loginRequired: true, | ||||
| }, { | ||||
| 	path: '/my/drive/file/:fileId', | ||||
| 	component: page(() => import('@/pages/drive.file.vue')), | ||||
| 	loginRequired: true, | ||||
| }, { | ||||
| 	path: '/my/follow-requests', | ||||
| 	component: page(() => import('@/pages/follow-requests.vue')), | ||||
| 	loginRequired: true, | ||||
| }, { | ||||
| 	path: '/my/lists/:listId', | ||||
| 	component: page(() => import('@/pages/my-lists/list.vue')), | ||||
| 	loginRequired: true, | ||||
| }, { | ||||
| 	path: '/my/lists', | ||||
| 	component: page(() => import('@/pages/my-lists/index.vue')), | ||||
| 	loginRequired: true, | ||||
| }, { | ||||
| 	path: '/my/clips', | ||||
| 	component: page(() => import('@/pages/my-clips/index.vue')), | ||||
| 	loginRequired: true, | ||||
| }, { | ||||
| 	path: '/my/antennas/create', | ||||
| 	component: page(() => import('@/pages/my-antennas/create.vue')), | ||||
| 	loginRequired: true, | ||||
| }, { | ||||
| 	path: '/my/antennas/:antennaId', | ||||
| 	component: page(() => import('@/pages/my-antennas/edit.vue')), | ||||
| 	loginRequired: true, | ||||
| }, { | ||||
| 	path: '/my/antennas', | ||||
| 	component: page(() => import('@/pages/my-antennas/index.vue')), | ||||
| 	loginRequired: true, | ||||
| }, { | ||||
| 	path: '/timeline/list/:listId', | ||||
| 	component: page(() => import('@/pages/user-list-timeline.vue')), | ||||
| 	loginRequired: true, | ||||
| }, { | ||||
| 	path: '/timeline/antenna/:antennaId', | ||||
| 	component: page(() => import('@/pages/antenna-timeline.vue')), | ||||
| 	loginRequired: true, | ||||
| }, { | ||||
| 	path: '/clicker', | ||||
| 	component: page(() => import('@/pages/clicker.vue')), | ||||
| 	loginRequired: true, | ||||
| }, { | ||||
| 	path: '/bubble-game', | ||||
| 	component: page(() => import('@/pages/drop-and-fusion.vue')), | ||||
| 	loginRequired: true, | ||||
| }, { | ||||
| 	path: '/timeline', | ||||
| 	component: page(() => import('@/pages/timeline.vue')), | ||||
| }, { | ||||
| 	name: 'index', | ||||
| 	path: '/', | ||||
| 	component: $i ? page(() => import('@/pages/timeline.vue')) : page(() => import('@/pages/welcome.vue')), | ||||
| 	globalCacheKey: 'index', | ||||
| }, { | ||||
| 	path: '/:(*)', | ||||
| 	component: page(() => import('@/pages/not-found.vue')), | ||||
| }]; | ||||
|  | ||||
| function createRouterImpl(path: string): IRouter { | ||||
| 	return new Router(routes, path, !!$i, page(() => import('@/pages/not-found.vue'))); | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * {@link Router}による画面遷移を可能とするために{@link mainRouter}をセットアップする。 | ||||
|  * また、{@link Router}のインスタンスを作成するためのファクトリも{@link provide}経由で公開する(`routerFactory`というキーで取得可能) | ||||
|  */ | ||||
| export function setupRouter(app: App) { | ||||
| 	app.provide('routerFactory', createRouterImpl); | ||||
|  | ||||
| 	const mainRouter = createRouterImpl(location.pathname + location.search + location.hash); | ||||
|  | ||||
| 	window.history.replaceState({ key: mainRouter.getCurrentKey() }, '', location.href); | ||||
|  | ||||
| 	window.addEventListener('popstate', (event) => { | ||||
| 		mainRouter.replace(location.pathname + location.search + location.hash, event.state?.key); | ||||
| 	}); | ||||
|  | ||||
| 	mainRouter.addListener('push', ctx => { | ||||
| 		window.history.pushState({ key: ctx.key }, '', ctx.path); | ||||
| 	}); | ||||
|  | ||||
| 	setMainRouter(mainRouter); | ||||
| } | ||||
							
								
								
									
										163
									
								
								packages/frontend/src/global/router/main.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										163
									
								
								packages/frontend/src/global/router/main.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,163 @@ | ||||
| /* | ||||
|  * SPDX-FileCopyrightText: syuilo and other misskey contributors | ||||
|  * SPDX-License-Identifier: AGPL-3.0-only | ||||
|  */ | ||||
|  | ||||
| import { ShallowRef } from 'vue'; | ||||
| import { EventEmitter } from 'eventemitter3'; | ||||
| import { IRouter, Resolved, RouteDef, RouterEvent } from '@/nirax.js'; | ||||
|  | ||||
| function getMainRouter(): IRouter { | ||||
| 	const router = mainRouterHolder; | ||||
| 	if (!router) { | ||||
| 		throw new Error('mainRouter is not found.'); | ||||
| 	} | ||||
|  | ||||
| 	return router; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * メインルータを設定する。一度設定すると、それ以降は変更できない。 | ||||
|  * {@link setupRouter}から呼び出されることのみを想定している。 | ||||
|  */ | ||||
| export function setMainRouter(router: IRouter) { | ||||
| 	if (mainRouterHolder) { | ||||
| 		throw new Error('mainRouter is already exists.'); | ||||
| 	} | ||||
|  | ||||
| 	mainRouterHolder = router; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * {@link mainRouter}用のプロキシ実装。 | ||||
|  * {@link mainRouter}は起動シーケンスの一部にて初期化されるため、僅かにundefinedになる期間がある。 | ||||
|  * その僅かな期間のためだけに型をundefined込みにしたくないのでこのクラスを緩衝材として使用する。 | ||||
|  */ | ||||
| class MainRouterProxy implements IRouter { | ||||
| 	private supplier: () => IRouter; | ||||
|  | ||||
| 	constructor(supplier: () => IRouter) { | ||||
| 		this.supplier = supplier; | ||||
| 	} | ||||
|  | ||||
| 	get current(): Resolved { | ||||
| 		return this.supplier().current; | ||||
| 	} | ||||
|  | ||||
| 	get currentRef(): ShallowRef<Resolved> { | ||||
| 		return this.supplier().currentRef; | ||||
| 	} | ||||
|  | ||||
| 	get currentRoute(): ShallowRef<RouteDef> { | ||||
| 		return this.supplier().currentRoute; | ||||
| 	} | ||||
|  | ||||
| 	get navHook(): ((path: string, flag?: any) => boolean) | null { | ||||
| 		return this.supplier().navHook; | ||||
| 	} | ||||
|  | ||||
| 	set navHook(value) { | ||||
| 		this.supplier().navHook = value; | ||||
| 	} | ||||
|  | ||||
| 	getCurrentKey(): string { | ||||
| 		return this.supplier().getCurrentKey(); | ||||
| 	} | ||||
|  | ||||
| 	getCurrentPath(): any { | ||||
| 		return this.supplier().getCurrentPath(); | ||||
| 	} | ||||
|  | ||||
| 	push(path: string, flag?: any): void { | ||||
| 		this.supplier().push(path, flag); | ||||
| 	} | ||||
|  | ||||
| 	replace(path: string, key?: string | null): void { | ||||
| 		this.supplier().replace(path, key); | ||||
| 	} | ||||
|  | ||||
| 	resolve(path: string): Resolved | null { | ||||
| 		return this.supplier().resolve(path); | ||||
| 	} | ||||
|  | ||||
| 	eventNames(): Array<EventEmitter.EventNames<RouterEvent>> { | ||||
| 		return this.supplier().eventNames(); | ||||
| 	} | ||||
|  | ||||
| 	listeners<T extends EventEmitter.EventNames<RouterEvent>>( | ||||
| 		event: T, | ||||
| 	): Array<EventEmitter.EventListener<RouterEvent, T>> { | ||||
| 		return this.supplier().listeners(event); | ||||
| 	} | ||||
|  | ||||
| 	listenerCount( | ||||
| 		event: EventEmitter.EventNames<RouterEvent>, | ||||
| 	): number { | ||||
| 		return this.supplier().listenerCount(event); | ||||
| 	} | ||||
|  | ||||
| 	emit<T extends EventEmitter.EventNames<RouterEvent>>( | ||||
| 		event: T, | ||||
| 		...args: EventEmitter.EventArgs<RouterEvent, T> | ||||
| 	): boolean { | ||||
| 		return this.supplier().emit(event, ...args); | ||||
| 	} | ||||
|  | ||||
| 	on<T extends EventEmitter.EventNames<RouterEvent>>( | ||||
| 		event: T, | ||||
| 		fn: EventEmitter.EventListener<RouterEvent, T>, | ||||
| 		context?: any, | ||||
| 	): this { | ||||
| 		this.supplier().on(event, fn, context); | ||||
| 		return this; | ||||
| 	} | ||||
|  | ||||
| 	addListener<T extends EventEmitter.EventNames<RouterEvent>>( | ||||
| 		event: T, | ||||
| 		fn: EventEmitter.EventListener<RouterEvent, T>, | ||||
| 		context?: any, | ||||
| 	): this { | ||||
| 		this.supplier().addListener(event, fn, context); | ||||
| 		return this; | ||||
| 	} | ||||
|  | ||||
| 	once<T extends EventEmitter.EventNames<RouterEvent>>( | ||||
| 		event: T, | ||||
| 		fn: EventEmitter.EventListener<RouterEvent, T>, | ||||
| 		context?: any, | ||||
| 	): this { | ||||
| 		this.supplier().once(event, fn, context); | ||||
| 		return this; | ||||
| 	} | ||||
|  | ||||
| 	removeListener<T extends EventEmitter.EventNames<RouterEvent>>( | ||||
| 		event: T, | ||||
| 		fn?: EventEmitter.EventListener<RouterEvent, T>, | ||||
| 		context?: any, | ||||
| 		once?: boolean, | ||||
| 	): this { | ||||
| 		this.supplier().removeListener(event, fn, context, once); | ||||
| 		return this; | ||||
| 	} | ||||
|  | ||||
| 	off<T extends EventEmitter.EventNames<RouterEvent>>( | ||||
| 		event: T, | ||||
| 		fn?: EventEmitter.EventListener<RouterEvent, T>, | ||||
| 		context?: any, | ||||
| 		once?: boolean, | ||||
| 	): this { | ||||
| 		this.supplier().off(event, fn, context, once); | ||||
| 		return this; | ||||
| 	} | ||||
|  | ||||
| 	removeAllListeners( | ||||
| 		event?: EventEmitter.EventNames<RouterEvent>, | ||||
| 	): this { | ||||
| 		this.supplier().removeAllListeners(event); | ||||
| 		return this; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| let mainRouterHolder: IRouter | null = null; | ||||
|  | ||||
| export const mainRouter: IRouter = new MainRouterProxy(getMainRouter); | ||||
							
								
								
									
										30
									
								
								packages/frontend/src/global/router/supplier.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										30
									
								
								packages/frontend/src/global/router/supplier.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,30 @@ | ||||
| /* | ||||
|  * SPDX-FileCopyrightText: syuilo and other misskey contributors | ||||
|  * SPDX-License-Identifier: AGPL-3.0-only | ||||
|  */ | ||||
|  | ||||
| import { inject } from 'vue'; | ||||
| import { IRouter, Router } from '@/nirax.js'; | ||||
| import { mainRouter } from '@/global/router/main.js'; | ||||
|  | ||||
| /** | ||||
|  * メインの{@link Router}を取得する。 | ||||
|  * あらかじめ{@link setupRouter}を実行しておく必要がある({@link provide}により{@link IRouter}のインスタンスを注入可能であるならばこの限りではない) | ||||
|  */ | ||||
| export function useRouter(): IRouter { | ||||
| 	return inject<Router | null>('router', null) ?? mainRouter; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * 任意の{@link Router}を取得するためのファクトリを取得する。 | ||||
|  * あらかじめ{@link setupRouter}を実行しておく必要がある。 | ||||
|  */ | ||||
| export function useRouterFactory(): (path: string) => IRouter { | ||||
| 	const factory = inject<(path: string) => IRouter>('routerFactory'); | ||||
| 	if (!factory) { | ||||
| 		console.error('routerFactory is not defined.'); | ||||
| 		throw new Error('routerFactory is not defined.'); | ||||
| 	} | ||||
|  | ||||
| 	return factory; | ||||
| } | ||||
| @@ -16,13 +16,13 @@ | ||||
| 	<!-- https://developer.mozilla.org/en-US/docs/Web/HTTP/CSP --> | ||||
| 	<meta | ||||
| 		http-equiv="Content-Security-Policy" | ||||
| 		content="default-src 'self'; | ||||
| 		content="default-src 'self' https://newassets.hcaptcha.com/ https://challenges.cloudflare.com/ http://localhost:7493/; | ||||
| 			worker-src 'self'; | ||||
| 			script-src 'self' 'unsafe-eval'; | ||||
| 			script-src 'self' 'unsafe-eval' https://*.hcaptcha.com https://challenges.cloudflare.com; | ||||
| 			style-src 'self' 'unsafe-inline'; | ||||
| 			img-src 'self' data: www.google.com xn--931a.moe localhost:3000 localhost:5173 127.0.0.1:5173 127.0.0.1:3000; | ||||
| 			img-src 'self' data: blob: www.google.com xn--931a.moe localhost:3000 localhost:5173 127.0.0.1:5173 127.0.0.1:3000; | ||||
| 			media-src 'self' localhost:3000 localhost:5173 127.0.0.1:5173 127.0.0.1:3000; | ||||
| 			connect-src 'self' localhost:3000 localhost:5173 127.0.0.1:5173 127.0.0.1:3000;" | ||||
| 			connect-src 'self' localhost:3000 localhost:5173 127.0.0.1:5173 127.0.0.1:3000 https://newassets.hcaptcha.com;" | ||||
| 	/> | ||||
| 	<meta property="og:site_name" content="[DEV BUILD] Misskey" /> | ||||
| 	<meta name="viewport" content="width=device-width, initial-scale=1"> | ||||
|   | ||||
| @@ -5,7 +5,7 @@ | ||||
|  | ||||
| import { computed, reactive } from 'vue'; | ||||
| import * as Misskey from 'misskey-js'; | ||||
| import { api } from '@/os.js'; | ||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||
| import { miLocalStorage } from '@/local-storage.js'; | ||||
| import { DEFAULT_INFO_IMAGE_URL, DEFAULT_NOT_FOUND_IMAGE_URL, DEFAULT_SERVER_ERROR_IMAGE_URL } from '@/const.js'; | ||||
|  | ||||
| @@ -26,7 +26,7 @@ export const infoImageUrl = computed(() => instance.infoImageUrl ?? DEFAULT_INFO | ||||
| export const notFoundImageUrl = computed(() => instance.notFoundImageUrl ?? DEFAULT_NOT_FOUND_IMAGE_URL); | ||||
|  | ||||
| export async function fetchInstance() { | ||||
| 	const meta = await api('meta', { | ||||
| 	const meta = await misskeyApi('meta', { | ||||
| 		detail: false, | ||||
| 	}); | ||||
|  | ||||
|   | ||||
| @@ -5,11 +5,11 @@ | ||||
|  | ||||
| // NIRAX --- A lightweight router | ||||
|  | ||||
| import { EventEmitter } from 'eventemitter3'; | ||||
| import { Component, onMounted, shallowRef, ShallowRef } from 'vue'; | ||||
| import { EventEmitter } from 'eventemitter3'; | ||||
| import { safeURIDecode } from '@/scripts/safe-uri-decode.js'; | ||||
|  | ||||
| type RouteDef = { | ||||
| export type RouteDef = { | ||||
| 	path: string; | ||||
| 	component: Component; | ||||
| 	query?: Record<string, string>; | ||||
| @@ -27,6 +27,27 @@ type ParsedPath = (string | { | ||||
| 	optional?: boolean; | ||||
| })[]; | ||||
|  | ||||
| export type RouterEvent = { | ||||
| 	change: (ctx: { | ||||
| 		beforePath: string; | ||||
| 		path: string; | ||||
| 		resolved: Resolved; | ||||
| 		key: string; | ||||
| 	}) => void; | ||||
| 	replace: (ctx: { | ||||
| 		path: string; | ||||
| 		key: string; | ||||
| 	}) => void; | ||||
| 	push: (ctx: { | ||||
| 		beforePath: string; | ||||
| 		path: string; | ||||
| 		route: RouteDef | null; | ||||
| 		props: Map<string, string> | null; | ||||
| 		key: string; | ||||
| 	}) => void; | ||||
| 	same: () => void; | ||||
| } | ||||
|  | ||||
| export type Resolved = { route: RouteDef; props: Map<string, string | boolean>; child?: Resolved; }; | ||||
|  | ||||
| function parsePath(path: string): ParsedPath { | ||||
| @@ -54,26 +75,85 @@ function parsePath(path: string): ParsedPath { | ||||
| 	return res; | ||||
| } | ||||
|  | ||||
| export class Router extends EventEmitter<{ | ||||
| 	change: (ctx: { | ||||
| 		beforePath: string; | ||||
| 		path: string; | ||||
| 		resolved: Resolved; | ||||
| 		key: string; | ||||
| 	}) => void; | ||||
| 	replace: (ctx: { | ||||
| 		path: string; | ||||
| 		key: string; | ||||
| 	}) => void; | ||||
| 	push: (ctx: { | ||||
| 		beforePath: string; | ||||
| 		path: string; | ||||
| 		route: RouteDef | null; | ||||
| 		props: Map<string, string> | null; | ||||
| 		key: string; | ||||
| 	}) => void; | ||||
| 	same: () => void; | ||||
| }> { | ||||
| export interface IRouter extends EventEmitter<RouterEvent> { | ||||
| 	current: Resolved; | ||||
| 	currentRef: ShallowRef<Resolved>; | ||||
| 	currentRoute: ShallowRef<RouteDef>; | ||||
| 	navHook: ((path: string, flag?: any) => boolean) | null; | ||||
|  | ||||
| 	resolve(path: string): Resolved | null; | ||||
|  | ||||
| 	getCurrentPath(): any; | ||||
|  | ||||
| 	getCurrentKey(): string; | ||||
|  | ||||
| 	push(path: string, flag?: any): void; | ||||
|  | ||||
| 	replace(path: string, key?: string | null): void; | ||||
|  | ||||
| 	/** @see EventEmitter */ | ||||
| 	eventNames(): Array<EventEmitter.EventNames<RouterEvent>>; | ||||
|  | ||||
| 	/** @see EventEmitter */ | ||||
| 	listeners<T extends EventEmitter.EventNames<RouterEvent>>( | ||||
| 		event: T | ||||
| 	): Array<EventEmitter.EventListener<RouterEvent, T>>; | ||||
|  | ||||
| 	/** @see EventEmitter */ | ||||
| 	listenerCount( | ||||
| 		event: EventEmitter.EventNames<RouterEvent> | ||||
| 	): number; | ||||
|  | ||||
| 	/** @see EventEmitter */ | ||||
| 	emit<T extends EventEmitter.EventNames<RouterEvent>>( | ||||
| 		event: T, | ||||
| 		...args: EventEmitter.EventArgs<RouterEvent, T> | ||||
| 	): boolean; | ||||
|  | ||||
| 	/** @see EventEmitter */ | ||||
| 	on<T extends EventEmitter.EventNames<RouterEvent>>( | ||||
| 		event: T, | ||||
| 		fn: EventEmitter.EventListener<RouterEvent, T>, | ||||
| 		context?: any | ||||
| 	): this; | ||||
|  | ||||
| 	/** @see EventEmitter */ | ||||
| 	addListener<T extends EventEmitter.EventNames<RouterEvent>>( | ||||
| 		event: T, | ||||
| 		fn: EventEmitter.EventListener<RouterEvent, T>, | ||||
| 		context?: any | ||||
| 	): this; | ||||
|  | ||||
| 	/** @see EventEmitter */ | ||||
| 	once<T extends EventEmitter.EventNames<RouterEvent>>( | ||||
| 		event: T, | ||||
| 		fn: EventEmitter.EventListener<RouterEvent, T>, | ||||
| 		context?: any | ||||
| 	): this; | ||||
|  | ||||
| 	/** @see EventEmitter */ | ||||
| 	removeListener<T extends EventEmitter.EventNames<RouterEvent>>( | ||||
| 		event: T, | ||||
| 		fn?: EventEmitter.EventListener<RouterEvent, T>, | ||||
| 		context?: any, | ||||
| 		once?: boolean | undefined | ||||
| 	): this; | ||||
|  | ||||
| 	/** @see EventEmitter */ | ||||
| 	off<T extends EventEmitter.EventNames<RouterEvent>>( | ||||
| 		event: T, | ||||
| 		fn?: EventEmitter.EventListener<RouterEvent, T>, | ||||
| 		context?: any, | ||||
| 		once?: boolean | undefined | ||||
| 	): this; | ||||
|  | ||||
| 	/** @see EventEmitter */ | ||||
| 	removeAllListeners( | ||||
| 		event?: EventEmitter.EventNames<RouterEvent> | ||||
| 	): this; | ||||
| } | ||||
|  | ||||
| export class Router extends EventEmitter<RouterEvent> implements IRouter { | ||||
| 	private routes: RouteDef[]; | ||||
| 	public current: Resolved; | ||||
| 	public currentRef: ShallowRef<Resolved> = shallowRef(); | ||||
| @@ -277,7 +357,7 @@ export class Router extends EventEmitter<{ | ||||
| 	} | ||||
| } | ||||
|  | ||||
| export function useScrollPositionManager(getScrollContainer: () => HTMLElement, router: Router) { | ||||
| export function useScrollPositionManager(getScrollContainer: () => HTMLElement, router: IRouter) { | ||||
| 	const scrollPosStore = new Map<string, number>(); | ||||
|  | ||||
| 	onMounted(() => { | ||||
|   | ||||
| @@ -5,12 +5,11 @@ | ||||
|  | ||||
| // TODO: なんでもかんでもos.tsに突っ込むのやめたいのでよしなに分割する | ||||
|  | ||||
| import { pendingApiRequestsCount, api, apiGet } from '@/scripts/api.js'; | ||||
| export { pendingApiRequestsCount, api, apiGet }; | ||||
| import { Component, markRaw, Ref, ref, defineAsyncComponent } from 'vue'; | ||||
| import { EventEmitter } from 'eventemitter3'; | ||||
| import insertTextAtCursor from 'insert-text-at-cursor'; | ||||
| import * as Misskey from 'misskey-js'; | ||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||
| import { i18n } from '@/i18n.js'; | ||||
| import { defaultStore } from '@/store.js'; | ||||
| import MkPostFormDialog from '@/components/MkPostFormDialog.vue'; | ||||
| @@ -34,7 +33,7 @@ export const apiWithDialog = (( | ||||
| 	data: Record<string, any> = {}, | ||||
| 	token?: string | null | undefined, | ||||
| ) => { | ||||
| 	const promise = api(endpoint, data, token); | ||||
| 	const promise = misskeyApi(endpoint, data, token); | ||||
| 	promiseDialog(promise, null, async (err) => { | ||||
| 		let title = null; | ||||
| 		let text = err.message + '\n' + (err as any).id; | ||||
| @@ -84,7 +83,7 @@ export const apiWithDialog = (( | ||||
| 	}); | ||||
|  | ||||
| 	return promise; | ||||
| }) as typeof api; | ||||
| }) as typeof misskeyApi; | ||||
|  | ||||
| export function promiseDialog<T extends Promise<any>>( | ||||
| 	promise: T, | ||||
| @@ -623,7 +622,7 @@ export function checkExistence(fileData: ArrayBuffer): Promise<any> { | ||||
| 		const data = new FormData(); | ||||
| 		data.append('md5', getMD5(fileData)); | ||||
|  | ||||
| 		os.api('drive/files/find-by-hash', { | ||||
| 		api('drive/files/find-by-hash', { | ||||
| 			md5: getMD5(fileData) | ||||
| 		}).then(resp => { | ||||
| 			resolve(resp.length > 0 ? resp[0] : null); | ||||
|   | ||||
| @@ -29,7 +29,7 @@ import { ref, computed } from 'vue'; | ||||
| import * as Misskey from 'misskey-js'; | ||||
| import MkButton from '@/components/MkButton.vue'; | ||||
| import { version } from '@/config.js'; | ||||
| import * as os from '@/os.js'; | ||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||
| import { unisonReload } from '@/scripts/unison-reload.js'; | ||||
| import { i18n } from '@/i18n.js'; | ||||
| import { definePageMetadata } from '@/scripts/page-metadata.js'; | ||||
| @@ -46,7 +46,7 @@ const loaded = ref(false); | ||||
| const serverIsDead = ref(false); | ||||
| const meta = ref<Misskey.entities.MetaResponse | null>(null); | ||||
|  | ||||
| os.api('meta', { | ||||
| misskeyApi('meta', { | ||||
| 	detail: false, | ||||
| }).then(res => { | ||||
| 	loaded.value = true; | ||||
|   | ||||
| @@ -114,7 +114,7 @@ import FormSplit from '@/components/form/split.vue'; | ||||
| import MkFolder from '@/components/MkFolder.vue'; | ||||
| import MkKeyValue from '@/components/MkKeyValue.vue'; | ||||
| import MkInstanceStats from '@/components/MkInstanceStats.vue'; | ||||
| import * as os from '@/os.js'; | ||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||
| import number from '@/filters/number.js'; | ||||
| import { i18n } from '@/i18n.js'; | ||||
| import { definePageMetadata } from '@/scripts/page-metadata.js'; | ||||
| @@ -136,7 +136,7 @@ watch(tab, () => { | ||||
| 	} | ||||
| }); | ||||
|  | ||||
| const initStats = () => os.api('stats', { | ||||
| const initStats = () => misskeyApi('stats', { | ||||
| }).then((res) => { | ||||
| 	stats.value = res; | ||||
| }); | ||||
|   | ||||
| @@ -79,6 +79,7 @@ import MkUserCardMini from '@/components/MkUserCardMini.vue'; | ||||
| import MkInfo from '@/components/MkInfo.vue'; | ||||
| import bytes from '@/filters/bytes.js'; | ||||
| import * as os from '@/os.js'; | ||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||
| import { i18n } from '@/i18n.js'; | ||||
| import { definePageMetadata } from '@/scripts/page-metadata.js'; | ||||
| import { iAmAdmin, iAmModerator } from '@/account.js'; | ||||
| @@ -93,8 +94,8 @@ const props = defineProps<{ | ||||
| }>(); | ||||
|  | ||||
| async function fetch() { | ||||
| 	file.value = await os.api('drive/files/show', { fileId: props.fileId }); | ||||
| 	info.value = await os.api('admin/drive/show-file', { fileId: props.fileId }); | ||||
| 	file.value = await misskeyApi('drive/files/show', { fileId: props.fileId }); | ||||
| 	info.value = await misskeyApi('admin/drive/show-file', { fileId: props.fileId }); | ||||
| 	isSensitive.value = file.value.isSensitive; | ||||
| } | ||||
|  | ||||
| @@ -113,7 +114,7 @@ async function del() { | ||||
| } | ||||
|  | ||||
| async function toggleIsSensitive(v) { | ||||
| 	await os.api('drive/files/update', { fileId: props.fileId, isSensitive: v }); | ||||
| 	await misskeyApi('drive/files/update', { fileId: props.fileId, isSensitive: v }); | ||||
| 	isSensitive.value = v; | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -220,6 +220,7 @@ import FormSuspense from '@/components/form/suspense.vue'; | ||||
| import MkFileListForAdmin from '@/components/MkFileListForAdmin.vue'; | ||||
| import MkInfo from '@/components/MkInfo.vue'; | ||||
| import * as os from '@/os.js'; | ||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||
| import { url } from '@/config.js'; | ||||
| import { acct } from '@/filters/user.js'; | ||||
| import { definePageMetadata } from '@/scripts/page-metadata.js'; | ||||
| @@ -266,11 +267,11 @@ const announcementsPagination = { | ||||
| const expandedRoles = ref([]); | ||||
|  | ||||
| function createFetcher() { | ||||
| 	return () => Promise.all([os.api('users/show', { | ||||
| 	return () => Promise.all([misskeyApi('users/show', { | ||||
| 		userId: props.userId, | ||||
| 	}), os.api('admin/show-user', { | ||||
| 	}), misskeyApi('admin/show-user', { | ||||
| 		userId: props.userId, | ||||
| 	}), iAmAdmin ? os.api('admin/get-user-ips', { | ||||
| 	}), iAmAdmin ? misskeyApi('admin/get-user-ips', { | ||||
| 		userId: props.userId, | ||||
| 	}) : Promise.resolve(null)]).then(([_user, _info, _ips]) => { | ||||
| 		user.value = _user; | ||||
| @@ -283,7 +284,7 @@ function createFetcher() { | ||||
| 		moderationNote.value = info.value.moderationNote; | ||||
|  | ||||
| 		watch(moderationNote, async () => { | ||||
| 			await os.api('admin/update-user-note', { userId: user.value.id, text: moderationNote.value }); | ||||
| 			await misskeyApi('admin/update-user-note', { userId: user.value.id, text: moderationNote.value }); | ||||
| 			await refreshUser(); | ||||
| 		}); | ||||
| 	}); | ||||
| @@ -306,7 +307,7 @@ async function resetPassword() { | ||||
| 	if (confirm.canceled) { | ||||
| 		return; | ||||
| 	} else { | ||||
| 		const { password } = await os.api('admin/reset-password', { | ||||
| 		const { password } = await misskeyApi('admin/reset-password', { | ||||
| 			userId: user.value.id, | ||||
| 		}); | ||||
| 		os.alert({ | ||||
| @@ -324,7 +325,7 @@ async function toggleSuspend(v) { | ||||
| 	if (confirm.canceled) { | ||||
| 		suspended.value = !v; | ||||
| 	} else { | ||||
| 		await os.api(v ? 'admin/suspend-user' : 'admin/unsuspend-user', { userId: user.value.id }); | ||||
| 		await misskeyApi(v ? 'admin/suspend-user' : 'admin/unsuspend-user', { userId: user.value.id }); | ||||
| 		await refreshUser(); | ||||
| 	} | ||||
| } | ||||
| @@ -336,7 +337,7 @@ async function unsetUserAvatar() { | ||||
| 	}); | ||||
| 	if (confirm.canceled) return; | ||||
| 	const process = async () => { | ||||
| 		await os.api('admin/unset-user-avatar', { userId: user.value.id }); | ||||
| 		await misskeyApi('admin/unset-user-avatar', { userId: user.value.id }); | ||||
| 		os.success(); | ||||
| 	}; | ||||
| 	await process().catch(err => { | ||||
| @@ -355,7 +356,7 @@ async function unsetUserBanner() { | ||||
| 	}); | ||||
| 	if (confirm.canceled) return; | ||||
| 	const process = async () => { | ||||
| 		await os.api('admin/unset-user-banner', { userId: user.value.id }); | ||||
| 		await misskeyApi('admin/unset-user-banner', { userId: user.value.id }); | ||||
| 		os.success(); | ||||
| 	}; | ||||
| 	await process().catch(err => { | ||||
| @@ -374,7 +375,7 @@ async function deleteAllFiles() { | ||||
| 	}); | ||||
| 	if (confirm.canceled) return; | ||||
| 	const process = async () => { | ||||
| 		await os.api('admin/delete-all-files-of-a-user', { userId: user.value.id }); | ||||
| 		await misskeyApi('admin/delete-all-files-of-a-user', { userId: user.value.id }); | ||||
| 		os.success(); | ||||
| 	}; | ||||
| 	await process().catch(err => { | ||||
| @@ -411,7 +412,7 @@ async function deleteAccount() { | ||||
| } | ||||
|  | ||||
| async function assignRole() { | ||||
| 	const roles = await os.api('admin/roles/list'); | ||||
| 	const roles = await misskeyApi('admin/roles/list'); | ||||
|  | ||||
| 	const { canceled, result: roleId } = await os.select({ | ||||
| 		title: i18n.ts._role.chooseRoleToAssign, | ||||
| @@ -495,7 +496,7 @@ watch(() => props.userId, () => { | ||||
| }); | ||||
|  | ||||
| watch(user, () => { | ||||
| 	os.api('ap/get', { | ||||
| 	misskeyApi('ap/get', { | ||||
| 		uri: user.value.uri ?? `${url}/users/${user.value.id}`, | ||||
| 	}).then(res => { | ||||
| 		ap.value = res; | ||||
|   | ||||
| @@ -97,6 +97,7 @@ import MkFolder from '@/components/MkFolder.vue'; | ||||
| import MkSelect from '@/components/MkSelect.vue'; | ||||
| import FormSplit from '@/components/form/split.vue'; | ||||
| import * as os from '@/os.js'; | ||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||
| import { i18n } from '@/i18n.js'; | ||||
| import { definePageMetadata } from '@/scripts/page-metadata.js'; | ||||
|  | ||||
| @@ -109,7 +110,7 @@ const daysOfWeek: string[] = [i18n.ts._weekday.sunday, i18n.ts._weekday.monday, | ||||
| const filterType = ref('all'); | ||||
| let publishing: boolean | null = null; | ||||
|  | ||||
| os.api('admin/ad/list', { publishing: publishing }).then(adsResponse => { | ||||
| misskeyApi('admin/ad/list', { publishing: publishing }).then(adsResponse => { | ||||
| 	if (adsResponse != null) { | ||||
| 		ads.value = adsResponse.map(r => { | ||||
| 			const exdate = new Date(r.expiresAt); | ||||
| @@ -175,7 +176,7 @@ function remove(ad) { | ||||
|  | ||||
| function save(ad) { | ||||
| 	if (ad.id == null) { | ||||
| 		os.api('admin/ad/create', { | ||||
| 		misskeyApi('admin/ad/create', { | ||||
| 			...ad, | ||||
| 			expiresAt: new Date(ad.expiresAt).getTime(), | ||||
| 			startsAt: new Date(ad.startsAt).getTime(), | ||||
| @@ -192,7 +193,7 @@ function save(ad) { | ||||
| 			}); | ||||
| 		}); | ||||
| 	} else { | ||||
| 		os.api('admin/ad/update', { | ||||
| 		misskeyApi('admin/ad/update', { | ||||
| 			...ad, | ||||
| 			expiresAt: new Date(ad.expiresAt).getTime(), | ||||
| 			startsAt: new Date(ad.startsAt).getTime(), | ||||
| @@ -211,7 +212,7 @@ function save(ad) { | ||||
| } | ||||
|  | ||||
| function more() { | ||||
| 	os.api('admin/ad/list', { untilId: ads.value.reduce((acc, ad) => ad.id != null ? ad : acc).id, publishing: publishing }).then(adsResponse => { | ||||
| 	misskeyApi('admin/ad/list', { untilId: ads.value.reduce((acc, ad) => ad.id != null ? ad : acc).id, publishing: publishing }).then(adsResponse => { | ||||
| 		if (adsResponse == null) return; | ||||
| 		ads.value = ads.value.concat(adsResponse.map(r => { | ||||
| 			const exdate = new Date(r.expiresAt); | ||||
| @@ -228,7 +229,7 @@ function more() { | ||||
| } | ||||
|  | ||||
| function refresh() { | ||||
| 	os.api('admin/ad/list', { publishing: publishing }).then(adsResponse => { | ||||
| 	misskeyApi('admin/ad/list', { publishing: publishing }).then(adsResponse => { | ||||
| 		if (adsResponse == null) return; | ||||
| 		ads.value = adsResponse.map(r => { | ||||
| 			const exdate = new Date(r.expiresAt); | ||||
|   | ||||
| @@ -104,6 +104,7 @@ import MkSwitch from '@/components/MkSwitch.vue'; | ||||
| import MkRadios from '@/components/MkRadios.vue'; | ||||
| import MkInfo from '@/components/MkInfo.vue'; | ||||
| import * as os from '@/os.js'; | ||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||
| import { i18n } from '@/i18n.js'; | ||||
| import { definePageMetadata } from '@/scripts/page-metadata.js'; | ||||
| import MkFolder from '@/components/MkFolder.vue'; | ||||
| @@ -162,7 +163,7 @@ function del(announcement) { | ||||
| 	}).then(({ canceled }) => { | ||||
| 		if (canceled) return; | ||||
| 		announcements.value = announcements.value.filter(x => x !== announcement); | ||||
| 		os.api('admin/announcements/delete', announcement); | ||||
| 		misskeyApi('admin/announcements/delete', announcement); | ||||
| 	}); | ||||
| } | ||||
|  | ||||
| @@ -189,7 +190,7 @@ function fetch(resetOffset = false): void { | ||||
| 		offset.value = 0; | ||||
| 	} | ||||
|  | ||||
| 	os.api('admin/announcements/list', { | ||||
| 	misskeyApi('admin/announcements/list', { | ||||
| 		offsetMode: true, | ||||
| 		offset: offset.value, | ||||
| 		limit: 10, | ||||
|   | ||||
| @@ -10,6 +10,7 @@ SPDX-License-Identifier: AGPL-3.0-only | ||||
| 			<MkRadios v-model="provider"> | ||||
| 				<option :value="null">{{ i18n.ts.none }} ({{ i18n.ts.notRecommended }})</option> | ||||
| 				<option value="hcaptcha">hCaptcha</option> | ||||
| 				<option value="mcaptcha">mCaptcha</option> | ||||
| 				<option value="recaptcha">reCAPTCHA</option> | ||||
| 				<option value="turnstile">Turnstile</option> | ||||
| 			</MkRadios> | ||||
| @@ -28,6 +29,24 @@ SPDX-License-Identifier: AGPL-3.0-only | ||||
| 					<MkCaptcha provider="hcaptcha" :sitekey="hcaptchaSiteKey || '10000000-ffff-ffff-ffff-000000000001'"/> | ||||
| 				</FormSlot> | ||||
| 			</template> | ||||
| 			<template v-else-if="provider === 'mcaptcha'"> | ||||
| 				<MkInput v-model="mcaptchaSiteKey"> | ||||
| 					<template #prefix><i class="ti ti-key"></i></template> | ||||
| 					<template #label>{{ i18n.ts.mcaptchaSiteKey }}</template> | ||||
| 				</MkInput> | ||||
| 				<MkInput v-model="mcaptchaSecretKey"> | ||||
| 					<template #prefix><i class="ti ti-key"></i></template> | ||||
| 					<template #label>{{ i18n.ts.mcaptchaSecretKey }}</template> | ||||
| 				</MkInput> | ||||
| 				<MkInput v-model="mcaptchaInstanceUrl"> | ||||
| 					<template #prefix><i class="ti ti-link"></i></template> | ||||
| 					<template #label>{{ i18n.ts.mcaptchaInstanceUrl }}</template> | ||||
| 				</MkInput> | ||||
| 				<FormSlot v-if="mcaptchaSiteKey && mcaptchaInstanceUrl"> | ||||
| 					<template #label>{{ i18n.ts.preview }}</template> | ||||
| 					<MkCaptcha provider="mcaptcha" :sitekey="mcaptchaSiteKey" :instanceUrl="mcaptchaInstanceUrl"/> | ||||
| 				</FormSlot> | ||||
| 			</template> | ||||
| 			<template v-else-if="provider === 'recaptcha'"> | ||||
| 				<MkInput v-model="recaptchaSiteKey"> | ||||
| 					<template #prefix><i class="ti ti-key"></i></template> | ||||
| @@ -72,6 +91,7 @@ import MkButton from '@/components/MkButton.vue'; | ||||
| import FormSuspense from '@/components/form/suspense.vue'; | ||||
| import FormSlot from '@/components/form/slot.vue'; | ||||
| import * as os from '@/os.js'; | ||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||
| import { fetchInstance } from '@/instance.js'; | ||||
| import { i18n } from '@/i18n.js'; | ||||
|  | ||||
| @@ -80,21 +100,30 @@ const MkCaptcha = defineAsyncComponent(() => import('@/components/MkCaptcha.vue' | ||||
| const provider = ref<CaptchaProvider | null>(null); | ||||
| const hcaptchaSiteKey = ref<string | null>(null); | ||||
| const hcaptchaSecretKey = ref<string | null>(null); | ||||
| const mcaptchaSiteKey = ref<string | null>(null); | ||||
| const mcaptchaSecretKey = ref<string | null>(null); | ||||
| const mcaptchaInstanceUrl = ref<string | null>(null); | ||||
| const recaptchaSiteKey = ref<string | null>(null); | ||||
| const recaptchaSecretKey = ref<string | null>(null); | ||||
| const turnstileSiteKey = ref<string | null>(null); | ||||
| const turnstileSecretKey = ref<string | null>(null); | ||||
|  | ||||
| async function init() { | ||||
| 	const meta = await os.api('admin/meta'); | ||||
| 	const meta = await misskeyApi('admin/meta'); | ||||
| 	hcaptchaSiteKey.value = meta.hcaptchaSiteKey; | ||||
| 	hcaptchaSecretKey.value = meta.hcaptchaSecretKey; | ||||
| 	mcaptchaSiteKey.value = meta.mcaptchaSiteKey; | ||||
| 	mcaptchaSecretKey.value = meta.mcaptchaSecretKey; | ||||
| 	mcaptchaInstanceUrl.value = meta.mcaptchaInstanceUrl; | ||||
| 	recaptchaSiteKey.value = meta.recaptchaSiteKey; | ||||
| 	recaptchaSecretKey.value = meta.recaptchaSecretKey; | ||||
| 	turnstileSiteKey.value = meta.turnstileSiteKey; | ||||
| 	turnstileSecretKey.value = meta.turnstileSecretKey; | ||||
|  | ||||
| 	provider.value = meta.enableHcaptcha ? 'hcaptcha' : meta.enableRecaptcha ? 'recaptcha' : meta.enableTurnstile ? 'turnstile' : null; | ||||
| 	provider.value = meta.enableHcaptcha ? 'hcaptcha' : | ||||
| 		meta.enableRecaptcha ? 'recaptcha' : | ||||
| 		meta.enableTurnstile ? 'turnstile' : | ||||
| 		meta.enableMcaptcha ? 'mcaptcha' : null; | ||||
| } | ||||
|  | ||||
| function save() { | ||||
| @@ -102,6 +131,10 @@ function save() { | ||||
| 		enableHcaptcha: provider.value === 'hcaptcha', | ||||
| 		hcaptchaSiteKey: hcaptchaSiteKey.value, | ||||
| 		hcaptchaSecretKey: hcaptchaSecretKey.value, | ||||
| 		enableMcaptcha: provider.value === 'mcaptcha', | ||||
| 		mcaptchaSiteKey: mcaptchaSiteKey.value, | ||||
| 		mcaptchaSecretKey: mcaptchaSecretKey.value, | ||||
| 		mcaptchaInstanceUrl: mcaptchaInstanceUrl.value, | ||||
| 		enableRecaptcha: provider.value === 'recaptcha', | ||||
| 		recaptchaSiteKey: recaptchaSiteKey.value, | ||||
| 		recaptchaSecretKey: recaptchaSecretKey.value, | ||||
|   | ||||
| @@ -101,6 +101,7 @@ import MkInput from '@/components/MkInput.vue'; | ||||
| import MkTextarea from '@/components/MkTextarea.vue'; | ||||
| import FormSuspense from '@/components/form/suspense.vue'; | ||||
| import * as os from '@/os.js'; | ||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||
| import { instance, fetchInstance } from '@/instance.js'; | ||||
| import { i18n } from '@/i18n.js'; | ||||
| import { definePageMetadata } from '@/scripts/page-metadata.js'; | ||||
| @@ -122,7 +123,7 @@ const notFoundImageUrl = ref<string | null>(null); | ||||
| const manifestJsonOverride = ref<string>('{}'); | ||||
|  | ||||
| async function init() { | ||||
| 	const meta = await os.api('admin/meta'); | ||||
| 	const meta = await misskeyApi('admin/meta'); | ||||
| 	iconUrl.value = meta.iconUrl; | ||||
| 	app192IconUrl.value = meta.app192IconUrl; | ||||
| 	app512IconUrl.value = meta.app512IconUrl; | ||||
|   | ||||
| @@ -21,13 +21,13 @@ SPDX-License-Identifier: AGPL-3.0-only | ||||
| import { computed } from 'vue'; | ||||
| import FormSuspense from '@/components/form/suspense.vue'; | ||||
| import MkKeyValue from '@/components/MkKeyValue.vue'; | ||||
| import * as os from '@/os.js'; | ||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||
| import bytes from '@/filters/bytes.js'; | ||||
| import number from '@/filters/number.js'; | ||||
| import { i18n } from '@/i18n.js'; | ||||
| import { definePageMetadata } from '@/scripts/page-metadata.js'; | ||||
|  | ||||
| const databasePromiseFactory = () => os.api('admin/get-table-stats').then(res => Object.entries(res).sort((a, b) => b[1].size - a[1].size)); | ||||
| const databasePromiseFactory = () => misskeyApi('admin/get-table-stats').then(res => Object.entries(res).sort((a, b) => b[1].size - a[1].size)); | ||||
|  | ||||
| const headerActions = computed(() => []); | ||||
|  | ||||
|   | ||||
| @@ -73,6 +73,7 @@ import FormSuspense from '@/components/form/suspense.vue'; | ||||
| import FormSplit from '@/components/form/split.vue'; | ||||
| import FormSection from '@/components/form/section.vue'; | ||||
| import * as os from '@/os.js'; | ||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||
| import { fetchInstance, instance } from '@/instance.js'; | ||||
| import { i18n } from '@/i18n.js'; | ||||
| import { definePageMetadata } from '@/scripts/page-metadata.js'; | ||||
| @@ -87,7 +88,7 @@ const smtpUser = ref<string>(''); | ||||
| const smtpPass = ref<string>(''); | ||||
|  | ||||
| async function init() { | ||||
| 	const meta = await os.api('admin/meta'); | ||||
| 	const meta = await misskeyApi('admin/meta'); | ||||
| 	enableEmail.value = meta.enableEmail; | ||||
| 	email.value = meta.email; | ||||
| 	smtpSecure.value = meta.smtpSecure; | ||||
|   | ||||
| @@ -42,6 +42,7 @@ import MkSwitch from '@/components/MkSwitch.vue'; | ||||
| import FormSuspense from '@/components/form/suspense.vue'; | ||||
| import FormSection from '@/components/form/section.vue'; | ||||
| import * as os from '@/os.js'; | ||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||
| import { fetchInstance } from '@/instance.js'; | ||||
| import { i18n } from '@/i18n.js'; | ||||
| import { definePageMetadata } from '@/scripts/page-metadata.js'; | ||||
| @@ -50,7 +51,7 @@ const deeplAuthKey = ref<string>(''); | ||||
| const deeplIsPro = ref<boolean>(false); | ||||
|  | ||||
| async function init() { | ||||
| 	const meta = await os.api('admin/meta'); | ||||
| 	const meta = await misskeyApi('admin/meta'); | ||||
| 	deeplAuthKey.value = meta.deeplAuthKey; | ||||
| 	deeplIsPro.value = meta.deeplIsPro; | ||||
| } | ||||
|   | ||||
| @@ -42,6 +42,7 @@ import MkInput from '@/components/MkInput.vue'; | ||||
| import MkSelect from '@/components/MkSelect.vue'; | ||||
| import MkFileListForAdmin from '@/components/MkFileListForAdmin.vue'; | ||||
| import * as os from '@/os.js'; | ||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||
| import { i18n } from '@/i18n.js'; | ||||
| import { definePageMetadata } from '@/scripts/page-metadata.js'; | ||||
|  | ||||
| @@ -83,7 +84,7 @@ async function find() { | ||||
| 	}); | ||||
| 	if (canceled) return; | ||||
|  | ||||
| 	os.api('admin/drive/show-file', q.startsWith('http://') || q.startsWith('https://') ? { url: q.trim() } : { fileId: q.trim() }).then(file => { | ||||
| 	misskeyApi('admin/drive/show-file', q.startsWith('http://') || q.startsWith('https://') ? { url: q.trim() } : { fileId: q.trim() }).then(file => { | ||||
| 		show(file); | ||||
| 	}).catch(err => { | ||||
| 		if (err.code === 'NO_SUCH_FILE') { | ||||
|   | ||||
| @@ -34,9 +34,10 @@ import MkSuperMenu from '@/components/MkSuperMenu.vue'; | ||||
| import MkInfo from '@/components/MkInfo.vue'; | ||||
| import { instance } from '@/instance.js'; | ||||
| import * as os from '@/os.js'; | ||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||
| import { lookupUser, lookupUserByEmail } from '@/scripts/lookup-user.js'; | ||||
| import { useRouter } from '@/router.js'; | ||||
| import { PageMetadata, definePageMetadata, provideMetadataReceiver } from '@/scripts/page-metadata.js'; | ||||
| import { useRouter } from '@/global/router/supplier.js'; | ||||
|  | ||||
| const isEmpty = (x: string | null) => x == null || x === ''; | ||||
|  | ||||
| @@ -62,7 +63,7 @@ let noEmailServer = !instance.enableEmail; | ||||
| const thereIsUnresolvedAbuseReport = ref(false); | ||||
| const currentPage = computed(() => router.currentRef.value.child); | ||||
|  | ||||
| os.api('admin/abuse-user-reports', { | ||||
| misskeyApi('admin/abuse-user-reports', { | ||||
| 	state: 'unresolved', | ||||
| 	limit: 1, | ||||
| }).then(reports => { | ||||
| @@ -266,7 +267,7 @@ provideMetadataReceiver((info) => { | ||||
| }); | ||||
|  | ||||
| function invite() { | ||||
| 	os.api('admin/invite/create').then(x => { | ||||
| 	misskeyApi('admin/invite/create').then(x => { | ||||
| 		os.alert({ | ||||
| 			type: 'info', | ||||
| 			text: x[0].code, | ||||
|   | ||||
| @@ -33,6 +33,7 @@ import MkButton from '@/components/MkButton.vue'; | ||||
| import MkTextarea from '@/components/MkTextarea.vue'; | ||||
| import FormSuspense from '@/components/form/suspense.vue'; | ||||
| import * as os from '@/os.js'; | ||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||
| import { fetchInstance } from '@/instance.js'; | ||||
| import { i18n } from '@/i18n.js'; | ||||
| import { definePageMetadata } from '@/scripts/page-metadata.js'; | ||||
| @@ -43,7 +44,7 @@ const sensitiveMediaHosts = ref<string>(''); | ||||
| const tab = ref('block'); | ||||
|  | ||||
| async function init() { | ||||
| 	const meta = await os.api('admin/meta'); | ||||
| 	const meta = await misskeyApi('admin/meta'); | ||||
| 	blockedHosts.value = meta.blockedHosts.join('\n'); | ||||
| 	silencedHosts.value = meta.silencedHosts.join('\n'); | ||||
| 	sensitiveMediaHosts.value = meta.sensitiveMediaHosts.join('\n'); | ||||
|   | ||||
| @@ -59,6 +59,7 @@ import { computed, ref, shallowRef } from 'vue'; | ||||
| import XHeader from './_header_.vue'; | ||||
| import { i18n } from '@/i18n.js'; | ||||
| import * as os from '@/os.js'; | ||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||
| import MkButton from '@/components/MkButton.vue'; | ||||
| import MkFolder from '@/components/MkFolder.vue'; | ||||
| import MkSelect from '@/components/MkSelect.vue'; | ||||
| @@ -93,14 +94,14 @@ async function createWithOptions() { | ||||
| 		count: createCount.value, | ||||
| 	}; | ||||
|  | ||||
| 	const tickets = await os.api('admin/invite/create', options); | ||||
| 	const tickets = await misskeyApi('admin/invite/create', options); | ||||
| 	os.alert({ | ||||
| 		type: 'success', | ||||
| 		title: i18n.ts.inviteCodeCreated, | ||||
| 		text: tickets?.map(x => x.code).join('\n'), | ||||
| 		text: tickets.map(x => x.code).join('\n'), | ||||
| 	}); | ||||
|  | ||||
| 	tickets?.forEach(ticket => pagingComponent.value?.prepend(ticket)); | ||||
| 	tickets.forEach(ticket => pagingComponent.value?.prepend(ticket)); | ||||
| } | ||||
|  | ||||
| function deleted(id: string) { | ||||
|   | ||||
| @@ -71,6 +71,7 @@ import MkInput from '@/components/MkInput.vue'; | ||||
| import MkTextarea from '@/components/MkTextarea.vue'; | ||||
| import FormSuspense from '@/components/form/suspense.vue'; | ||||
| import * as os from '@/os.js'; | ||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||
| import { fetchInstance } from '@/instance.js'; | ||||
| import { i18n } from '@/i18n.js'; | ||||
| import { definePageMetadata } from '@/scripts/page-metadata.js'; | ||||
| @@ -87,7 +88,7 @@ const privacyPolicyUrl = ref<string | null>(null); | ||||
| const urlPreviewDenyList = ref<string | undefined>(''); | ||||
|  | ||||
| async function init() { | ||||
| 	const meta = await os.api('admin/meta'); | ||||
| 	const meta = await misskeyApi('admin/meta'); | ||||
| 	enableRegistration.value = !meta.disableRegistration; | ||||
| 	emailRequiredForSignup.value = meta.emailRequiredForSignup; | ||||
| 	sensitiveWords.value = meta.sensitiveWords.join('\n'); | ||||
|   | ||||
| @@ -90,6 +90,7 @@ import MkInput from '@/components/MkInput.vue'; | ||||
| import FormSuspense from '@/components/form/suspense.vue'; | ||||
| import FormSplit from '@/components/form/split.vue'; | ||||
| import * as os from '@/os.js'; | ||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||
| import { fetchInstance } from '@/instance.js'; | ||||
| import { i18n } from '@/i18n.js'; | ||||
| import { definePageMetadata } from '@/scripts/page-metadata.js'; | ||||
| @@ -110,7 +111,7 @@ const objectStorageSetPublicRead = ref<boolean>(false); | ||||
| const objectStorageS3ForcePathStyle = ref<boolean>(true); | ||||
|  | ||||
| async function init() { | ||||
| 	const meta = await os.api('admin/meta'); | ||||
| 	const meta = await misskeyApi('admin/meta'); | ||||
| 	useObjectStorage.value = meta.useObjectStorage; | ||||
| 	objectStorageBaseUrl.value = meta.objectStorageBaseUrl; | ||||
| 	objectStorageBucket.value = meta.objectStorageBucket; | ||||
|   | ||||
| @@ -47,6 +47,7 @@ import { ref, computed } from 'vue'; | ||||
| import XHeader from './_header_.vue'; | ||||
| import FormSuspense from '@/components/form/suspense.vue'; | ||||
| import * as os from '@/os.js'; | ||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||
| import { fetchInstance } from '@/instance.js'; | ||||
| import { i18n } from '@/i18n.js'; | ||||
| import { definePageMetadata } from '@/scripts/page-metadata.js'; | ||||
| @@ -58,7 +59,7 @@ const enableChartsForRemoteUser = ref<boolean>(false); | ||||
| const enableChartsForFederatedInstances = ref<boolean>(false); | ||||
|  | ||||
| async function init() { | ||||
| 	const meta = await os.api('admin/meta'); | ||||
| 	const meta = await misskeyApi('admin/meta'); | ||||
| 	enableServerMachineStats.value = meta.enableServerMachineStats; | ||||
| 	enableIdenticonGeneration.value = meta.enableIdenticonGeneration; | ||||
| 	enableChartsForRemoteUser.value = meta.enableChartsForRemoteUser; | ||||
|   | ||||
| @@ -16,7 +16,7 @@ SPDX-License-Identifier: AGPL-3.0-only | ||||
| import { onMounted, shallowRef, ref } from 'vue'; | ||||
| import { Chart } from 'chart.js'; | ||||
| import gradient from 'chartjs-plugin-gradient'; | ||||
| import * as os from '@/os.js'; | ||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||
| import { defaultStore } from '@/store.js'; | ||||
| import { useChartTooltip } from '@/scripts/use-chart-tooltip.js'; | ||||
| import { chartVLine } from '@/scripts/chart-vline.js'; | ||||
| @@ -52,7 +52,7 @@ async function renderChart() { | ||||
| 		})); | ||||
| 	}; | ||||
|  | ||||
| 	const raw = await os.api('charts/active-users', { limit: chartLimit, span: 'day' }); | ||||
| 	const raw = await misskeyApi('charts/active-users', { limit: chartLimit, span: 'day' }); | ||||
|  | ||||
| 	const vLineColor = defaultStore.state.darkMode ? 'rgba(255, 255, 255, 0.2)' : 'rgba(0, 0, 0, 0.2)'; | ||||
|  | ||||
|   | ||||
| @@ -23,7 +23,7 @@ SPDX-License-Identifier: AGPL-3.0-only | ||||
| import { onMounted, shallowRef, ref } from 'vue'; | ||||
| import { Chart } from 'chart.js'; | ||||
| import gradient from 'chartjs-plugin-gradient'; | ||||
| import * as os from '@/os.js'; | ||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||
| import { useChartTooltip } from '@/scripts/use-chart-tooltip.js'; | ||||
| import { chartVLine } from '@/scripts/chart-vline.js'; | ||||
| import { defaultStore } from '@/store.js'; | ||||
| @@ -65,7 +65,7 @@ onMounted(async () => { | ||||
| 		})); | ||||
| 	}; | ||||
|  | ||||
| 	const raw = await os.api('charts/ap-request', { limit: chartLimit, span: 'day' }); | ||||
| 	const raw = await misskeyApi('charts/ap-request', { limit: chartLimit, span: 'day' }); | ||||
|  | ||||
| 	const vLineColor = defaultStore.state.darkMode ? 'rgba(255, 255, 255, 0.2)' : 'rgba(0, 0, 0, 0.2)'; | ||||
| 	const succColor = '#87e000'; | ||||
|   | ||||
| @@ -49,6 +49,7 @@ SPDX-License-Identifier: AGPL-3.0-only | ||||
| import { onMounted, ref } from 'vue'; | ||||
| import XPie, { type InstanceForPie } from './overview.pie.vue'; | ||||
| import * as os from '@/os.js'; | ||||
| import { misskeyApiGet } from '@/scripts/misskey-api.js'; | ||||
| import number from '@/filters/number.js'; | ||||
| import MkNumberDiff from '@/components/MkNumberDiff.vue'; | ||||
| import { i18n } from '@/i18n.js'; | ||||
| @@ -65,13 +66,13 @@ const fetching = ref(true); | ||||
| const { handler: externalTooltipHandler } = useChartTooltip(); | ||||
|  | ||||
| onMounted(async () => { | ||||
| 	const chart = await os.apiGet('charts/federation', { limit: 2, span: 'day' }); | ||||
| 	const chart = await misskeyApiGet('charts/federation', { limit: 2, span: 'day' }); | ||||
| 	federationPubActive.value = chart.pubActive[0]; | ||||
| 	federationPubActiveDiff.value = chart.pubActive[0] - chart.pubActive[1]; | ||||
| 	federationSubActive.value = chart.subActive[0]; | ||||
| 	federationSubActiveDiff.value = chart.subActive[0] - chart.subActive[1]; | ||||
|  | ||||
| 	os.apiGet('federation/stats', { limit: 10 }).then(res => { | ||||
| 	misskeyApiGet('federation/stats', { limit: 10 }).then(res => { | ||||
| 		topSubInstancesForPie.value = [ | ||||
| 			...res.topSubInstances.map(x => ({ | ||||
| 				name: x.host, | ||||
|   | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user
	 まっちゃとーにゅ
					まっちゃとーにゅ