mirror of
				https://codeberg.org/yeentown/barkey.git
				synced 2025-11-04 15:34:13 +00:00 
			
		
		
		
	enhance(client): tweak ui
This commit is contained in:
		
							parent
							
								
									3b69a563f8
								
							
						
					
					
						commit
						d7222dd56a
					
				
					 10 changed files with 462 additions and 275 deletions
				
			
		| 
						 | 
				
			
			@ -15,20 +15,6 @@
 | 
			
		|||
				</MkA>
 | 
			
		||||
			</template>
 | 
			
		||||
		</div>
 | 
			
		||||
		<div class="sub">
 | 
			
		||||
			<button v-click-anime class="_button" @click="help">
 | 
			
		||||
				<i class="fas fa-question-circle icon"></i>
 | 
			
		||||
				<div class="text">{{ $ts.help }}</div>
 | 
			
		||||
			</button>
 | 
			
		||||
			<MkA v-click-anime to="/about" @click.passive="close()">
 | 
			
		||||
				<i class="fas fa-info-circle icon"></i>
 | 
			
		||||
				<div class="text">{{ $ts.instanceInfo }}</div>
 | 
			
		||||
			</MkA>
 | 
			
		||||
			<MkA v-click-anime to="/about-misskey" @click.passive="close()">
 | 
			
		||||
				<img src="/static-assets/favicon.png" class="icon"/>
 | 
			
		||||
				<div class="text">{{ $ts.aboutMisskey }}</div>
 | 
			
		||||
			</MkA>
 | 
			
		||||
		</div>
 | 
			
		||||
	</div>
 | 
			
		||||
</MkModal>
 | 
			
		||||
</template>
 | 
			
		||||
| 
						 | 
				
			
			@ -74,28 +60,6 @@ const items = Object.keys(navbarItemDef).filter(k => !menu.includes(k)).map(k =>
 | 
			
		|||
function close() {
 | 
			
		||||
	modal.close();
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function help(ev: MouseEvent) {
 | 
			
		||||
	os.popupMenu([{
 | 
			
		||||
		type: 'link',
 | 
			
		||||
		to: '/mfm-cheat-sheet',
 | 
			
		||||
		text: i18n.ts._mfm.cheatSheet,
 | 
			
		||||
		icon: 'fas fa-code',
 | 
			
		||||
	}, {
 | 
			
		||||
		type: 'link',
 | 
			
		||||
		to: '/scratchpad',
 | 
			
		||||
		text: i18n.ts.scratchpad,
 | 
			
		||||
		icon: 'fas fa-terminal',
 | 
			
		||||
	}, null, {
 | 
			
		||||
		text: i18n.ts.document,
 | 
			
		||||
		icon: 'fas fa-question-circle',
 | 
			
		||||
		action: () => {
 | 
			
		||||
			window.open('https://misskey-hub.net/help.html', '_blank');
 | 
			
		||||
		},
 | 
			
		||||
	}], ev.currentTarget ?? ev.target);
 | 
			
		||||
 | 
			
		||||
	close();
 | 
			
		||||
}
 | 
			
		||||
</script>
 | 
			
		||||
 | 
			
		||||
<style lang="scss" scoped>
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
							
								
								
									
										63
									
								
								packages/client/src/components/ui/child-menu.vue
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										63
									
								
								packages/client/src/components/ui/child-menu.vue
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,63 @@
 | 
			
		|||
<template>
 | 
			
		||||
<div ref="el" class="sfhdhdhr">
 | 
			
		||||
	<MkMenu ref="menu" :items="items" :align="align" :width="width" :as-drawer="false" @close="onChildClosed"/>
 | 
			
		||||
</div>
 | 
			
		||||
</template>
 | 
			
		||||
 | 
			
		||||
<script lang="ts" setup>
 | 
			
		||||
import { on } from 'events';
 | 
			
		||||
import { nextTick, onBeforeUnmount, onMounted, onUnmounted, ref, watch } from 'vue';
 | 
			
		||||
import MkMenu from './menu.vue';
 | 
			
		||||
import { MenuItem } from '@/types/menu';
 | 
			
		||||
import * as os from '@/os';
 | 
			
		||||
 | 
			
		||||
const props = defineProps<{
 | 
			
		||||
	items: MenuItem[];
 | 
			
		||||
	targetElement: HTMLElement;
 | 
			
		||||
	width?: number;
 | 
			
		||||
	viaKeyboard?: boolean;
 | 
			
		||||
}>();
 | 
			
		||||
 | 
			
		||||
const emit = defineEmits<{
 | 
			
		||||
	(ev: 'closed'): void;
 | 
			
		||||
	(ev: 'actioned'): void;
 | 
			
		||||
}>();
 | 
			
		||||
 | 
			
		||||
const el = ref<HTMLElement>();
 | 
			
		||||
const align = 'left';
 | 
			
		||||
 | 
			
		||||
function setPosition() {
 | 
			
		||||
	const rect = props.targetElement.getBoundingClientRect();
 | 
			
		||||
	const left = rect.left + props.targetElement.offsetWidth;
 | 
			
		||||
	const top = rect.top - 8;
 | 
			
		||||
	el.value.style.left = left + 'px';
 | 
			
		||||
	el.value.style.top = top + 'px';
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function onChildClosed(actioned?: boolean) {
 | 
			
		||||
	if (actioned) {
 | 
			
		||||
		emit('actioned');
 | 
			
		||||
	} else {
 | 
			
		||||
		emit('closed');
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
onMounted(() => {
 | 
			
		||||
	setPosition();
 | 
			
		||||
	nextTick(() => {
 | 
			
		||||
		setPosition();
 | 
			
		||||
	});
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
defineExpose({
 | 
			
		||||
	checkHit: (ev: MouseEvent) => {
 | 
			
		||||
		return (ev.target === el.value || el.value.contains(ev.target));
 | 
			
		||||
	},
 | 
			
		||||
});
 | 
			
		||||
</script>
 | 
			
		||||
 | 
			
		||||
<style lang="scss" scoped>
 | 
			
		||||
.sfhdhdhr {
 | 
			
		||||
	position: fixed;
 | 
			
		||||
}
 | 
			
		||||
</style>
 | 
			
		||||
| 
						 | 
				
			
			@ -1,55 +1,67 @@
 | 
			
		|||
<template>
 | 
			
		||||
<div
 | 
			
		||||
	ref="itemsEl" v-hotkey="keymap"
 | 
			
		||||
	class="rrevdjwt"
 | 
			
		||||
	:class="{ center: align === 'center', asDrawer }"
 | 
			
		||||
	:style="{ width: (width && !asDrawer) ? width + 'px' : '', maxHeight: maxHeight ? maxHeight + 'px' : '' }"
 | 
			
		||||
	@contextmenu.self="e => e.preventDefault()"
 | 
			
		||||
>
 | 
			
		||||
	<template v-for="(item, i) in items2">
 | 
			
		||||
		<div v-if="item === null" class="divider"></div>
 | 
			
		||||
		<span v-else-if="item.type === 'label'" class="label item">
 | 
			
		||||
			<span>{{ item.text }}</span>
 | 
			
		||||
<div>
 | 
			
		||||
	<div
 | 
			
		||||
		ref="itemsEl" v-hotkey="keymap"
 | 
			
		||||
		class="rrevdjwt _popup _shadow"
 | 
			
		||||
		:class="{ center: align === 'center', asDrawer }"
 | 
			
		||||
		:style="{ width: (width && !asDrawer) ? width + 'px' : '', maxHeight: maxHeight ? maxHeight + 'px' : '' }"
 | 
			
		||||
		@contextmenu.self="e => e.preventDefault()"
 | 
			
		||||
	>
 | 
			
		||||
		<template v-for="(item, i) in items2">
 | 
			
		||||
			<div v-if="item === null" class="divider"></div>
 | 
			
		||||
			<span v-else-if="item.type === 'label'" class="label item">
 | 
			
		||||
				<span>{{ item.text }}</span>
 | 
			
		||||
			</span>
 | 
			
		||||
			<span v-else-if="item.type === 'pending'" :tabindex="i" class="pending item">
 | 
			
		||||
				<span><MkEllipsis/></span>
 | 
			
		||||
			</span>
 | 
			
		||||
			<MkA v-else-if="item.type === 'link'" :to="item.to" :tabindex="i" class="_button item" @click.passive="close(true)" @mouseenter.passive="onItemMouseEnter(item)" @mouseleave.passive="onItemMouseLeave(item)">
 | 
			
		||||
				<i v-if="item.icon" class="fa-fw" :class="item.icon"></i>
 | 
			
		||||
				<MkAvatar v-if="item.avatar" :user="item.avatar" class="avatar"/>
 | 
			
		||||
				<span>{{ item.text }}</span>
 | 
			
		||||
				<span v-if="item.indicate" class="indicator"><i class="fas fa-circle"></i></span>
 | 
			
		||||
			</MkA>
 | 
			
		||||
			<a v-else-if="item.type === 'a'" :href="item.href" :target="item.target" :download="item.download" :tabindex="i" class="_button item" @click="close(true)" @mouseenter.passive="onItemMouseEnter(item)" @mouseleave.passive="onItemMouseLeave(item)">
 | 
			
		||||
				<i v-if="item.icon" class="fa-fw" :class="item.icon"></i>
 | 
			
		||||
				<span>{{ item.text }}</span>
 | 
			
		||||
				<span v-if="item.indicate" class="indicator"><i class="fas fa-circle"></i></span>
 | 
			
		||||
			</a>
 | 
			
		||||
			<button v-else-if="item.type === 'user'" :tabindex="i" class="_button item" :class="{ active: item.active }" :disabled="item.active" @click="clicked(item.action, $event)" @mouseenter.passive="onItemMouseEnter(item)" @mouseleave.passive="onItemMouseLeave(item)">
 | 
			
		||||
				<MkAvatar :user="item.user" class="avatar"/><MkUserName :user="item.user"/>
 | 
			
		||||
				<span v-if="item.indicate" class="indicator"><i class="fas fa-circle"></i></span>
 | 
			
		||||
			</button>
 | 
			
		||||
			<span v-else-if="item.type === 'switch'" :tabindex="i" class="item" @mouseenter.passive="onItemMouseEnter(item)" @mouseleave.passive="onItemMouseLeave(item)">
 | 
			
		||||
				<FormSwitch v-model="item.ref" :disabled="item.disabled" class="form-switch">{{ item.text }}</FormSwitch>
 | 
			
		||||
			</span>
 | 
			
		||||
			<button v-else-if="item.type === 'parent'" :tabindex="i" class="_button item parent" :class="{ childShowing: childShowingItem === item }" @mouseenter="showChildren(item, $event)">
 | 
			
		||||
				<i v-if="item.icon" class="fa-fw" :class="item.icon"></i>
 | 
			
		||||
				<span>{{ item.text }}</span>
 | 
			
		||||
				<span class="caret"><i class="fas fa-caret-right fa-fw"></i></span>
 | 
			
		||||
			</button>
 | 
			
		||||
			<button v-else :tabindex="i" class="_button item" :class="{ danger: item.danger, active: item.active }" :disabled="item.active" @click="clicked(item.action, $event)" @mouseenter.passive="onItemMouseEnter(item)" @mouseleave.passive="onItemMouseLeave(item)">
 | 
			
		||||
				<i v-if="item.icon" class="fa-fw" :class="item.icon"></i>
 | 
			
		||||
				<MkAvatar v-if="item.avatar" :user="item.avatar" class="avatar"/>
 | 
			
		||||
				<span>{{ item.text }}</span>
 | 
			
		||||
				<span v-if="item.indicate" class="indicator"><i class="fas fa-circle"></i></span>
 | 
			
		||||
			</button>
 | 
			
		||||
		</template>
 | 
			
		||||
		<span v-if="items2.length === 0" class="none item">
 | 
			
		||||
			<span>{{ $ts.none }}</span>
 | 
			
		||||
		</span>
 | 
			
		||||
		<span v-else-if="item.type === 'pending'" :tabindex="i" class="pending item">
 | 
			
		||||
			<span><MkEllipsis/></span>
 | 
			
		||||
		</span>
 | 
			
		||||
		<MkA v-else-if="item.type === 'link'" :to="item.to" :tabindex="i" class="_button item" @click.passive="close()">
 | 
			
		||||
			<i v-if="item.icon" class="fa-fw" :class="item.icon"></i>
 | 
			
		||||
			<MkAvatar v-if="item.avatar" :user="item.avatar" class="avatar"/>
 | 
			
		||||
			<span>{{ item.text }}</span>
 | 
			
		||||
			<span v-if="item.indicate" class="indicator"><i class="fas fa-circle"></i></span>
 | 
			
		||||
		</MkA>
 | 
			
		||||
		<a v-else-if="item.type === 'a'" :href="item.href" :target="item.target" :download="item.download" :tabindex="i" class="_button item" @click="close()">
 | 
			
		||||
			<i v-if="item.icon" class="fa-fw" :class="item.icon"></i>
 | 
			
		||||
			<span>{{ item.text }}</span>
 | 
			
		||||
			<span v-if="item.indicate" class="indicator"><i class="fas fa-circle"></i></span>
 | 
			
		||||
		</a>
 | 
			
		||||
		<button v-else-if="item.type === 'user'" :tabindex="i" class="_button item" :class="{ active: item.active }" :disabled="item.active" @click="clicked(item.action, $event)">
 | 
			
		||||
			<MkAvatar :user="item.user" class="avatar"/><MkUserName :user="item.user"/>
 | 
			
		||||
			<span v-if="item.indicate" class="indicator"><i class="fas fa-circle"></i></span>
 | 
			
		||||
		</button>
 | 
			
		||||
		<span v-else-if="item.type === 'switch'" :tabindex="i" class="item">
 | 
			
		||||
			<FormSwitch v-model="item.ref" :disabled="item.disabled" class="form-switch">{{ item.text }}</FormSwitch>
 | 
			
		||||
		</span>
 | 
			
		||||
		<button v-else :tabindex="i" class="_button item" :class="{ danger: item.danger, active: item.active }" :disabled="item.active" @click="clicked(item.action, $event)">
 | 
			
		||||
			<i v-if="item.icon" class="fa-fw" :class="item.icon"></i>
 | 
			
		||||
			<MkAvatar v-if="item.avatar" :user="item.avatar" class="avatar"/>
 | 
			
		||||
			<span>{{ item.text }}</span>
 | 
			
		||||
			<span v-if="item.indicate" class="indicator"><i class="fas fa-circle"></i></span>
 | 
			
		||||
		</button>
 | 
			
		||||
	</template>
 | 
			
		||||
	<span v-if="items2.length === 0" class="none item">
 | 
			
		||||
		<span>{{ $ts.none }}</span>
 | 
			
		||||
	</span>
 | 
			
		||||
	</div>
 | 
			
		||||
	<div v-if="childMenu" class="child">
 | 
			
		||||
		<XChild ref="child" :items="childMenu" :target-element="childTarget" showing @actioned="childActioned"/>
 | 
			
		||||
	</div>
 | 
			
		||||
</div>
 | 
			
		||||
</template>
 | 
			
		||||
 | 
			
		||||
<script lang="ts" setup>
 | 
			
		||||
import { nextTick, onMounted, watch } from 'vue';
 | 
			
		||||
import { defineAsyncComponent, nextTick, onBeforeUnmount, onMounted, onUnmounted, Ref, ref, watch } from 'vue';
 | 
			
		||||
import { focusPrev, focusNext } from '@/scripts/focus';
 | 
			
		||||
import FormSwitch from '@/components/form/switch.vue';
 | 
			
		||||
import { MenuItem, InnerMenuItem, MenuPending, MenuAction } from '@/types/menu';
 | 
			
		||||
import * as os from '@/os';
 | 
			
		||||
const XChild = defineAsyncComponent(() => import('./child-menu.vue'));
 | 
			
		||||
 | 
			
		||||
const props = defineProps<{
 | 
			
		||||
	items: MenuItem[];
 | 
			
		||||
| 
						 | 
				
			
			@ -61,19 +73,23 @@ const props = defineProps<{
 | 
			
		|||
}>();
 | 
			
		||||
 | 
			
		||||
const emit = defineEmits<{
 | 
			
		||||
	(ev: 'close'): void;
 | 
			
		||||
	(ev: 'close', actioned?: boolean): void;
 | 
			
		||||
}>();
 | 
			
		||||
 | 
			
		||||
let itemsEl = $ref<HTMLDivElement>();
 | 
			
		||||
 | 
			
		||||
let items2: InnerMenuItem[] = $ref([]);
 | 
			
		||||
 | 
			
		||||
let child = $ref<InstanceType<typeof XChild>>();
 | 
			
		||||
 | 
			
		||||
let keymap = $computed(() => ({
 | 
			
		||||
	'up|k|shift+tab': focusUp,
 | 
			
		||||
	'down|j|tab': focusDown,
 | 
			
		||||
	'esc': close,
 | 
			
		||||
}));
 | 
			
		||||
 | 
			
		||||
let childShowingItem = $ref<MenuItem | null>();
 | 
			
		||||
 | 
			
		||||
watch(() => props.items, () => {
 | 
			
		||||
	const items: (MenuItem | MenuPending)[] = [...props.items].filter(item => item !== undefined);
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -93,21 +109,53 @@ watch(() => props.items, () => {
 | 
			
		|||
	immediate: true,
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
onMounted(() => {
 | 
			
		||||
	if (props.viaKeyboard) {
 | 
			
		||||
		nextTick(() => {
 | 
			
		||||
			focusNext(itemsEl.children[0], true, false);
 | 
			
		||||
		});
 | 
			
		||||
let childMenu = $ref<MenuItem[] | null>();
 | 
			
		||||
let childTarget = $ref<HTMLElement | null>();
 | 
			
		||||
 | 
			
		||||
function closeChild() {
 | 
			
		||||
	childMenu = null;
 | 
			
		||||
	childShowingItem = null;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function childActioned() {
 | 
			
		||||
	closeChild();
 | 
			
		||||
	close(true);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function onGlobalMousedown(event: MouseEvent) {
 | 
			
		||||
	if (childTarget && (event.target === childTarget || childTarget.contains(event.target))) return;
 | 
			
		||||
	if (child && child.checkHit(event)) return;
 | 
			
		||||
	closeChild();
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
let childCloseTimer: null | number = null;
 | 
			
		||||
function onItemMouseEnter(item) {
 | 
			
		||||
	childCloseTimer = window.setTimeout(() => {
 | 
			
		||||
		closeChild();
 | 
			
		||||
	}, 300);
 | 
			
		||||
}
 | 
			
		||||
function onItemMouseLeave(item) {
 | 
			
		||||
	if (childCloseTimer) window.clearTimeout(childCloseTimer);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
async function showChildren(item: MenuItem, ev: MouseEvent) {
 | 
			
		||||
	if (props.asDrawer) {
 | 
			
		||||
		os.popupMenu(item.children, ev.currentTarget ?? ev.target);
 | 
			
		||||
		close();
 | 
			
		||||
	} else {
 | 
			
		||||
		childTarget = ev.currentTarget ?? ev.target;
 | 
			
		||||
		childMenu = item.children;
 | 
			
		||||
		childShowingItem = item;
 | 
			
		||||
	}
 | 
			
		||||
});
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function clicked(fn: MenuAction, ev: MouseEvent) {
 | 
			
		||||
	fn(ev);
 | 
			
		||||
	close();
 | 
			
		||||
	close(true);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function close() {
 | 
			
		||||
	emit('close');
 | 
			
		||||
function close(actioned = false) {
 | 
			
		||||
	emit('close', actioned);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function focusUp() {
 | 
			
		||||
| 
						 | 
				
			
			@ -117,6 +165,20 @@ function focusUp() {
 | 
			
		|||
function focusDown() {
 | 
			
		||||
	focusNext(document.activeElement);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
onMounted(() => {
 | 
			
		||||
	if (props.viaKeyboard) {
 | 
			
		||||
		nextTick(() => {
 | 
			
		||||
			focusNext(itemsEl.children[0], true, false);
 | 
			
		||||
		});
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	document.addEventListener('mousedown', onGlobalMousedown, { passive: true });
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
onBeforeUnmount(() => {
 | 
			
		||||
	document.removeEventListener('mousedown', onGlobalMousedown);
 | 
			
		||||
});
 | 
			
		||||
</script>
 | 
			
		||||
 | 
			
		||||
<style lang="scss" scoped>
 | 
			
		||||
| 
						 | 
				
			
			@ -225,6 +287,25 @@ function focusDown() {
 | 
			
		|||
			opacity: 0.7;
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		&.parent {
 | 
			
		||||
			display: flex;
 | 
			
		||||
			align-items: center;
 | 
			
		||||
			cursor: default;
 | 
			
		||||
 | 
			
		||||
			> .caret {
 | 
			
		||||
				margin-left: auto;
 | 
			
		||||
			}
 | 
			
		||||
 | 
			
		||||
			&.childShowing {
 | 
			
		||||
				color: var(--accent);
 | 
			
		||||
				text-decoration: none;
 | 
			
		||||
 | 
			
		||||
				&:before {
 | 
			
		||||
					background: var(--accentedBg);
 | 
			
		||||
				}
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		> i {
 | 
			
		||||
			margin-right: 5px;
 | 
			
		||||
			width: 20px;
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -1,6 +1,6 @@
 | 
			
		|||
<template>
 | 
			
		||||
<MkModal ref="modal" v-slot="{ type, maxHeight }" :z-priority="'high'" :src="src" :transparent-bg="true" @click="modal.close()" @closed="emit('closed')">
 | 
			
		||||
	<MkMenu :items="items" :align="align" :width="width" :max-height="maxHeight" :as-drawer="type === 'drawer'" class="sfhdhdhq _popup _shadow" :class="{ drawer: type === 'drawer' }" @close="modal.close()"/>
 | 
			
		||||
	<MkMenu :items="items" :align="align" :width="width" :max-height="maxHeight" :as-drawer="type === 'drawer'" class="sfhdhdhq" :class="{ drawer: type === 'drawer' }" @close="modal.close()"/>
 | 
			
		||||
</MkModal>
 | 
			
		||||
</template>
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -12,6 +12,7 @@
 | 
			
		|||
<script lang="ts" setup>
 | 
			
		||||
import { nextTick, onMounted, onUnmounted, ref } from 'vue';
 | 
			
		||||
import * as os from '@/os';
 | 
			
		||||
import { calcPopupPosition } from '@/scripts/popup-position';
 | 
			
		||||
 | 
			
		||||
const props = withDefaults(defineProps<{
 | 
			
		||||
	showing: boolean;
 | 
			
		||||
| 
						 | 
				
			
			@ -36,151 +37,20 @@ const emit = defineEmits<{
 | 
			
		|||
const el = ref<HTMLElement>();
 | 
			
		||||
const zIndex = os.claimZIndex('high');
 | 
			
		||||
 | 
			
		||||
const setPosition = () => {
 | 
			
		||||
	if (el.value == null) return;
 | 
			
		||||
function setPosition() {
 | 
			
		||||
	const data = calcPopupPosition(el.value, {
 | 
			
		||||
		anchorElement: props.targetElement,
 | 
			
		||||
		direction: props.direction,
 | 
			
		||||
		align: 'center',
 | 
			
		||||
		innerMargin: props.innerMargin,
 | 
			
		||||
		x: props.x,
 | 
			
		||||
		y: props.y,
 | 
			
		||||
	});
 | 
			
		||||
 | 
			
		||||
	const contentWidth = el.value.offsetWidth;
 | 
			
		||||
	const contentHeight = el.value.offsetHeight;
 | 
			
		||||
 | 
			
		||||
	let rect: DOMRect;
 | 
			
		||||
 | 
			
		||||
	if (props.targetElement) {
 | 
			
		||||
		rect = props.targetElement.getBoundingClientRect();
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	const calcPosWhenTop = () => {
 | 
			
		||||
		let left: number;
 | 
			
		||||
		let top: number;
 | 
			
		||||
 | 
			
		||||
		if (props.targetElement) {
 | 
			
		||||
			left = rect.left + window.pageXOffset + (props.targetElement.offsetWidth / 2);
 | 
			
		||||
			top = (rect.top + window.pageYOffset - contentHeight) - props.innerMargin;
 | 
			
		||||
		} else {
 | 
			
		||||
			left = props.x;
 | 
			
		||||
			top = (props.y - contentHeight) - props.innerMargin;
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		left -= (el.value.offsetWidth / 2);
 | 
			
		||||
 | 
			
		||||
		if (left + contentWidth - window.pageXOffset > window.innerWidth) {
 | 
			
		||||
			left = window.innerWidth - contentWidth + window.pageXOffset - 1;
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		return [left, top];
 | 
			
		||||
	};
 | 
			
		||||
 | 
			
		||||
	const calcPosWhenBottom = () => {
 | 
			
		||||
		let left: number;
 | 
			
		||||
		let top: number;
 | 
			
		||||
 | 
			
		||||
		if (props.targetElement) {
 | 
			
		||||
			left = rect.left + window.pageXOffset + (props.targetElement.offsetWidth / 2);
 | 
			
		||||
			top = (rect.top + window.pageYOffset + props.targetElement.offsetHeight) + props.innerMargin;
 | 
			
		||||
		} else {
 | 
			
		||||
			left = props.x;
 | 
			
		||||
			top = (props.y) + props.innerMargin;
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		left -= (el.value.offsetWidth / 2);
 | 
			
		||||
 | 
			
		||||
		if (left + contentWidth - window.pageXOffset > window.innerWidth) {
 | 
			
		||||
			left = window.innerWidth - contentWidth + window.pageXOffset - 1;
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		return [left, top];
 | 
			
		||||
	};
 | 
			
		||||
 | 
			
		||||
	const calcPosWhenLeft = () => {
 | 
			
		||||
		let left: number;
 | 
			
		||||
		let top: number;
 | 
			
		||||
 | 
			
		||||
		if (props.targetElement) {
 | 
			
		||||
			left = (rect.left + window.pageXOffset - contentWidth) - props.innerMargin;
 | 
			
		||||
			top = rect.top + window.pageYOffset + (props.targetElement.offsetHeight / 2);
 | 
			
		||||
		} else {
 | 
			
		||||
			left = (props.x - contentWidth) - props.innerMargin;
 | 
			
		||||
			top = props.y;
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		top -= (el.value.offsetHeight / 2);
 | 
			
		||||
 | 
			
		||||
		if (top + contentHeight - window.pageYOffset > window.innerHeight) {
 | 
			
		||||
			top = window.innerHeight - contentHeight + window.pageYOffset - 1;
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		return [left, top];
 | 
			
		||||
	};
 | 
			
		||||
 | 
			
		||||
	const calcPosWhenRight = () => {
 | 
			
		||||
		let left: number;
 | 
			
		||||
		let top: number;
 | 
			
		||||
 | 
			
		||||
		if (props.targetElement) {
 | 
			
		||||
			left = (rect.left + props.targetElement.offsetWidth + window.pageXOffset) + props.innerMargin;
 | 
			
		||||
			top = rect.top + window.pageYOffset + (props.targetElement.offsetHeight / 2);
 | 
			
		||||
		} else {
 | 
			
		||||
			left = props.x + props.innerMargin;
 | 
			
		||||
			top = props.y;
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		top -= (el.value.offsetHeight / 2);
 | 
			
		||||
 | 
			
		||||
		if (top + contentHeight - window.pageYOffset > window.innerHeight) {
 | 
			
		||||
			top = window.innerHeight - contentHeight + window.pageYOffset - 1;
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		return [left, top];
 | 
			
		||||
	};
 | 
			
		||||
 | 
			
		||||
	const calc = (): {
 | 
			
		||||
		left: number;
 | 
			
		||||
		top: number;
 | 
			
		||||
		transformOrigin: string;
 | 
			
		||||
	} => {
 | 
			
		||||
		switch (props.direction) {
 | 
			
		||||
			case 'top': {
 | 
			
		||||
				const [left, top] = calcPosWhenTop();
 | 
			
		||||
 | 
			
		||||
				// ツールチップを上に向かって表示するスペースがなければ下に向かって出す
 | 
			
		||||
				if (top - window.pageYOffset < 0) {
 | 
			
		||||
					const [left, top] = calcPosWhenBottom();
 | 
			
		||||
					return { left, top, transformOrigin: 'center top' };
 | 
			
		||||
				}
 | 
			
		||||
 | 
			
		||||
				return { left, top, transformOrigin: 'center bottom' };
 | 
			
		||||
			}
 | 
			
		||||
 | 
			
		||||
			case 'bottom': {
 | 
			
		||||
				const [left, top] = calcPosWhenBottom();
 | 
			
		||||
				// TODO: ツールチップを下に向かって表示するスペースがなければ上に向かって出す
 | 
			
		||||
				return { left, top, transformOrigin: 'center top' };
 | 
			
		||||
			}
 | 
			
		||||
 | 
			
		||||
			case 'left': {
 | 
			
		||||
				const [left, top] = calcPosWhenLeft();
 | 
			
		||||
 | 
			
		||||
				// ツールチップを左に向かって表示するスペースがなければ右に向かって出す
 | 
			
		||||
				if (left - window.pageXOffset < 0) {
 | 
			
		||||
					const [left, top] = calcPosWhenRight();
 | 
			
		||||
					return { left, top, transformOrigin: 'left center' };
 | 
			
		||||
				}
 | 
			
		||||
 | 
			
		||||
				return { left, top, transformOrigin: 'right center' };
 | 
			
		||||
			}
 | 
			
		||||
 | 
			
		||||
			case 'right': {
 | 
			
		||||
				const [left, top] = calcPosWhenRight();
 | 
			
		||||
				// TODO: ツールチップを右に向かって表示するスペースがなければ左に向かって出す
 | 
			
		||||
				return { left, top, transformOrigin: 'left center' };
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
	};
 | 
			
		||||
 | 
			
		||||
	const { left, top, transformOrigin } = calc();
 | 
			
		||||
	el.value.style.transformOrigin = transformOrigin;
 | 
			
		||||
	el.value.style.left = left + 'px';
 | 
			
		||||
	el.value.style.top = top + 'px';
 | 
			
		||||
};
 | 
			
		||||
	el.value.style.transformOrigin = data.transformOrigin;
 | 
			
		||||
	el.value.style.left = data.left + 'px';
 | 
			
		||||
	el.value.style.top = data.top + 'px';
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
let loopHandler;
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
							
								
								
									
										158
									
								
								packages/client/src/scripts/popup-position.ts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										158
									
								
								packages/client/src/scripts/popup-position.ts
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,158 @@
 | 
			
		|||
import { Ref } from 'vue';
 | 
			
		||||
 | 
			
		||||
export function calcPopupPosition(el: HTMLElement, props: {
 | 
			
		||||
	anchorElement: HTMLElement | null;
 | 
			
		||||
	innerMargin: number;
 | 
			
		||||
	direction: 'top' | 'bottom' | 'left' | 'right';
 | 
			
		||||
	align: 'top' | 'bottom' | 'left' | 'right' | 'center';
 | 
			
		||||
	alignOffset?: number;
 | 
			
		||||
	x?: number;
 | 
			
		||||
	y?: number;
 | 
			
		||||
}): { top: number; left: number; transformOrigin: string; } {
 | 
			
		||||
	const contentWidth = el.offsetWidth;
 | 
			
		||||
	const contentHeight = el.offsetHeight;
 | 
			
		||||
 | 
			
		||||
	let rect: DOMRect;
 | 
			
		||||
 | 
			
		||||
	if (props.anchorElement) {
 | 
			
		||||
		rect = props.anchorElement.getBoundingClientRect();
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	const calcPosWhenTop = () => {
 | 
			
		||||
		let left: number;
 | 
			
		||||
		let top: number;
 | 
			
		||||
 | 
			
		||||
		if (props.anchorElement) {
 | 
			
		||||
			left = rect.left + window.pageXOffset + (props.anchorElement.offsetWidth / 2);
 | 
			
		||||
			top = (rect.top + window.pageYOffset - contentHeight) - props.innerMargin;
 | 
			
		||||
		} else {
 | 
			
		||||
			left = props.x;
 | 
			
		||||
			top = (props.y - contentHeight) - props.innerMargin;
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		left -= (el.offsetWidth / 2);
 | 
			
		||||
 | 
			
		||||
		if (left + contentWidth - window.pageXOffset > window.innerWidth) {
 | 
			
		||||
			left = window.innerWidth - contentWidth + window.pageXOffset - 1;
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		return [left, top];
 | 
			
		||||
	};
 | 
			
		||||
 | 
			
		||||
	const calcPosWhenBottom = () => {
 | 
			
		||||
		let left: number;
 | 
			
		||||
		let top: number;
 | 
			
		||||
 | 
			
		||||
		if (props.anchorElement) {
 | 
			
		||||
			left = rect.left + window.pageXOffset + (props.anchorElement.offsetWidth / 2);
 | 
			
		||||
			top = (rect.top + window.pageYOffset + props.anchorElement.offsetHeight) + props.innerMargin;
 | 
			
		||||
		} else {
 | 
			
		||||
			left = props.x;
 | 
			
		||||
			top = (props.y) + props.innerMargin;
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		left -= (el.offsetWidth / 2);
 | 
			
		||||
 | 
			
		||||
		if (left + contentWidth - window.pageXOffset > window.innerWidth) {
 | 
			
		||||
			left = window.innerWidth - contentWidth + window.pageXOffset - 1;
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		return [left, top];
 | 
			
		||||
	};
 | 
			
		||||
 | 
			
		||||
	const calcPosWhenLeft = () => {
 | 
			
		||||
		let left: number;
 | 
			
		||||
		let top: number;
 | 
			
		||||
 | 
			
		||||
		if (props.anchorElement) {
 | 
			
		||||
			left = (rect.left + window.pageXOffset - contentWidth) - props.innerMargin;
 | 
			
		||||
			top = rect.top + window.pageYOffset + (props.anchorElement.offsetHeight / 2);
 | 
			
		||||
		} else {
 | 
			
		||||
			left = (props.x - contentWidth) - props.innerMargin;
 | 
			
		||||
			top = props.y;
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		top -= (el.offsetHeight / 2);
 | 
			
		||||
 | 
			
		||||
		if (top + contentHeight - window.pageYOffset > window.innerHeight) {
 | 
			
		||||
			top = window.innerHeight - contentHeight + window.pageYOffset - 1;
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		return [left, top];
 | 
			
		||||
	};
 | 
			
		||||
 | 
			
		||||
	const calcPosWhenRight = () => {
 | 
			
		||||
		let left: number;
 | 
			
		||||
		let top: number;
 | 
			
		||||
 | 
			
		||||
		if (props.anchorElement) {
 | 
			
		||||
			left = (rect.left + props.anchorElement.offsetWidth + window.pageXOffset) + props.innerMargin;
 | 
			
		||||
 | 
			
		||||
			if (props.align === 'top') {
 | 
			
		||||
				top = rect.top + window.pageYOffset;
 | 
			
		||||
				if (props.alignOffset != null) top += props.alignOffset;
 | 
			
		||||
			} else if (props.align === 'bottom') {
 | 
			
		||||
				// TODO
 | 
			
		||||
			} else { // center
 | 
			
		||||
				top = rect.top + window.pageYOffset + (props.anchorElement.offsetHeight / 2);
 | 
			
		||||
				top -= (el.offsetHeight / 2);
 | 
			
		||||
			}
 | 
			
		||||
		} else {
 | 
			
		||||
			left = props.x + props.innerMargin;
 | 
			
		||||
			top = props.y;
 | 
			
		||||
			top -= (el.offsetHeight / 2);
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		if (top + contentHeight - window.pageYOffset > window.innerHeight) {
 | 
			
		||||
			top = window.innerHeight - contentHeight + window.pageYOffset - 1;
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		return [left, top];
 | 
			
		||||
	};
 | 
			
		||||
 | 
			
		||||
	const calc = (): {
 | 
			
		||||
		left: number;
 | 
			
		||||
		top: number;
 | 
			
		||||
		transformOrigin: string;
 | 
			
		||||
	} => {
 | 
			
		||||
		switch (props.direction) {
 | 
			
		||||
			case 'top': {
 | 
			
		||||
				const [left, top] = calcPosWhenTop();
 | 
			
		||||
 | 
			
		||||
				// ツールチップを上に向かって表示するスペースがなければ下に向かって出す
 | 
			
		||||
				if (top - window.pageYOffset < 0) {
 | 
			
		||||
					const [left, top] = calcPosWhenBottom();
 | 
			
		||||
					return { left, top, transformOrigin: 'center top' };
 | 
			
		||||
				}
 | 
			
		||||
 | 
			
		||||
				return { left, top, transformOrigin: 'center bottom' };
 | 
			
		||||
			}
 | 
			
		||||
 | 
			
		||||
			case 'bottom': {
 | 
			
		||||
				const [left, top] = calcPosWhenBottom();
 | 
			
		||||
				// TODO: ツールチップを下に向かって表示するスペースがなければ上に向かって出す
 | 
			
		||||
				return { left, top, transformOrigin: 'center top' };
 | 
			
		||||
			}
 | 
			
		||||
 | 
			
		||||
			case 'left': {
 | 
			
		||||
				const [left, top] = calcPosWhenLeft();
 | 
			
		||||
 | 
			
		||||
				// ツールチップを左に向かって表示するスペースがなければ右に向かって出す
 | 
			
		||||
				if (left - window.pageXOffset < 0) {
 | 
			
		||||
					const [left, top] = calcPosWhenRight();
 | 
			
		||||
					return { left, top, transformOrigin: 'left center' };
 | 
			
		||||
				}
 | 
			
		||||
 | 
			
		||||
				return { left, top, transformOrigin: 'right center' };
 | 
			
		||||
			}
 | 
			
		||||
 | 
			
		||||
			case 'right': {
 | 
			
		||||
				const [left, top] = calcPosWhenRight();
 | 
			
		||||
				// TODO: ツールチップを右に向かって表示するスペースがなければ左に向かって出す
 | 
			
		||||
				return { left, top, transformOrigin: 'left center' };
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
	};
 | 
			
		||||
 | 
			
		||||
	return calc();
 | 
			
		||||
}
 | 
			
		||||
| 
						 | 
				
			
			@ -11,10 +11,11 @@ export type MenuA = { type: 'a', href: string, target?: string, download?: strin
 | 
			
		|||
export type MenuUser = { type: 'user', user: Misskey.entities.User, active?: boolean, indicate?: boolean, action: MenuAction };
 | 
			
		||||
export type MenuSwitch = { type: 'switch', ref: Ref<boolean>, text: string, disabled?: boolean };
 | 
			
		||||
export type MenuButton = { type?: 'button', text: string, icon?: string, indicate?: boolean, danger?: boolean, active?: boolean, avatar?: Misskey.entities.User; action: MenuAction };
 | 
			
		||||
export type MenuParent = { type: 'parent', text: string, icon?: string, children: OuterMenuItem[] };
 | 
			
		||||
 | 
			
		||||
export type MenuPending = { type: 'pending' };
 | 
			
		||||
 | 
			
		||||
type OuterMenuItem = MenuDivider | MenuNull | MenuLabel | MenuLink | MenuA | MenuUser | MenuSwitch | MenuButton;
 | 
			
		||||
type OuterPromiseMenuItem = Promise<MenuLabel | MenuLink | MenuA | MenuUser | MenuSwitch | MenuButton>;
 | 
			
		||||
type OuterMenuItem = MenuDivider | MenuNull | MenuLabel | MenuLink | MenuA | MenuUser | MenuSwitch | MenuButton | MenuParent;
 | 
			
		||||
type OuterPromiseMenuItem = Promise<MenuLabel | MenuLink | MenuA | MenuUser | MenuSwitch | MenuButton | MenuParent>;
 | 
			
		||||
export type MenuItem = OuterMenuItem | OuterPromiseMenuItem;
 | 
			
		||||
export type InnerMenuItem = MenuDivider | MenuPending | MenuLabel | MenuLink | MenuA | MenuUser | MenuSwitch | MenuButton;
 | 
			
		||||
export type InnerMenuItem = MenuDivider | MenuPending | MenuLabel | MenuLink | MenuA | MenuUser | MenuSwitch | MenuButton | MenuParent;
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -1,5 +1,6 @@
 | 
			
		|||
<template>
 | 
			
		||||
<component :is="popup.component"
 | 
			
		||||
<component
 | 
			
		||||
	:is="popup.component"
 | 
			
		||||
	v-for="popup in popups"
 | 
			
		||||
	:key="popup.id"
 | 
			
		||||
	v-bind="popup.props"
 | 
			
		||||
| 
						 | 
				
			
			@ -15,56 +16,45 @@
 | 
			
		|||
<div v-if="dev" id="devTicker"><span>DEV BUILD</span></div>
 | 
			
		||||
</template>
 | 
			
		||||
 | 
			
		||||
<script lang="ts">
 | 
			
		||||
import { defineAsyncComponent, defineComponent } from 'vue';
 | 
			
		||||
<script lang="ts" setup>
 | 
			
		||||
import { defineAsyncComponent } from 'vue';
 | 
			
		||||
import { swInject } from './sw-inject';
 | 
			
		||||
import { popup, popups, pendingApiRequestsCount } from '@/os';
 | 
			
		||||
import { uploads } from '@/scripts/upload';
 | 
			
		||||
import * as sound from '@/scripts/sound';
 | 
			
		||||
import { $i } from '@/account';
 | 
			
		||||
import { swInject } from './sw-inject';
 | 
			
		||||
import { stream } from '@/stream';
 | 
			
		||||
 | 
			
		||||
export default defineComponent({
 | 
			
		||||
	components: {
 | 
			
		||||
		XStreamIndicator: defineAsyncComponent(() => import('./stream-indicator.vue')),
 | 
			
		||||
		XUpload: defineAsyncComponent(() => import('./upload.vue')),
 | 
			
		||||
	},
 | 
			
		||||
const XStreamIndicator = defineAsyncComponent(() => import('./stream-indicator.vue'));
 | 
			
		||||
const XUpload = defineAsyncComponent(() => import('./upload.vue'));
 | 
			
		||||
 | 
			
		||||
	setup() {
 | 
			
		||||
		const onNotification = notification => {
 | 
			
		||||
			if ($i.mutingNotificationTypes.includes(notification.type)) return;
 | 
			
		||||
const dev = _DEV_;
 | 
			
		||||
 | 
			
		||||
			if (document.visibilityState === 'visible') {
 | 
			
		||||
				stream.send('readNotification', {
 | 
			
		||||
					id: notification.id
 | 
			
		||||
				});
 | 
			
		||||
const onNotification = notification => {
 | 
			
		||||
	if ($i.mutingNotificationTypes.includes(notification.type)) return;
 | 
			
		||||
 | 
			
		||||
				popup(defineAsyncComponent(() => import('@/components/notification-toast.vue')), {
 | 
			
		||||
					notification
 | 
			
		||||
				}, {}, 'closed');
 | 
			
		||||
			}
 | 
			
		||||
	if (document.visibilityState === 'visible') {
 | 
			
		||||
		stream.send('readNotification', {
 | 
			
		||||
			id: notification.id,
 | 
			
		||||
		});
 | 
			
		||||
 | 
			
		||||
			sound.play('notification');
 | 
			
		||||
		};
 | 
			
		||||
		popup(defineAsyncComponent(() => import('@/components/notification-toast.vue')), {
 | 
			
		||||
			notification,
 | 
			
		||||
		}, {}, 'closed');
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
		if ($i) {
 | 
			
		||||
			const connection = stream.useChannel('main', null, 'UI');
 | 
			
		||||
			connection.on('notification', onNotification);
 | 
			
		||||
	sound.play('notification');
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
			//#region Listen message from SW
 | 
			
		||||
			if ('serviceWorker' in navigator) {
 | 
			
		||||
				swInject();
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
if ($i) {
 | 
			
		||||
	const connection = stream.useChannel('main', null, 'UI');
 | 
			
		||||
	connection.on('notification', onNotification);
 | 
			
		||||
 | 
			
		||||
		return {
 | 
			
		||||
			uploads,
 | 
			
		||||
			popups,
 | 
			
		||||
			pendingApiRequestsCount,
 | 
			
		||||
			dev: _DEV_,
 | 
			
		||||
		};
 | 
			
		||||
	},
 | 
			
		||||
});
 | 
			
		||||
	//#region Listen message from SW
 | 
			
		||||
	if ('serviceWorker' in navigator) {
 | 
			
		||||
		swInject();
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
</script>
 | 
			
		||||
 | 
			
		||||
<style lang="scss">
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -87,6 +87,36 @@ function openInstanceMenu(ev: MouseEvent) {
 | 
			
		|||
		text: i18n.ts.federation,
 | 
			
		||||
		icon: 'fas fa-globe',
 | 
			
		||||
		to: '/about#federation',
 | 
			
		||||
	}, null, {
 | 
			
		||||
		type: 'parent',
 | 
			
		||||
		text: i18n.ts.help,
 | 
			
		||||
		icon: 'fas fa-question-circle',
 | 
			
		||||
		children: [{
 | 
			
		||||
			type: 'link',
 | 
			
		||||
			to: '/mfm-cheat-sheet',
 | 
			
		||||
			text: i18n.ts._mfm.cheatSheet,
 | 
			
		||||
			icon: 'fas fa-code',
 | 
			
		||||
		}, {
 | 
			
		||||
			type: 'link',
 | 
			
		||||
			to: '/scratchpad',
 | 
			
		||||
			text: i18n.ts.scratchpad,
 | 
			
		||||
			icon: 'fas fa-terminal',
 | 
			
		||||
		}, {
 | 
			
		||||
			type: 'link',
 | 
			
		||||
			to: '/api-console',
 | 
			
		||||
			text: 'API Console',
 | 
			
		||||
			icon: 'fas fa-terminal',
 | 
			
		||||
		}, null, {
 | 
			
		||||
			text: i18n.ts.document,
 | 
			
		||||
			icon: 'fas fa-question-circle',
 | 
			
		||||
			action: () => {
 | 
			
		||||
				window.open('https://misskey-hub.net/help.html', '_blank');
 | 
			
		||||
			},
 | 
			
		||||
		}],
 | 
			
		||||
	}, {
 | 
			
		||||
		type: 'link',
 | 
			
		||||
		text: i18n.ts.aboutMisskey,
 | 
			
		||||
		to: '/about-misskey',
 | 
			
		||||
	}], ev.currentTarget ?? ev.target, {
 | 
			
		||||
		align: 'left',
 | 
			
		||||
	});
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -110,6 +110,36 @@ function openInstanceMenu(ev: MouseEvent) {
 | 
			
		|||
		text: i18n.ts.federation,
 | 
			
		||||
		icon: 'fas fa-globe',
 | 
			
		||||
		to: '/about#federation',
 | 
			
		||||
	}, null, {
 | 
			
		||||
		type: 'parent',
 | 
			
		||||
		text: i18n.ts.help,
 | 
			
		||||
		icon: 'fas fa-question-circle',
 | 
			
		||||
		children: [{
 | 
			
		||||
			type: 'link',
 | 
			
		||||
			to: '/mfm-cheat-sheet',
 | 
			
		||||
			text: i18n.ts._mfm.cheatSheet,
 | 
			
		||||
			icon: 'fas fa-code',
 | 
			
		||||
		}, {
 | 
			
		||||
			type: 'link',
 | 
			
		||||
			to: '/scratchpad',
 | 
			
		||||
			text: i18n.ts.scratchpad,
 | 
			
		||||
			icon: 'fas fa-terminal',
 | 
			
		||||
		}, {
 | 
			
		||||
			type: 'link',
 | 
			
		||||
			to: '/api-console',
 | 
			
		||||
			text: 'API Console',
 | 
			
		||||
			icon: 'fas fa-terminal',
 | 
			
		||||
		}, null, {
 | 
			
		||||
			text: i18n.ts.document,
 | 
			
		||||
			icon: 'fas fa-question-circle',
 | 
			
		||||
			action: () => {
 | 
			
		||||
				window.open('https://misskey-hub.net/help.html', '_blank');
 | 
			
		||||
			},
 | 
			
		||||
		}],
 | 
			
		||||
	}, {
 | 
			
		||||
		type: 'link',
 | 
			
		||||
		text: i18n.ts.aboutMisskey,
 | 
			
		||||
		to: '/about-misskey',
 | 
			
		||||
	}], ev.currentTarget ?? ev.target, {
 | 
			
		||||
		align: 'left',
 | 
			
		||||
	});
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
		Loading…
	
	Add table
		
		Reference in a new issue