Skip to content
Snippets Groups Projects
post-form.vue 12.9 KiB
Newer Older
syuilo's avatar
wip
syuilo committed
<template>
<div class="mk-post-form"
syuilo's avatar
syuilo committed
	@dragover.stop="onDragover"
syuilo's avatar
wip
syuilo committed
	@dragenter="onDragenter"
	@dragleave="onDragleave"
syuilo's avatar
syuilo committed
	@drop.stop="onDrop"
syuilo's avatar
wip
syuilo committed
>
	<div class="content">
syuilo's avatar
syuilo committed
		<textarea :class="{ with: (files.length != 0 || poll) }"
			ref="text" v-model="text" :disabled="posting"
syuilo's avatar
wip
syuilo committed
			@keydown="onKeydown" @paste="onPaste" :placeholder="placeholder"
syuilo's avatar
syuilo committed
			v-autocomplete="'text'"
syuilo's avatar
wip
syuilo committed
		></textarea>
		<div class="medias" :class="{ with: poll }" v-show="files.length != 0">
syuilo's avatar
syuilo committed
			<x-draggable :list="files" :options="{ animation: 150 }">
				<div v-for="file in files" :key="file.id">
syuilo's avatar
syuilo committed
					<div class="img" :style="{ backgroundImage: `url(${file.url}?thumbnail&size=64)` }" :title="file.name"></div>
syuilo's avatar
syuilo committed
					<img class="remove" @click="detachMedia(file.id)" src="/assets/desktop/remove.png" title="%i18n:@attach-cancel%" alt=""/>
syuilo's avatar
syuilo committed
				</div>
			</x-draggable>
syuilo's avatar
syuilo committed
			<p class="remain">{{ 4 - files.length }}/4</p>
syuilo's avatar
wip
syuilo committed
		</div>
syuilo's avatar
syuilo committed
		<mk-poll-editor v-if="poll" ref="poll" @destroyed="poll = false" @updated="saveDraft()"/>
syuilo's avatar
wip
syuilo committed
	</div>
syuilo's avatar
syuilo committed
	<mk-uploader ref="uploader" @uploaded="attachMedia" @change="onChangeUploadings"/>
syuilo's avatar
syuilo committed
	<button class="upload" title="%i18n:@attach-media-from-local%" @click="chooseFile">%fa:upload%</button>
	<button class="drive" title="%i18n:@attach-media-from-drive%" @click="chooseFileFromDrive">%fa:cloud%</button>
	<button class="kao" title="%i18n:@insert-a-kao%" @click="kao">%fa:R smile%</button>
	<button class="poll" title="%i18n:@create-poll%" @click="poll = true">%fa:chart-pie%</button>
syuilo's avatar
syuilo committed
	<button class="geo" title="位置情報を添付する" @click="geo ? removeGeo() : setGeo()">%fa:map-marker-alt%</button>
syuilo's avatar
syuilo committed
	<p class="text-count" :class="{ over: text.length > 1000 }">{{ '%i18n:!@text-remain%'.replace('{}', 1000 - text.length) }}</p>
syuilo's avatar
syuilo committed
	<button :class="{ posting }" class="submit" :disabled="!canPost" @click="post">
syuilo's avatar
syuilo committed
		{{ posting ? '%i18n:!@posting%' : submitText }}<mk-ellipsis v-if="posting"/>
syuilo's avatar
wip
syuilo committed
	</button>
syuilo's avatar
syuilo committed
	<input ref="file" type="file" accept="image/*" multiple="multiple" tabindex="-1" @change="onChangeFile"/>
syuilo's avatar
wip
syuilo committed
	<div class="dropzone" v-if="draghover"></div>
</div>
</template>
syuilo's avatar
syuilo committed

<script lang="ts">
import Vue from 'vue';
syuilo's avatar
syuilo committed
import * as XDraggable from 'vuedraggable';
syuilo's avatar
syuilo committed
import getKao from '../../../common/scripts/get-kao';
syuilo's avatar
syuilo committed

export default Vue.extend({
syuilo's avatar
syuilo committed
	components: {
		XDraggable
	},
syuilo's avatar
syuilo committed
	props: ['reply', 'renote'],
syuilo's avatar
syuilo committed
	data() {
		return {
			posting: false,
syuilo's avatar
syuilo committed
			text: '',
			files: [],
			uploadings: [],
			poll: false,
syuilo's avatar
syuilo committed
			geo: null,
syuilo's avatar
syuilo committed
			autocomplete: null,
			draghover: false
syuilo's avatar
syuilo committed
		};
	},
	computed: {
syuilo's avatar
syuilo committed
		draftId(): string {
syuilo's avatar
syuilo committed
			return this.renote
				? 'renote:' + this.renote.id
syuilo's avatar
syuilo committed
				: this.reply
					? 'reply:' + this.reply.id
syuilo's avatar
syuilo committed
					: 'note';
syuilo's avatar
syuilo committed
		},
		placeholder(): string {
syuilo's avatar
syuilo committed
			return this.renote
syuilo's avatar
syuilo committed
				? '%i18n:!@quote-placeholder%'
syuilo's avatar
syuilo committed
				: this.reply
syuilo's avatar
syuilo committed
					? '%i18n:!@reply-placeholder%'
					: '%i18n:!@note-placeholder%';
syuilo's avatar
syuilo committed
		},
		submitText(): string {
syuilo's avatar
syuilo committed
			return this.renote
syuilo's avatar
syuilo committed
				? '%i18n:!@renote%'
syuilo's avatar
syuilo committed
				: this.reply
syuilo's avatar
syuilo committed
					? '%i18n:!@reply%'
					: '%i18n:!@note%';
syuilo's avatar
syuilo committed
		},
syuilo's avatar
syuilo committed
		canPost(): boolean {
syuilo's avatar
syuilo committed
			return !this.posting && (this.text.length != 0 || this.files.length != 0 || this.poll || this.renote);
syuilo's avatar
syuilo committed
		}
	},
syuilo's avatar
syuilo committed
	watch: {
		text() {
			this.saveDraft();
		},
		poll() {
			this.saveDraft();
		},
		files() {
			this.saveDraft();
		}
	},
syuilo's avatar
syuilo committed
	mounted() {
syuilo's avatar
syuilo committed
		this.$nextTick(() => {
syuilo's avatar
syuilo committed
			// 書きかけの投稿を復元
			const draft = JSON.parse(localStorage.getItem('drafts') || '{}')[this.draftId];
			if (draft) {
				this.text = draft.data.text;
				this.files = draft.data.files;
				if (draft.data.poll) {
					this.poll = true;
syuilo's avatar
syuilo committed
					this.$nextTick(() => {
						(this.$refs.poll as any).set(draft.data.poll);
					});
syuilo's avatar
syuilo committed
				}
				this.$emit('change-attached-media', this.files);
syuilo's avatar
syuilo committed
			}
		});
	},
	methods: {
		focus() {
			(this.$refs.text as any).focus();
		},
		chooseFile() {
			(this.$refs.file as any).click();
		},
syuilo's avatar
syuilo committed
		chooseFileFromDrive() {
			(this as any).apis.chooseDriveFile({
				multiple: true
			}).then(files => {
syuilo's avatar
syuilo committed
				files.forEach(this.attachMedia);
syuilo's avatar
syuilo committed
			});
syuilo's avatar
syuilo committed
		},
		attachMedia(driveFile) {
			this.files.push(driveFile);
			this.$emit('change-attached-media', this.files);
		},
		detachMedia(id) {
			this.files = this.files.filter(x => x.id != id);
			this.$emit('change-attached-media', this.files);
		},
		onChangeFile() {
			Array.from((this.$refs.file as any).files).forEach(this.upload);
		},
		upload(file) {
			(this.$refs.uploader as any).upload(file);
		},
		onChangeUploadings(uploads) {
			this.$emit('change-uploadings', uploads);
		},
		clear() {
			this.text = '';
			this.files = [];
			this.poll = false;
syuilo's avatar
syuilo committed
			this.$emit('change-attached-media', this.files);
syuilo's avatar
syuilo committed
		},
		onKeydown(e) {
			if ((e.which == 10 || e.which == 13) && (e.ctrlKey || e.metaKey)) this.post();
		},
		onPaste(e) {
			Array.from(e.clipboardData.items).forEach((item: any) => {
				if (item.kind == 'file') {
					this.upload(item.getAsFile());
				}
			});
		},
		onDragover(e) {
syuilo's avatar
syuilo committed
			const isFile = e.dataTransfer.items[0].kind == 'file';
			const isDriveFile = e.dataTransfer.types[0] == 'mk_drive_file';
			if (isFile || isDriveFile) {
				e.preventDefault();
				this.draghover = true;
				e.dataTransfer.dropEffect = e.dataTransfer.effectAllowed == 'all' ? 'copy' : 'move';
			}
syuilo's avatar
syuilo committed
		},
		onDragenter(e) {
			this.draghover = true;
		},
		onDragleave(e) {
			this.draghover = false;
		},
		onDrop(e): void {
			this.draghover = false;

			// ファイルだったら
			if (e.dataTransfer.files.length > 0) {
syuilo's avatar
syuilo committed
				e.preventDefault();
syuilo's avatar
syuilo committed
				Array.from(e.dataTransfer.files).forEach(this.upload);
				return;
			}

syuilo's avatar
syuilo committed
			//#region ドライブのファイル
			const driveFile = e.dataTransfer.getData('mk_drive_file');
			if (driveFile != null && driveFile != '') {
				const file = JSON.parse(driveFile);
				this.files.push(file);
				this.$emit('change-attached-media', this.files);
				e.preventDefault();
syuilo's avatar
syuilo committed
			}
syuilo's avatar
syuilo committed
			//#endregion
syuilo's avatar
syuilo committed
		},
syuilo's avatar
syuilo committed
		setGeo() {
			if (navigator.geolocation == null) {
				alert('お使いの端末は位置情報に対応していません');
				return;
			}

			navigator.geolocation.getCurrentPosition(pos => {
				this.geo = pos.coords;
				this.$emit('geo-attached', this.geo);
			}, err => {
				alert('エラー: ' + err.message);
			}, {
				enableHighAccuracy: true
			});
		},
syuilo's avatar
syuilo committed
		removeGeo() {
			this.geo = null;
			this.$emit('geo-dettached');
		},
syuilo's avatar
syuilo committed
		post() {
			this.posting = true;

syuilo's avatar
syuilo committed
			(this as any).api('notes/create', {
syuilo's avatar
syuilo committed
				text: this.text == '' ? undefined : this.text,
syuilo's avatar
syuilo committed
				mediaIds: this.files.length > 0 ? this.files.map(f => f.id) : undefined,
				replyId: this.reply ? this.reply.id : undefined,
syuilo's avatar
syuilo committed
				renoteId: this.renote ? this.renote.id : undefined,
syuilo's avatar
syuilo committed
				poll: this.poll ? (this.$refs.poll as any).get() : undefined,
syuilo's avatar
syuilo committed
				geo: this.geo ? {
syuilo's avatar
syuilo committed
					coordinates: [this.geo.longitude, this.geo.latitude],
syuilo's avatar
syuilo committed
					altitude: this.geo.altitude,
					accuracy: this.geo.accuracy,
					altitudeAccuracy: this.geo.altitudeAccuracy,
					heading: isNaN(this.geo.heading) ? null : this.geo.heading,
					speed: this.geo.speed,
				} : null
syuilo's avatar
syuilo committed
			}).then(data => {
				this.clear();
				this.deleteDraft();
				this.$emit('posted');
syuilo's avatar
syuilo committed
				(this as any).apis.notify(this.renote
syuilo's avatar
syuilo committed
					? '%i18n:!@reposted%'
syuilo's avatar
syuilo committed
					: this.reply
syuilo's avatar
syuilo committed
						? '%i18n:!@replied%'
						: '%i18n:!@posted%');
syuilo's avatar
syuilo committed
			}).catch(err => {
syuilo's avatar
syuilo committed
				(this as any).apis.notify(this.renote
syuilo's avatar
syuilo committed
					? '%i18n:!@renote-failed%'
syuilo's avatar
syuilo committed
					: this.reply
syuilo's avatar
syuilo committed
						? '%i18n:!@reply-failed%'
						: '%i18n:!@note-failed%');
syuilo's avatar
syuilo committed
			}).then(() => {
				this.posting = false;
			});
		},
		saveDraft() {
			const data = JSON.parse(localStorage.getItem('drafts') || '{}');

			data[this.draftId] = {
syuilo's avatar
syuilo committed
				updatedAt: new Date(),
syuilo's avatar
syuilo committed
				data: {
					text: this.text,
					files: this.files,
					poll: this.poll && this.$refs.poll ? (this.$refs.poll as any).get() : undefined
				}
			}

			localStorage.setItem('drafts', JSON.stringify(data));
		},
		deleteDraft() {
			const data = JSON.parse(localStorage.getItem('drafts') || '{}');

			delete data[this.draftId];

			localStorage.setItem('drafts', JSON.stringify(data));
		},
		kao() {
			this.text += getKao();
syuilo's avatar
syuilo committed
		}
	}
});
</script>

syuilo's avatar
syuilo committed
<style lang="stylus" scoped>
syuilo's avatar
syuilo committed
@import '~const.styl'

syuilo's avatar
syuilo committed
root(isDark)
syuilo's avatar
syuilo committed
	display block
	padding 16px
syuilo's avatar
syuilo committed
	background isDark ? #282C37 : lighten($theme-color, 95%)
syuilo's avatar
syuilo committed

	&:after
		content ""
		display block
		clear both

	> .content

syuilo's avatar
syuilo committed
		textarea
syuilo's avatar
syuilo committed
			display block
			padding 12px
			margin 0
			width 100%
			max-width 100%
			min-width 100%
			min-height calc(16px + 12px + 12px)
			font-size 16px
			color #333
syuilo's avatar
syuilo committed
			background isDark ? #191d23 : #fff
syuilo's avatar
syuilo committed
			outline none
			border solid 1px rgba($theme-color, 0.1)
			border-radius 4px
			transition border-color .3s ease

			&:hover
				border-color rgba($theme-color, 0.2)
				transition border-color .1s ease

				& + *
				& + * + *
					border-color rgba($theme-color, 0.2)
					transition border-color .1s ease

			&:focus
				color $theme-color
				border-color rgba($theme-color, 0.5)
				transition border-color 0s ease

				& + *
				& + * + *
					border-color rgba($theme-color, 0.5)
					transition border-color 0s ease

			&:disabled
				opacity 0.5

			&::-webkit-input-placeholder
				color rgba($theme-color, 0.3)

			&.with
				border-bottom solid 1px rgba($theme-color, 0.1) !important
				border-radius 4px 4px 0 0

		> .medias
			margin 0
			padding 0
syuilo's avatar
syuilo committed
			background isDark ? #181b23 : lighten($theme-color, 98%)
syuilo's avatar
syuilo committed
			border solid 1px rgba($theme-color, 0.1)
			border-top none
			border-radius 0 0 4px 4px
			transition border-color .3s ease

			&.with
				border-bottom solid 1px rgba($theme-color, 0.1) !important
				border-radius 0

			> .remain
				display block
				position absolute
				top 8px
				right 8px
				margin 0
				padding 0
				color rgba($theme-color, 0.4)

syuilo's avatar
syuilo committed
			> div
syuilo's avatar
syuilo committed
				padding 4px

				&:after
					content ""
					display block
					clear both

syuilo's avatar
syuilo committed
				> div
syuilo's avatar
syuilo committed
					float left
					border solid 4px transparent
					cursor move

					&:hover > .remove
						display block

					> .img
						width 64px
						height 64px
						background-size cover
						background-position center center

					> .remove
						display none
						position absolute
						top -6px
						right -6px
						width 16px
						height 16px
						cursor pointer

syuilo's avatar
syuilo committed
		> .mk-poll-editor
syuilo's avatar
syuilo committed
			background isDark ? #181b23 : lighten($theme-color, 98%)
syuilo's avatar
syuilo committed
			border solid 1px rgba($theme-color, 0.1)
			border-top none
			border-radius 0 0 4px 4px
			transition border-color .3s ease

syuilo's avatar
syuilo committed
	> .mk-uploader
syuilo's avatar
syuilo committed
		margin 8px 0 0 0
		padding 8px
		border solid 1px rgba($theme-color, 0.2)
		border-radius 4px

syuilo's avatar
syuilo committed
	input[type='file']
syuilo's avatar
syuilo committed
		display none

	.text-count
		pointer-events none
		display block
		position absolute
		bottom 16px
		right 138px
		margin 0
		line-height 40px
		color rgba($theme-color, 0.5)

		&.over
			color #ec3828

syuilo's avatar
syuilo committed
	.submit
syuilo's avatar
syuilo committed
		display block
		position absolute
		bottom 16px
		right 16px
		cursor pointer
		padding 0
		margin 0
		width 110px
		height 40px
		font-size 1em
		color $theme-color-foreground
		background linear-gradient(to bottom, lighten($theme-color, 25%) 0%, lighten($theme-color, 10%) 100%)
		outline none
		border solid 1px lighten($theme-color, 15%)
		border-radius 4px

		&:not(:disabled)
			font-weight bold

		&:hover:not(:disabled)
			background linear-gradient(to bottom, lighten($theme-color, 8%) 0%, darken($theme-color, 8%) 100%)
			border-color $theme-color

		&:active:not(:disabled)
			background $theme-color
			border-color $theme-color

		&:focus
			&:after
				content ""
				pointer-events none
				position absolute
				top -5px
				right -5px
				bottom -5px
				left -5px
				border 2px solid rgba($theme-color, 0.3)
				border-radius 8px

		&:disabled
			opacity 0.7
			cursor default

		&.wait
			background linear-gradient(
				45deg,
				darken($theme-color, 10%) 25%,
				$theme-color              25%,
				$theme-color              50%,
				darken($theme-color, 10%) 50%,
				darken($theme-color, 10%) 75%,
				$theme-color              75%,
				$theme-color
			)
			background-size 32px 32px
			animation stripe-bg 1.5s linear infinite
			opacity 0.7
			cursor wait

			@keyframes stripe-bg
				from {background-position: 0 0;}
				to   {background-position: -64px 32px;}

syuilo's avatar
syuilo committed
	> .upload
	> .drive
	> .kao
	> .poll
syuilo's avatar
syuilo committed
	> .geo
syuilo's avatar
syuilo committed
		display inline-block
		cursor pointer
		padding 0
		margin 8px 4px 0 0
		width 40px
		height 40px
		font-size 1em
syuilo's avatar
syuilo committed
		color isDark ? $theme-color : rgba($theme-color, 0.5)
syuilo's avatar
syuilo committed
		background transparent
		outline none
		border solid 1px transparent
		border-radius 4px

		&:hover
			background transparent
syuilo's avatar
syuilo committed
			border-color isDark ? rgba($theme-color, 0.5) : rgba($theme-color, 0.3)
syuilo's avatar
syuilo committed

		&:active
			color rgba($theme-color, 0.6)
syuilo's avatar
syuilo committed
			background isDark ? transparent : linear-gradient(to bottom, lighten($theme-color, 80%) 0%, lighten($theme-color, 90%) 100%)
syuilo's avatar
syuilo committed
			border-color rgba($theme-color, 0.5)
			box-shadow 0 2px 4px rgba(0, 0, 0, 0.15) inset

		&:focus
			&:after
				content ""
				pointer-events none
				position absolute
				top -5px
				right -5px
				bottom -5px
				left -5px
				border 2px solid rgba($theme-color, 0.3)
				border-radius 8px

	> .dropzone
		position absolute
		left 0
		top 0
		width 100%
		height 100%
		border dashed 2px rgba($theme-color, 0.5)
		pointer-events none

syuilo's avatar
syuilo committed
.mk-post-form[data-darkmode]
	root(true)

.mk-post-form:not([data-darkmode])
	root(false)

syuilo's avatar
syuilo committed
</style>