Skip to content
Snippets Groups Projects
header.vue 4.94 KiB
Newer Older
syuilo's avatar
syuilo committed
<template>
syuilo's avatar
syuilo committed
<div class="fdidabkb" :class="{ center }" :style="`--height:${height};`" :key="key">
syuilo's avatar
syuilo committed
	<transition :name="$store.state.animation ? 'header' : ''" mode="out-in" appear>
syuilo's avatar
syuilo committed
		<div class="buttons left" v-if="backButton">
			<button class="_button button back" @click.stop="$emit('back')" @touchstart="preventDrag" v-tooltip="$ts.goBack"><i class="fas fa-chevron-left"></i></button>
syuilo's avatar
syuilo committed
		</div>
syuilo's avatar
syuilo committed
	</transition>
	<template v-if="info">
		<div class="titleContainer">
syuilo's avatar
syuilo committed
			<i v-if="info.icon" class="icon" :class="info.icon"></i>
			<MkAvatar v-else-if="info.avatar" class="avatar" :user="info.avatar" :disable-preview="true" :show-indicator="true"/>

syuilo's avatar
syuilo committed
			<div class="title">
syuilo's avatar
syuilo committed
				<MkUserName v-if="info.userName" :user="info.userName" :nowrap="false" class="title"/>
				<div v-else-if="info.title" class="title">{{ info.title }}</div>
				<div class="subtitle" v-if="info.subtitle">
					{{ info.subtitle }}
				</div>
syuilo's avatar
syuilo committed
			</div>
syuilo's avatar
syuilo committed
		</div>
syuilo's avatar
syuilo committed
		<div class="buttons right">
syuilo's avatar
syuilo committed
			<template v-if="info.actions && showActions">
				<button v-for="action in info.actions" class="_button button" :class="{ highlighted: action.highlighted }" @click.stop="action.handler" @touchstart="preventDrag" v-tooltip="action.text"><i :class="action.icon"></i></button>
syuilo's avatar
syuilo committed
			</template>
			<button v-if="shouldShowMenu" class="_button button" @click.stop="showMenu" @touchstart="preventDrag" v-tooltip="$ts.menu"><i class="fas fa-ellipsis-h"></i></button>
			<button v-if="closeButton" class="_button button" @click.stop="$emit('close')" @touchstart="preventDrag" v-tooltip="$ts.close"><i class="fas fa-times"></i></button>
syuilo's avatar
syuilo committed
		</div>
syuilo's avatar
syuilo committed
	</template>
</div>
</template>

<script lang="ts">
import { defineComponent } from 'vue';
syuilo's avatar
syuilo committed
import { popupMenu } from '@client/os';
syuilo's avatar
syuilo committed
import { url } from '@client/config';
syuilo's avatar
syuilo committed

export default defineComponent({
	props: {
		info: {
			required: true
		},
syuilo's avatar
syuilo committed
		menu: {
			required: false
		},
		backButton: {
syuilo's avatar
syuilo committed
			type: Boolean,
			required: false,
syuilo's avatar
syuilo committed
			default: false,
		},
		closeButton: {
			type: Boolean,
			required: false,
			default: false,
syuilo's avatar
syuilo committed
		},
syuilo's avatar
syuilo committed
		center: {
			type: Boolean,
			required: false,
			default: true,
		},
syuilo's avatar
syuilo committed
	},

	data() {
		return {
syuilo's avatar
syuilo committed
			showActions: false,
syuilo's avatar
syuilo committed
			height: 0,
syuilo's avatar
syuilo committed
			key: 0,
syuilo's avatar
syuilo committed
		};
	},

syuilo's avatar
syuilo committed
	computed: {
syuilo's avatar
syuilo committed
		shouldShowMenu() {
syuilo's avatar
syuilo committed
			if (this.info.actions != null && !this.showActions) return true;
			if (this.info.menu != null) return true;
			if (this.info.share != null) return true;
syuilo's avatar
syuilo committed
			if (this.menu != null) return true;
syuilo's avatar
syuilo committed
			return false;
		}
	},

syuilo's avatar
syuilo committed
	watch: {
syuilo's avatar
syuilo committed
		info() {
			this.key++;
		},
syuilo's avatar
syuilo committed
	},

	mounted() {
		this.height = this.$el.parentElement.offsetHeight + 'px';
syuilo's avatar
syuilo committed
		this.showActions = this.$el.parentElement.offsetWidth >= 500;
syuilo's avatar
syuilo committed
		new ResizeObserver((entries, observer) => {
			this.height = this.$el.parentElement.offsetHeight + 'px';
syuilo's avatar
syuilo committed
			this.showActions = this.$el.parentElement.offsetWidth >= 500;
syuilo's avatar
syuilo committed
		}).observe(this.$el);
	},

	methods: {
syuilo's avatar
syuilo committed
		share() {
syuilo's avatar
syuilo committed
			navigator.share({
				url: url + this.info.path,
				...this.info.share,
			});
syuilo's avatar
syuilo committed
		showMenu(ev) {
syuilo's avatar
syuilo committed
			let menu = this.info.menu ? this.info.menu() : [];
			if (!this.showActions && this.info.actions) {
				menu = [...this.info.actions.map(x => ({
					text: x.text,
					icon: x.icon,
					action: x.handler
				})), menu.length > 0 ? null : undefined, ...menu];
			}
syuilo's avatar
syuilo committed
			if (this.info.share) {
				if (menu.length > 0) menu.push(null);
				menu.push({
					text: this.$ts.share,
					icon: 'fas fa-share-alt',
syuilo's avatar
syuilo committed
					action: this.share
				});
			}
syuilo's avatar
syuilo committed
			if (this.menu) {
				if (menu.length > 0) menu.push(null);
				menu = menu.concat(this.menu);
			}
syuilo's avatar
syuilo committed
			popupMenu(menu, ev.currentTarget || ev.target);
syuilo's avatar
syuilo committed
		}
syuilo's avatar
syuilo committed
	}
});
</script>

syuilo's avatar
syuilo committed
<style lang="scss" scoped>
syuilo's avatar
syuilo committed
.fdidabkb {
syuilo's avatar
syuilo committed
	display: flex;

syuilo's avatar
syuilo committed
	&.center {
		text-align: center;
syuilo's avatar
syuilo committed

syuilo's avatar
syuilo committed
		> .titleContainer {
			margin: 0 auto;
		}
syuilo's avatar
syuilo committed

syuilo's avatar
syuilo committed
		> .buttons {
			&.right {
				margin-left: 0;
			}
		}
syuilo's avatar
syuilo committed
	> .buttons {
syuilo's avatar
syuilo committed
		--margin: 8px;
		display: flex;
    align-items: center;
		height: var(--height);
		margin: 0 var(--margin);
syuilo's avatar
syuilo committed

syuilo's avatar
syuilo committed
		&.right {
			margin-left: auto;
		}

		&:empty {
syuilo's avatar
syuilo committed
			width: var(--height);
		}
syuilo's avatar
syuilo committed

		> .button {
			display: flex;
			align-items: center;
			justify-content: center;
			height: calc(var(--height) - (var(--margin) * 2));
			width: calc(var(--height) - (var(--margin) * 2));
			box-sizing: border-box;
			position: relative;
			border-radius: 5px;

			&:hover {
				background: rgba(0, 0, 0, 0.05);
			}

			&.highlighted {
				color: var(--accent);
			}
		}
syuilo's avatar
syuilo committed
	}

	> .titleContainer {
syuilo's avatar
syuilo committed
		display: flex;
		align-items: center;
syuilo's avatar
syuilo committed
		overflow: auto;
		white-space: nowrap;
syuilo's avatar
syuilo committed
		text-align: left;
syuilo's avatar
syuilo committed

syuilo's avatar
syuilo committed
		> .avatar {
			$size: 32px;
syuilo's avatar
syuilo committed
			display: inline-block;
syuilo's avatar
syuilo committed
			width: $size;
			height: $size;
syuilo's avatar
syuilo committed
			vertical-align: bottom;
syuilo's avatar
syuilo committed
			margin: 0 8px;
			pointer-events: none;
		}
syuilo's avatar
syuilo committed

syuilo's avatar
syuilo committed
		> .icon {
			margin-right: 8px;
		}
syuilo's avatar
syuilo committed

syuilo's avatar
syuilo committed
		> .title {
			min-width: 0;
			overflow: hidden;
			text-overflow: ellipsis;
			white-space: nowrap;
			line-height: 1.1;

			> .subtitle {
				opacity: 0.6;
				font-size: 0.8em;
				font-weight: normal;
				white-space: nowrap;
				overflow: hidden;
				text-overflow: ellipsis;
syuilo's avatar
syuilo committed
			}
		}
	}
}
</style>