Skip to content
Snippets Groups Projects
column.vue 8.74 KiB
Newer Older
syuilo's avatar
syuilo committed
<template>
syuilo's avatar
syuilo committed
<div
	:class="[$style.root, { [$style.paged]: isMainColumn, [$style.naked]: naked, [$style.active]: active, [$style.isStacked]: isStacked, [$style.draghover]: draghover, [$style.dragging]: dragging, [$style.dropready]: dropready }]"
syuilo's avatar
syuilo committed
	@dragover.prevent.stop="onDragover"
	@dragleave="onDragleave"
	@drop.prevent.stop="onDrop"
>
syuilo's avatar
syuilo committed
	<header
		:class="[$style.header]"
syuilo's avatar
syuilo committed
		draggable="true"
		@click="goTop"
		@dragstart="onDragstart"
		@dragend="onDragend"
		@contextmenu.prevent.stop="onContextmenu"
	>
syuilo's avatar
syuilo committed
		<svg viewBox="0 0 256 128" :class="$style.tabShape">
			<g transform="matrix(6.2431,0,0,6.2431,-677.417,-29.3839)">
syuilo's avatar
syuilo committed
				<path d="M149.512,4.707L108.507,4.707C116.252,4.719 118.758,14.958 118.758,14.958C118.758,14.958 121.381,25.283 129.009,25.209L149.512,25.209L149.512,4.707Z" style="fill:var(--deckBg);"/>
syuilo's avatar
syuilo committed
			</g>
		</svg>
syuilo's avatar
syuilo committed
		<div :class="$style.color"></div>
		<button v-if="isStacked && !isMainColumn" :class="$style.toggleActive" class="_button" @click="toggleActive">
syuilo's avatar
syuilo committed
			<template v-if="active"><i class="ti ti-chevron-up"></i></template>
			<template v-else><i class="ti ti-chevron-down"></i></template>
syuilo's avatar
syuilo committed
		</button>
		<span :class="$style.title"><slot name="header"></slot></span>
		<button v-tooltip="i18n.ts.settings" :class="$style.menu" class="_button" @click.stop="showSettingsMenu"><i class="ti ti-dots"></i></button>
syuilo's avatar
syuilo committed
	</header>
syuilo's avatar
syuilo committed
	<div v-if="active" ref="body" :class="$style.body">
syuilo's avatar
syuilo committed
		<slot></slot>
	</div>
syuilo's avatar
syuilo committed
</div>
syuilo's avatar
syuilo committed
</template>

<script lang="ts" setup>
import { onBeforeUnmount, onMounted, provide, watch } from 'vue';
import { updateColumn, swapLeftColumn, swapRightColumn, swapUpColumn, swapDownColumn, stackLeftColumn, popRightColumn, removeColumn, swapColumn, Column } from './deck-store';
syuilo's avatar
syuilo committed
import * as os from '@/os';
import { i18n } from '@/i18n';
syuilo's avatar
syuilo committed
import { MenuItem } from '@/types/menu';

provide('shouldHeaderThin', true);
provide('shouldOmitHeaderTitle', true);
provide('forceSpacerMin', true);

const props = withDefaults(defineProps<{
	column: Column;
	isStacked?: boolean;
	naked?: boolean;
syuilo's avatar
syuilo committed
	menu?: MenuItem[];
}>(), {
	isStacked: false,
	naked: false,
});
syuilo's avatar
syuilo committed

syuilo's avatar
syuilo committed
let body = $shallowRef<HTMLDivElement | null>();
syuilo's avatar
syuilo committed

let dragging = $ref(false);
watch($$(dragging), v => os.deckGlobalEvents.emit(v ? 'column.dragStart' : 'column.dragEnd'));

let draghover = $ref(false);
let dropready = $ref(false);
syuilo's avatar
syuilo committed

const isMainColumn = $computed(() => props.column.type === 'main');
const active = $computed(() => props.column.active !== false);

onMounted(() => {
	os.deckGlobalEvents.on('column.dragStart', onOtherDragStart);
	os.deckGlobalEvents.on('column.dragEnd', onOtherDragEnd);
});

onBeforeUnmount(() => {
	os.deckGlobalEvents.off('column.dragStart', onOtherDragStart);
	os.deckGlobalEvents.off('column.dragEnd', onOtherDragEnd);
});
syuilo's avatar
syuilo committed

function onOtherDragStart() {
	dropready = true;
}

function onOtherDragEnd() {
	dropready = false;
}

function toggleActive() {
	if (!props.isStacked) return;
	updateColumn(props.column.id, {
syuilo's avatar
syuilo committed
		active: !props.column.active,
syuilo's avatar
syuilo committed
	let items = [{
syuilo's avatar
syuilo committed
		icon: 'ti ti-settings',
syuilo's avatar
syuilo committed
		text: i18n.ts._deck.configureColumn,
		action: async () => {
			const { canceled, result } = await os.form(props.column.name, {
				name: {
					type: 'string',
					label: i18n.ts.name,
syuilo's avatar
syuilo committed
					default: props.column.name,
				},
				width: {
					type: 'number',
					label: i18n.ts.width,
syuilo's avatar
syuilo committed
					default: props.column.width,
				},
				flexible: {
					type: 'boolean',
					label: i18n.ts.flexible,
syuilo's avatar
syuilo committed
					default: props.column.flexible,
				},
syuilo's avatar
syuilo committed
			});
			if (canceled) return;
			updateColumn(props.column.id, result);
syuilo's avatar
syuilo committed
		},
syuilo's avatar
syuilo committed
		type: 'parent',
		text: i18n.ts.move + '...',
syuilo's avatar
syuilo committed
		icon: 'ti ti-arrows-move',
syuilo's avatar
syuilo committed
		children: [{
syuilo's avatar
syuilo committed
			icon: 'ti ti-arrow-left',
syuilo's avatar
syuilo committed
			text: i18n.ts._deck.swapLeft,
			action: () => {
				swapLeftColumn(props.column.id);
			},
		}, {
syuilo's avatar
syuilo committed
			icon: 'ti ti-arrow-right',
syuilo's avatar
syuilo committed
			text: i18n.ts._deck.swapRight,
			action: () => {
				swapRightColumn(props.column.id);
			},
		}, props.isStacked ? {
syuilo's avatar
syuilo committed
			icon: 'ti ti-arrow-up',
syuilo's avatar
syuilo committed
			text: i18n.ts._deck.swapUp,
			action: () => {
				swapUpColumn(props.column.id);
			},
		} : undefined, props.isStacked ? {
syuilo's avatar
syuilo committed
			icon: 'ti ti-arrow-down',
syuilo's avatar
syuilo committed
			text: i18n.ts._deck.swapDown,
			action: () => {
				swapDownColumn(props.column.id);
			},
		} : undefined],
	}, {
syuilo's avatar
syuilo committed
		icon: 'ti ti-stack-2',
		text: i18n.ts._deck.stackLeft,
		action: () => {
			stackLeftColumn(props.column.id);
syuilo's avatar
syuilo committed
		},
syuilo's avatar
syuilo committed
		icon: 'ti ti-window-maximize',
		text: i18n.ts._deck.popRight,
		action: () => {
			popRightColumn(props.column.id);
syuilo's avatar
syuilo committed
		},
syuilo's avatar
syuilo committed
		icon: 'ti ti-trash',
		text: i18n.ts.remove,
		danger: true,
		action: () => {
			removeColumn(props.column.id);
syuilo's avatar
syuilo committed
		},
syuilo's avatar
syuilo committed
	if (props.menu) {
syuilo's avatar
syuilo committed
		items.unshift(null);
syuilo's avatar
syuilo committed
		items = props.menu.concat(items);
syuilo's avatar
syuilo committed
function showSettingsMenu(ev: MouseEvent) {
	os.popupMenu(getMenu(), ev.currentTarget ?? ev.target);
}

function onContextmenu(ev: MouseEvent) {
	os.contextMenu(getMenu(), ev);
}

function goTop() {
syuilo's avatar
syuilo committed
	if (body) {
		body.scrollTo({
			top: 0,
			behavior: 'smooth',
		});
	}
function onDragstart(ev) {
	ev.dataTransfer.effectAllowed = 'move';
	ev.dataTransfer.setData(_DATA_TRANSFER_DECK_COLUMN_, props.column.id);

	// Chromeのバグで、Dragstartハンドラ内ですぐにDOMを変更する(=リアクティブなプロパティを変更する)とDragが終了してしまう
	// SEE: https://stackoverflow.com/questions/19639969/html5-dragend-event-firing-immediately
	window.setTimeout(() => {
		dragging = true;
	}, 10);
}

	// 自分自身がドラッグされている場合
	if (dragging) {
		// 自分自身にはドロップさせない
		ev.dataTransfer.dropEffect = 'none';
	} else {
		const isDeckColumn = ev.dataTransfer.types[0] === _DATA_TRANSFER_DECK_COLUMN_;
		ev.dataTransfer.dropEffect = isDeckColumn ? 'move' : 'none';
		if (isDeckColumn) draghover = true;
	}
}

function onDragleave() {
	draghover = false;
}

	draghover = false;
	os.deckGlobalEvents.emit('column.dragEnd');

	const id = ev.dataTransfer.getData(_DATA_TRANSFER_DECK_COLUMN_);
	if (id != null && id !== '') {
		swapColumn(props.column.id, id);
	}
}
syuilo's avatar
syuilo committed
</script>

<style lang="scss" module>
.root {
syuilo's avatar
syuilo committed
	--root-margin: 10px;
syuilo's avatar
syuilo committed
	--deckColumnHeaderHeight: 38px;
syuilo's avatar
syuilo committed

syuilo's avatar
syuilo committed
	height: 100%;
syuilo's avatar
syuilo committed
	contain: strict;
syuilo's avatar
syuilo committed
	border-radius: 10px;
syuilo's avatar
syuilo committed

	&.draghover {
		&:after {
			content: "";
			display: block;
			position: absolute;
			z-index: 1000;
			top: 0;
			left: 0;
			width: 100%;
			height: 100%;
			background: var(--focus);
		}
	}

	&.dragging {
syuilo's avatar
syuilo committed
		&:after {
			content: "";
			display: block;
			position: absolute;
			z-index: 1000;
			top: 0;
			left: 0;
			width: 100%;
			height: 100%;
			background: var(--focus);
			opacity: 0.5;
		}
syuilo's avatar
syuilo committed
	}

	&.dropready {
		* {
			pointer-events: none;
		}
	}

	&:not(.active) {
syuilo's avatar
syuilo committed
		flex-basis: var(--deckColumnHeaderHeight);
		min-height: var(--deckColumnHeaderHeight);
syuilo's avatar
syuilo committed
		border-bottom-right-radius: 0;
syuilo's avatar
syuilo committed
	}

	&.naked {
syuilo's avatar
syuilo committed
		background: var(--acrylicBg) !important;
		-webkit-backdrop-filter: var(--blur, blur(10px));
		backdrop-filter: var(--blur, blur(10px));
syuilo's avatar
syuilo committed

		> .header {
syuilo's avatar
syuilo committed
			background: transparent;
			box-shadow: none;
			color: var(--fg);
syuilo's avatar
syuilo committed
		}
syuilo's avatar
syuilo committed

		> .body {
syuilo's avatar
syuilo committed
			background: transparent !important;

syuilo's avatar
syuilo committed
			&::-webkit-scrollbar-track {
syuilo's avatar
syuilo committed
				background: transparent;
syuilo's avatar
syuilo committed
			}
		}
syuilo's avatar
syuilo committed
	}

	&.paged {
syuilo's avatar
syuilo committed
		background: var(--bg) !important;
syuilo's avatar
syuilo committed

		> .body {
syuilo's avatar
syuilo committed
			background: var(--bg) !important;

syuilo's avatar
syuilo committed
			&::-webkit-scrollbar-track {
				background: inherit;
			}
		}
syuilo's avatar
syuilo committed
	}
syuilo's avatar
syuilo committed

.header {
	position: relative;
	display: flex;
	z-index: 2;
	line-height: var(--deckColumnHeaderHeight);
	height: var(--deckColumnHeaderHeight);
syuilo's avatar
syuilo committed
	padding: 0 16px 0 30px;
	font-size: 0.9em;
	color: var(--panelHeaderFg);
	background: var(--panelHeaderBg);
syuilo's avatar
syuilo committed
	box-shadow: 0 1px 0 0 var(--panelHeaderDivider);
	cursor: pointer;
syuilo's avatar
syuilo committed
	user-select: none;
syuilo's avatar
syuilo committed

syuilo's avatar
syuilo committed
.color {
	position: absolute;
syuilo's avatar
syuilo committed
	top: 12px;
	left: 12px;
syuilo's avatar
syuilo committed
	width: 3px;
syuilo's avatar
syuilo committed
	height: calc(100% - 24px);
syuilo's avatar
syuilo committed
	background: var(--accent);
	border-radius: 999px;
}

syuilo's avatar
syuilo committed
.tabShape {
	position: absolute;
	top: 0;
	right: -8px;
	width: auto;
	height: calc(100% - 6px);
}

.title {
	display: inline-block;
	align-items: center;
	overflow: hidden;
	text-overflow: ellipsis;
	white-space: nowrap;
	width: 100%;
}
syuilo's avatar
syuilo committed

.toggleActive,
.menu {
	z-index: 1;
	width: var(--deckColumnHeaderHeight);
	line-height: var(--deckColumnHeaderHeight);
	color: var(--faceTextButton);
syuilo's avatar
syuilo committed

	&:hover {
		color: var(--faceTextButtonHover);
	}
syuilo's avatar
syuilo committed

	&:active {
		color: var(--faceTextButtonActive);
	}
}
syuilo's avatar
syuilo committed

.toggleActive {
	margin-left: -16px;
}
syuilo's avatar
syuilo committed

.menu {
	margin-left: auto;
	margin-right: -16px;
}
syuilo's avatar
syuilo committed

.body {
	height: calc(100% - var(--deckColumnHeaderHeight));
	overflow-y: auto;
	overflow-x: hidden; // Safari does not supports clip
	overflow-x: clip;
	-webkit-overflow-scrolling: touch;
	box-sizing: border-box;
syuilo's avatar
syuilo committed
	container-type: size;
	background-color: var(--bg);
syuilo's avatar
syuilo committed

	&::-webkit-scrollbar-track {
		background: var(--panel);
	}
syuilo's avatar
syuilo committed
}
</style>