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>
|
</MkA>
|
||||||
</template>
|
</template>
|
||||||
</div>
|
</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>
|
</div>
|
||||||
</MkModal>
|
</MkModal>
|
||||||
</template>
|
</template>
|
||||||
|
@ -74,28 +60,6 @@ const items = Object.keys(navbarItemDef).filter(k => !menu.includes(k)).map(k =>
|
||||||
function close() {
|
function close() {
|
||||||
modal.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>
|
</script>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<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>
|
<template>
|
||||||
<div
|
<div>
|
||||||
ref="itemsEl" v-hotkey="keymap"
|
<div
|
||||||
class="rrevdjwt"
|
ref="itemsEl" v-hotkey="keymap"
|
||||||
:class="{ center: align === 'center', asDrawer }"
|
class="rrevdjwt _popup _shadow"
|
||||||
:style="{ width: (width && !asDrawer) ? width + 'px' : '', maxHeight: maxHeight ? maxHeight + 'px' : '' }"
|
:class="{ center: align === 'center', asDrawer }"
|
||||||
@contextmenu.self="e => e.preventDefault()"
|
: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>
|
<template v-for="(item, i) in items2">
|
||||||
<span v-else-if="item.type === 'label'" class="label item">
|
<div v-if="item === null" class="divider"></div>
|
||||||
<span>{{ item.text }}</span>
|
<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>
|
||||||
<span v-else-if="item.type === 'pending'" :tabindex="i" class="pending item">
|
</div>
|
||||||
<span><MkEllipsis/></span>
|
<div v-if="childMenu" class="child">
|
||||||
</span>
|
<XChild ref="child" :items="childMenu" :target-element="childTarget" showing @actioned="childActioned"/>
|
||||||
<MkA v-else-if="item.type === 'link'" :to="item.to" :tabindex="i" class="_button item" @click.passive="close()">
|
</div>
|
||||||
<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>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script lang="ts" setup>
|
<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 { focusPrev, focusNext } from '@/scripts/focus';
|
||||||
import FormSwitch from '@/components/form/switch.vue';
|
import FormSwitch from '@/components/form/switch.vue';
|
||||||
import { MenuItem, InnerMenuItem, MenuPending, MenuAction } from '@/types/menu';
|
import { MenuItem, InnerMenuItem, MenuPending, MenuAction } from '@/types/menu';
|
||||||
|
import * as os from '@/os';
|
||||||
|
const XChild = defineAsyncComponent(() => import('./child-menu.vue'));
|
||||||
|
|
||||||
const props = defineProps<{
|
const props = defineProps<{
|
||||||
items: MenuItem[];
|
items: MenuItem[];
|
||||||
|
@ -61,19 +73,23 @@ const props = defineProps<{
|
||||||
}>();
|
}>();
|
||||||
|
|
||||||
const emit = defineEmits<{
|
const emit = defineEmits<{
|
||||||
(ev: 'close'): void;
|
(ev: 'close', actioned?: boolean): void;
|
||||||
}>();
|
}>();
|
||||||
|
|
||||||
let itemsEl = $ref<HTMLDivElement>();
|
let itemsEl = $ref<HTMLDivElement>();
|
||||||
|
|
||||||
let items2: InnerMenuItem[] = $ref([]);
|
let items2: InnerMenuItem[] = $ref([]);
|
||||||
|
|
||||||
|
let child = $ref<InstanceType<typeof XChild>>();
|
||||||
|
|
||||||
let keymap = $computed(() => ({
|
let keymap = $computed(() => ({
|
||||||
'up|k|shift+tab': focusUp,
|
'up|k|shift+tab': focusUp,
|
||||||
'down|j|tab': focusDown,
|
'down|j|tab': focusDown,
|
||||||
'esc': close,
|
'esc': close,
|
||||||
}));
|
}));
|
||||||
|
|
||||||
|
let childShowingItem = $ref<MenuItem | null>();
|
||||||
|
|
||||||
watch(() => props.items, () => {
|
watch(() => props.items, () => {
|
||||||
const items: (MenuItem | MenuPending)[] = [...props.items].filter(item => item !== undefined);
|
const items: (MenuItem | MenuPending)[] = [...props.items].filter(item => item !== undefined);
|
||||||
|
|
||||||
|
@ -93,21 +109,53 @@ watch(() => props.items, () => {
|
||||||
immediate: true,
|
immediate: true,
|
||||||
});
|
});
|
||||||
|
|
||||||
onMounted(() => {
|
let childMenu = $ref<MenuItem[] | null>();
|
||||||
if (props.viaKeyboard) {
|
let childTarget = $ref<HTMLElement | null>();
|
||||||
nextTick(() => {
|
|
||||||
focusNext(itemsEl.children[0], true, false);
|
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) {
|
function clicked(fn: MenuAction, ev: MouseEvent) {
|
||||||
fn(ev);
|
fn(ev);
|
||||||
close();
|
close(true);
|
||||||
}
|
}
|
||||||
|
|
||||||
function close() {
|
function close(actioned = false) {
|
||||||
emit('close');
|
emit('close', actioned);
|
||||||
}
|
}
|
||||||
|
|
||||||
function focusUp() {
|
function focusUp() {
|
||||||
|
@ -117,6 +165,20 @@ function focusUp() {
|
||||||
function focusDown() {
|
function focusDown() {
|
||||||
focusNext(document.activeElement);
|
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>
|
</script>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
|
@ -225,6 +287,25 @@ function focusDown() {
|
||||||
opacity: 0.7;
|
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 {
|
> i {
|
||||||
margin-right: 5px;
|
margin-right: 5px;
|
||||||
width: 20px;
|
width: 20px;
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
<template>
|
<template>
|
||||||
<MkModal ref="modal" v-slot="{ type, maxHeight }" :z-priority="'high'" :src="src" :transparent-bg="true" @click="modal.close()" @closed="emit('closed')">
|
<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>
|
</MkModal>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
|
|
|
@ -12,6 +12,7 @@
|
||||||
<script lang="ts" setup>
|
<script lang="ts" setup>
|
||||||
import { nextTick, onMounted, onUnmounted, ref } from 'vue';
|
import { nextTick, onMounted, onUnmounted, ref } from 'vue';
|
||||||
import * as os from '@/os';
|
import * as os from '@/os';
|
||||||
|
import { calcPopupPosition } from '@/scripts/popup-position';
|
||||||
|
|
||||||
const props = withDefaults(defineProps<{
|
const props = withDefaults(defineProps<{
|
||||||
showing: boolean;
|
showing: boolean;
|
||||||
|
@ -36,151 +37,20 @@ const emit = defineEmits<{
|
||||||
const el = ref<HTMLElement>();
|
const el = ref<HTMLElement>();
|
||||||
const zIndex = os.claimZIndex('high');
|
const zIndex = os.claimZIndex('high');
|
||||||
|
|
||||||
const setPosition = () => {
|
function setPosition() {
|
||||||
if (el.value == null) return;
|
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;
|
el.value.style.transformOrigin = data.transformOrigin;
|
||||||
const contentHeight = el.value.offsetHeight;
|
el.value.style.left = data.left + 'px';
|
||||||
|
el.value.style.top = data.top + 'px';
|
||||||
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';
|
|
||||||
};
|
|
||||||
|
|
||||||
let loopHandler;
|
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 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 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 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' };
|
export type MenuPending = { type: 'pending' };
|
||||||
|
|
||||||
type OuterMenuItem = MenuDivider | MenuNull | 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>;
|
type OuterPromiseMenuItem = Promise<MenuLabel | MenuLink | MenuA | MenuUser | MenuSwitch | MenuButton | MenuParent>;
|
||||||
export type MenuItem = OuterMenuItem | OuterPromiseMenuItem;
|
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>
|
<template>
|
||||||
<component :is="popup.component"
|
<component
|
||||||
|
:is="popup.component"
|
||||||
v-for="popup in popups"
|
v-for="popup in popups"
|
||||||
:key="popup.id"
|
:key="popup.id"
|
||||||
v-bind="popup.props"
|
v-bind="popup.props"
|
||||||
|
@ -15,56 +16,45 @@
|
||||||
<div v-if="dev" id="devTicker"><span>DEV BUILD</span></div>
|
<div v-if="dev" id="devTicker"><span>DEV BUILD</span></div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script lang="ts">
|
<script lang="ts" setup>
|
||||||
import { defineAsyncComponent, defineComponent } from 'vue';
|
import { defineAsyncComponent } from 'vue';
|
||||||
|
import { swInject } from './sw-inject';
|
||||||
import { popup, popups, pendingApiRequestsCount } from '@/os';
|
import { popup, popups, pendingApiRequestsCount } from '@/os';
|
||||||
import { uploads } from '@/scripts/upload';
|
import { uploads } from '@/scripts/upload';
|
||||||
import * as sound from '@/scripts/sound';
|
import * as sound from '@/scripts/sound';
|
||||||
import { $i } from '@/account';
|
import { $i } from '@/account';
|
||||||
import { swInject } from './sw-inject';
|
|
||||||
import { stream } from '@/stream';
|
import { stream } from '@/stream';
|
||||||
|
|
||||||
export default defineComponent({
|
const XStreamIndicator = defineAsyncComponent(() => import('./stream-indicator.vue'));
|
||||||
components: {
|
const XUpload = defineAsyncComponent(() => import('./upload.vue'));
|
||||||
XStreamIndicator: defineAsyncComponent(() => import('./stream-indicator.vue')),
|
|
||||||
XUpload: defineAsyncComponent(() => import('./upload.vue')),
|
|
||||||
},
|
|
||||||
|
|
||||||
setup() {
|
const dev = _DEV_;
|
||||||
const onNotification = notification => {
|
|
||||||
if ($i.mutingNotificationTypes.includes(notification.type)) return;
|
|
||||||
|
|
||||||
if (document.visibilityState === 'visible') {
|
const onNotification = notification => {
|
||||||
stream.send('readNotification', {
|
if ($i.mutingNotificationTypes.includes(notification.type)) return;
|
||||||
id: notification.id
|
|
||||||
});
|
|
||||||
|
|
||||||
popup(defineAsyncComponent(() => import('@/components/notification-toast.vue')), {
|
if (document.visibilityState === 'visible') {
|
||||||
notification
|
stream.send('readNotification', {
|
||||||
}, {}, 'closed');
|
id: notification.id,
|
||||||
}
|
});
|
||||||
|
|
||||||
sound.play('notification');
|
popup(defineAsyncComponent(() => import('@/components/notification-toast.vue')), {
|
||||||
};
|
notification,
|
||||||
|
}, {}, 'closed');
|
||||||
|
}
|
||||||
|
|
||||||
if ($i) {
|
sound.play('notification');
|
||||||
const connection = stream.useChannel('main', null, 'UI');
|
};
|
||||||
connection.on('notification', onNotification);
|
|
||||||
|
|
||||||
//#region Listen message from SW
|
if ($i) {
|
||||||
if ('serviceWorker' in navigator) {
|
const connection = stream.useChannel('main', null, 'UI');
|
||||||
swInject();
|
connection.on('notification', onNotification);
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return {
|
//#region Listen message from SW
|
||||||
uploads,
|
if ('serviceWorker' in navigator) {
|
||||||
popups,
|
swInject();
|
||||||
pendingApiRequestsCount,
|
}
|
||||||
dev: _DEV_,
|
}
|
||||||
};
|
|
||||||
},
|
|
||||||
});
|
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss">
|
<style lang="scss">
|
||||||
|
|
|
@ -87,6 +87,36 @@ function openInstanceMenu(ev: MouseEvent) {
|
||||||
text: i18n.ts.federation,
|
text: i18n.ts.federation,
|
||||||
icon: 'fas fa-globe',
|
icon: 'fas fa-globe',
|
||||||
to: '/about#federation',
|
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, {
|
}], ev.currentTarget ?? ev.target, {
|
||||||
align: 'left',
|
align: 'left',
|
||||||
});
|
});
|
||||||
|
|
|
@ -110,6 +110,36 @@ function openInstanceMenu(ev: MouseEvent) {
|
||||||
text: i18n.ts.federation,
|
text: i18n.ts.federation,
|
||||||
icon: 'fas fa-globe',
|
icon: 'fas fa-globe',
|
||||||
to: '/about#federation',
|
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, {
|
}], ev.currentTarget ?? ev.target, {
|
||||||
align: 'left',
|
align: 'left',
|
||||||
});
|
});
|
||||||
|
|
Loading…
Reference in a new issue