mirror of
				https://codeberg.org/yeentown/barkey.git
				synced 2025-10-25 02:34:51 +00:00 
			
		
		
		
	# Conflicts: # .github/workflows/api-misskey-js.yml # .github/workflows/changelog-check.yml # .github/workflows/check-misskey-js-autogen.yml # .github/workflows/get-api-diff.yml # .github/workflows/lint.yml # .github/workflows/locale.yml # .github/workflows/on-release-created.yml # .github/workflows/storybook.yml # .github/workflows/test-backend.yml # .github/workflows/test-federation.yml # .github/workflows/test-frontend.yml # .github/workflows/test-misskey-js.yml # .github/workflows/test-production.yml # .github/workflows/validate-api-json.yml # package.json # packages/backend/package.json # packages/backend/src/server/api/ApiCallService.ts # packages/backend/src/server/api/endpoints/drive/files/create.ts # packages/frontend-shared/js/url.ts # packages/frontend/package.json # packages/frontend/src/components/MkFileCaptionEditWindow.vue # packages/frontend/src/components/MkInfo.vue # packages/frontend/src/components/MkLink.vue # packages/frontend/src/components/MkNote.vue # packages/frontend/src/components/MkNotes.vue # packages/frontend/src/components/MkPageWindow.vue # packages/frontend/src/components/MkReactionsViewer.vue # packages/frontend/src/components/MkTimeline.vue # packages/frontend/src/components/MkUrlPreview.vue # packages/frontend/src/components/MkUserPopup.vue # packages/frontend/src/components/global/MkPageHeader.vue # packages/frontend/src/components/global/MkUrl.vue # packages/frontend/src/components/global/PageWithHeader.vue # packages/frontend/src/pages/about-misskey.vue # packages/frontend/src/pages/announcements.vue # packages/frontend/src/pages/antenna-timeline.vue # packages/frontend/src/pages/channel.vue # packages/frontend/src/pages/instance-info.vue # packages/frontend/src/pages/note.vue # packages/frontend/src/pages/page.vue # packages/frontend/src/pages/role.vue # packages/frontend/src/pages/tag.vue # packages/frontend/src/pages/timeline.vue # packages/frontend/src/pages/user-list-timeline.vue # packages/frontend/src/pages/user/followers.vue # packages/frontend/src/pages/user/following.vue # packages/frontend/src/pages/user/home.vue # packages/frontend/src/pages/user/index.vue # packages/frontend/src/ui/deck.vue # packages/misskey-js/generator/package.json # pnpm-lock.yaml # scripts/changelog-checker/package-lock.json # scripts/changelog-checker/package.json
		
			
				
	
	
		
			274 lines
		
	
	
	
		
			6.5 KiB
		
	
	
	
		
			Vue
		
	
	
	
	
	
			
		
		
	
	
			274 lines
		
	
	
	
		
			6.5 KiB
		
	
	
	
		
			Vue
		
	
	
	
	
	
| <!--
 | |
| SPDX-FileCopyrightText: syuilo and misskey-project
 | |
| SPDX-License-Identifier: AGPL-3.0-only
 | |
| -->
 | |
| 
 | |
| <template>
 | |
| <PageWithHeader>
 | |
| 	<div class="_spacer" style="--MI_SPACER-w: 800px;">
 | |
| 		<div class="_gaps">
 | |
| 			<div class="_gaps_s">
 | |
| 				<div :class="$style.editor" class="_panel">
 | |
| 					<MkCodeEditor v-model="code" lang="aiscript" debounce/>
 | |
| 				</div>
 | |
| 				<MkButton primary @click="run()"><i class="ti ti-player-play"></i></MkButton>
 | |
| 			</div>
 | |
| 
 | |
| 			<MkContainer v-if="root && components.length > 1" :key="uiKey" :foldable="true">
 | |
| 				<template #header>UI</template>
 | |
| 				<div :class="$style.ui">
 | |
| 					<MkAsUi :component="root" :components="components" size="small"/>
 | |
| 				</div>
 | |
| 			</MkContainer>
 | |
| 
 | |
| 			<MkContainer :foldable="true" class="">
 | |
| 				<template #header>{{ i18n.ts.output }}</template>
 | |
| 				<div :class="$style.logs">
 | |
| 					<div v-for="log in logs" :key="log.id" class="log" :class="{ print: log.print }">{{ log.text }}</div>
 | |
| 				</div>
 | |
| 			</MkContainer>
 | |
| 
 | |
| 			<MkContainer :foldable="true" :expanded="false">
 | |
| 				<template #header>{{ i18n.ts.uiInspector }}</template>
 | |
| 				<div :class="$style.uiInspector">
 | |
| 					<div v-for="c in components" :key="c.value.id" :class="{ [$style.uiInspectorUnShown]: !showns.has(c.value.id) }">
 | |
| 						<div :class="$style.uiInspectorType">{{ c.value.type }}</div>
 | |
| 						<div :class="$style.uiInspectorId">{{ c.value.id }}</div>
 | |
| 						<button :class="$style.uiInspectorPropsToggle" @click="() => uiInspectorOpenedComponents.set(c, !uiInspectorOpenedComponents.get(c))">
 | |
| 							<i v-if="uiInspectorOpenedComponents.get(c)" class="ti ti-chevron-up icon"></i>
 | |
| 							<i v-else class="ti ti-chevron-down icon"></i>
 | |
| 						</button>
 | |
| 						<div v-if="uiInspectorOpenedComponents.get(c)">
 | |
| 							<MkTextarea :modelValue="stringifyUiProps(c.value)" code readonly></MkTextarea>
 | |
| 						</div>
 | |
| 					</div>
 | |
| 					<div :class="$style.uiInspectorDescription">{{ i18n.ts.uiInspectorDescription }}</div>
 | |
| 				</div>
 | |
| 			</MkContainer>
 | |
| 
 | |
| 			<div class="">
 | |
| 				{{ i18n.ts.scratchpadDescription }}
 | |
| 			</div>
 | |
| 		</div>
 | |
| 	</div>
 | |
| </PageWithHeader>
 | |
| </template>
 | |
| 
 | |
| <script lang="ts" setup>
 | |
| import { onDeactivated, onUnmounted, ref, watch, computed } from 'vue';
 | |
| import { Interpreter, Parser, utils } from '@syuilo/aiscript';
 | |
| import type { Ref } from 'vue';
 | |
| import type { AsUiComponent } from '@/aiscript/ui.js';
 | |
| import type { AsUiRoot } from '@/aiscript/ui.js';
 | |
| import MkContainer from '@/components/MkContainer.vue';
 | |
| import MkButton from '@/components/MkButton.vue';
 | |
| import MkTextarea from '@/components/MkTextarea.vue';
 | |
| import MkCodeEditor from '@/components/MkCodeEditor.vue';
 | |
| import { aiScriptReadline, createAiScriptEnv } from '@/aiscript/api.js';
 | |
| import * as os from '@/os.js';
 | |
| import { $i } from '@/i.js';
 | |
| import { i18n } from '@/i18n.js';
 | |
| import { definePage } from '@/page.js';
 | |
| import { registerAsUiLib } from '@/aiscript/ui.js';
 | |
| import MkAsUi from '@/components/MkAsUi.vue';
 | |
| import { miLocalStorage } from '@/local-storage.js';
 | |
| import { claimAchievement } from '@/utility/achievements.js';
 | |
| 
 | |
| const parser = new Parser();
 | |
| let aiscript: Interpreter;
 | |
| const code = ref('');
 | |
| const logs = ref<{
 | |
| 	id: number;
 | |
| 	text: string;
 | |
| 	print: boolean;
 | |
| }[]>([]);
 | |
| const root = ref<AsUiRoot>();
 | |
| const components = ref<Ref<AsUiComponent>[]>([]);
 | |
| const uiKey = ref(0);
 | |
| const uiInspectorOpenedComponents = ref(new Map<string, boolean>);
 | |
| 
 | |
| const saved = miLocalStorage.getItem('scratchpad');
 | |
| if (saved) {
 | |
| 	code.value = saved;
 | |
| }
 | |
| 
 | |
| watch(code, () => {
 | |
| 	miLocalStorage.setItem('scratchpad', code.value);
 | |
| });
 | |
| 
 | |
| function stringifyUiProps(uiProps) {
 | |
| 	return JSON.stringify(
 | |
| 		{ ...uiProps, type: undefined, id: undefined },
 | |
| 		(k, v) => typeof v === 'function' ? '<function>' : v,
 | |
| 		2,
 | |
| 	);
 | |
| }
 | |
| 
 | |
| async function run() {
 | |
| 	if (aiscript) aiscript.abort();
 | |
| 	root.value = undefined;
 | |
| 	components.value = [];
 | |
| 	uiKey.value++;
 | |
| 	logs.value = [];
 | |
| 	aiscript = new Interpreter(({
 | |
| 		...createAiScriptEnv({
 | |
| 			storageKey: 'widget',
 | |
| 			token: $i?.token,
 | |
| 		}),
 | |
| 		...registerAsUiLib(components.value, (_root) => {
 | |
| 			root.value = _root.value;
 | |
| 		}),
 | |
| 	}), {
 | |
| 		in: aiScriptReadline,
 | |
| 		out: (value) => {
 | |
| 			if (value.type === 'str' && value.value.toLowerCase().replace(',', '').includes('hello world')) {
 | |
| 				claimAchievement('outputHelloWorldOnScratchpad');
 | |
| 			}
 | |
| 			logs.value.push({
 | |
| 				id: Math.random(),
 | |
| 				text: value.type === 'str' ? value.value : utils.valToString(value),
 | |
| 				print: true,
 | |
| 			});
 | |
| 		},
 | |
| 		err: (err) => {
 | |
| 			os.alert({
 | |
| 				type: 'error',
 | |
| 				title: 'AiScript Error',
 | |
| 				text: err.toString(),
 | |
| 			});
 | |
| 		},
 | |
| 		log: (type, params) => {
 | |
| 			switch (type) {
 | |
| 				case 'end': logs.value.push({
 | |
| 					id: Math.random(),
 | |
| 					text: utils.valToString(params.val, true),
 | |
| 					print: false,
 | |
| 				}); break;
 | |
| 				default: break;
 | |
| 			}
 | |
| 		},
 | |
| 	});
 | |
| 
 | |
| 	let ast;
 | |
| 	try {
 | |
| 		ast = parser.parse(code.value);
 | |
| 	} catch (err: any) {
 | |
| 		os.alert({
 | |
| 			type: 'error',
 | |
| 			title: 'Syntax Error',
 | |
| 			text: err.toString(),
 | |
| 		});
 | |
| 		return;
 | |
| 	}
 | |
| 	try {
 | |
| 		await aiscript.exec(ast);
 | |
| 	} catch (err: any) {
 | |
| 		// AiScript runtime errors should be processed by error callback function
 | |
| 		// so errors caught here are AiScript's internal errors.
 | |
| 		os.alert({
 | |
| 			type: 'error',
 | |
| 			title: 'Internal Error',
 | |
| 			text: err.toString(),
 | |
| 		});
 | |
| 	}
 | |
| }
 | |
| 
 | |
| onDeactivated(() => {
 | |
| 	if (aiscript) aiscript.abort();
 | |
| });
 | |
| 
 | |
| onUnmounted(() => {
 | |
| 	if (aiscript) aiscript.abort();
 | |
| });
 | |
| 
 | |
| const headerActions = computed(() => []);
 | |
| 
 | |
| const headerTabs = computed(() => []);
 | |
| 
 | |
| const showns = computed(() => {
 | |
| 	const result = new Set<string>();
 | |
| 	(function addChildrenToResult(c: AsUiComponent) {
 | |
| 		result.add(c.id);
 | |
| 		if (c.children) {
 | |
| 			const childComponents = components.value.filter(v => c.children.includes(v.value.id));
 | |
| 			for (const child of childComponents) {
 | |
| 				addChildrenToResult(child.value);
 | |
| 			}
 | |
| 		}
 | |
| 	})(root.value);
 | |
| 	return result;
 | |
| });
 | |
| 
 | |
| definePage(() => ({
 | |
| 	title: i18n.ts.scratchpad,
 | |
| 	icon: 'ti ti-terminal-2',
 | |
| }));
 | |
| </script>
 | |
| 
 | |
| <style lang="scss" module>
 | |
| .root {
 | |
| }
 | |
| 
 | |
| .editor {
 | |
| 	position: relative;
 | |
| }
 | |
| 
 | |
| .code {
 | |
| 	background: #2d2d2d;
 | |
| 	color: #ccc;
 | |
| 	font-size: 14px;
 | |
| 	line-height: 1.5;
 | |
| 	padding: 5px;
 | |
| }
 | |
| 
 | |
| .ui {
 | |
| 	padding: 32px;
 | |
| }
 | |
| 
 | |
| .logs {
 | |
| 	padding: 16px;
 | |
| 
 | |
| 	&:global {
 | |
| 		> .log {
 | |
| 			&:not(.print) {
 | |
| 				opacity: 0.7;
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| .uiInspector {
 | |
| 	display: grid;
 | |
| 	gap: 8px;
 | |
| 	padding: 16px;
 | |
| }
 | |
| 
 | |
| .uiInspectorUnShown {
 | |
| 	color: color(from var(--MI_THEME-fg) srgb r g b / 0.5);
 | |
| }
 | |
| 
 | |
| .uiInspectorType {
 | |
| 	display: inline-block;
 | |
| 	border: hidden;
 | |
| 	border-radius: 10px;
 | |
| 	background-color: var(--MI_THEME-panelHighlight);
 | |
| 	padding: 2px 8px;
 | |
| 	font-size: 12px;
 | |
| }
 | |
| 
 | |
| .uiInspectorId {
 | |
| 	display: inline-block;
 | |
| 	padding-left: 8px;
 | |
| }
 | |
| 
 | |
| .uiInspectorDescription {
 | |
| 	display: block;
 | |
| 	font-size: 12px;
 | |
| 	padding-top: 16px;
 | |
| }
 | |
| 
 | |
| .uiInspectorPropsToggle {
 | |
| 	background: none;
 | |
| 	border: none;
 | |
| }
 | |
| </style>
 |