mirror of
				https://codeberg.org/yeentown/barkey.git
				synced 2025-10-25 10:44:51 +00:00 
			
		
		
		
	# Conflicts: # .github/workflows/api-misskey-js.yml # .github/workflows/changelog-check.yml # .github/workflows/check-misskey-js-autogen.yml # .github/workflows/get-api-diff.yml # .github/workflows/lint.yml # .github/workflows/locale.yml # .github/workflows/on-release-created.yml # .github/workflows/storybook.yml # .github/workflows/test-backend.yml # .github/workflows/test-federation.yml # .github/workflows/test-frontend.yml # .github/workflows/test-misskey-js.yml # .github/workflows/test-production.yml # .github/workflows/validate-api-json.yml # package.json # packages/backend/package.json # packages/backend/src/server/api/ApiCallService.ts # packages/backend/src/server/api/endpoints/drive/files/create.ts # packages/frontend-shared/js/url.ts # packages/frontend/package.json # packages/frontend/src/components/MkFileCaptionEditWindow.vue # packages/frontend/src/components/MkInfo.vue # packages/frontend/src/components/MkLink.vue # packages/frontend/src/components/MkNote.vue # packages/frontend/src/components/MkNotes.vue # packages/frontend/src/components/MkPageWindow.vue # packages/frontend/src/components/MkReactionsViewer.vue # packages/frontend/src/components/MkTimeline.vue # packages/frontend/src/components/MkUrlPreview.vue # packages/frontend/src/components/MkUserPopup.vue # packages/frontend/src/components/global/MkPageHeader.vue # packages/frontend/src/components/global/MkUrl.vue # packages/frontend/src/components/global/PageWithHeader.vue # packages/frontend/src/pages/about-misskey.vue # packages/frontend/src/pages/announcements.vue # packages/frontend/src/pages/antenna-timeline.vue # packages/frontend/src/pages/channel.vue # packages/frontend/src/pages/instance-info.vue # packages/frontend/src/pages/note.vue # packages/frontend/src/pages/page.vue # packages/frontend/src/pages/role.vue # packages/frontend/src/pages/tag.vue # packages/frontend/src/pages/timeline.vue # packages/frontend/src/pages/user-list-timeline.vue # packages/frontend/src/pages/user/followers.vue # packages/frontend/src/pages/user/following.vue # packages/frontend/src/pages/user/home.vue # packages/frontend/src/pages/user/index.vue # packages/frontend/src/ui/deck.vue # packages/misskey-js/generator/package.json # pnpm-lock.yaml # scripts/changelog-checker/package-lock.json # scripts/changelog-checker/package.json
		
			
				
	
	
		
			216 lines
		
	
	
	
		
			5.5 KiB
		
	
	
	
		
			Vue
		
	
	
	
	
	
			
		
		
	
	
			216 lines
		
	
	
	
		
			5.5 KiB
		
	
	
	
		
			Vue
		
	
	
	
	
	
| <!--
 | |
| SPDX-FileCopyrightText: syuilo and misskey-project
 | |
| SPDX-License-Identifier: AGPL-3.0-only
 | |
| -->
 | |
| 
 | |
| <template>
 | |
| <PageWithHeader :actions="headerActions">
 | |
| 	<div class="_spacer" style="--MI_SPACER-w: 800px;">
 | |
| 		<div v-if="clip" class="_gaps">
 | |
| 			<div class="_panel">
 | |
| 				<div class="_gaps_s" :class="$style.description">
 | |
| 					<div v-if="clip.description">
 | |
| 						<Mfm :text="clip.description" :isBlock="true" :isNote="false"/>
 | |
| 					</div>
 | |
| 					<div v-else>({{ i18n.ts.noDescription }})</div>
 | |
| 					<div>
 | |
| 						<MkButton v-if="favorited" v-tooltip="i18n.ts.unfavorite" asLike rounded primary @click="unfavorite()"><i class="ti ti-heart"></i><span v-if="clip.favoritedCount > 0" style="margin-left: 6px;">{{ clip.favoritedCount }}</span></MkButton>
 | |
| 						<MkButton v-else v-tooltip="i18n.ts.favorite" asLike rounded @click="favorite()"><i class="ti ti-heart"></i><span v-if="clip.favoritedCount > 0" style="margin-left: 6px;">{{ clip.favoritedCount }}</span></MkButton>
 | |
| 					</div>
 | |
| 				</div>
 | |
| 				<div :class="$style.user">
 | |
| 					<MkAvatar :user="clip.user" :class="$style.avatar" indicator link preview/> <MkUserName :user="clip.user" :nowrap="false"/>
 | |
| 				</div>
 | |
| 			</div>
 | |
| 
 | |
| 			<MkNotes :pagination="pagination" :detail="true"/>
 | |
| 		</div>
 | |
| 	</div>
 | |
| </PageWithHeader>
 | |
| </template>
 | |
| 
 | |
| <script lang="ts" setup>
 | |
| import { computed, watch, provide, ref } from 'vue';
 | |
| import * as Misskey from 'misskey-js';
 | |
| import { url } from '@@/js/config.js';
 | |
| import type { MenuItem } from '@/types/menu.js';
 | |
| import MkNotes from '@/components/MkNotes.vue';
 | |
| import { $i } from '@/i.js';
 | |
| import { i18n } from '@/i18n.js';
 | |
| import * as os from '@/os.js';
 | |
| import { misskeyApi } from '@/utility/misskey-api.js';
 | |
| import { definePage } from '@/page.js';
 | |
| import MkButton from '@/components/MkButton.vue';
 | |
| import { clipsCache } from '@/cache.js';
 | |
| import { isSupportShare } from '@/utility/navigator.js';
 | |
| import { copyToClipboard } from '@/utility/copy-to-clipboard.js';
 | |
| import { genEmbedCode } from '@/utility/get-embed-code.js';
 | |
| import { assertServerContext, serverContext } from '@/server-context.js';
 | |
| 
 | |
| // contextは非ログイン状態の情報しかないためログイン時は利用できない
 | |
| const CTX_CLIP = !$i && assertServerContext(serverContext, 'clip') ? serverContext.clip : null;
 | |
| 
 | |
| const props = defineProps<{
 | |
| 	clipId: string,
 | |
| }>();
 | |
| 
 | |
| const clip = ref<Misskey.entities.Clip | null>(CTX_CLIP);
 | |
| const favorited = ref(false);
 | |
| const pagination = {
 | |
| 	endpoint: 'clips/notes' as const,
 | |
| 	limit: 10,
 | |
| 	params: computed(() => ({
 | |
| 		clipId: props.clipId,
 | |
| 	})),
 | |
| };
 | |
| 
 | |
| const isOwned = computed<boolean | null>(() => $i && clip.value && ($i.id === clip.value.userId));
 | |
| 
 | |
| watch(() => props.clipId, async () => {
 | |
| 	if (CTX_CLIP && CTX_CLIP.id === props.clipId) {
 | |
| 		clip.value = CTX_CLIP;
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	clip.value = await misskeyApi('clips/show', {
 | |
| 		clipId: props.clipId,
 | |
| 	});
 | |
| 	favorited.value = clip.value.isFavorited;
 | |
| }, {
 | |
| 	immediate: true,
 | |
| });
 | |
| 
 | |
| provide('currentClip', clip);
 | |
| 
 | |
| function favorite() {
 | |
| 	os.apiWithDialog('clips/favorite', {
 | |
| 		clipId: props.clipId,
 | |
| 	}).then(() => {
 | |
| 		favorited.value = true;
 | |
| 	});
 | |
| }
 | |
| 
 | |
| async function unfavorite() {
 | |
| 	const confirm = await os.confirm({
 | |
| 		type: 'warning',
 | |
| 		text: i18n.ts.unfavoriteConfirm,
 | |
| 	});
 | |
| 	if (confirm.canceled) return;
 | |
| 	os.apiWithDialog('clips/unfavorite', {
 | |
| 		clipId: props.clipId,
 | |
| 	}).then(() => {
 | |
| 		favorited.value = false;
 | |
| 	});
 | |
| }
 | |
| 
 | |
| const headerActions = computed(() => clip.value && isOwned.value ? [{
 | |
| 	icon: 'ti ti-pencil',
 | |
| 	text: i18n.ts.edit,
 | |
| 	handler: async (): Promise<void> => {
 | |
| 		const { canceled, result } = await os.form(clip.value.name, {
 | |
| 			name: {
 | |
| 				type: 'string',
 | |
| 				label: i18n.ts.name,
 | |
| 				default: clip.value.name,
 | |
| 			},
 | |
| 			description: {
 | |
| 				type: 'string',
 | |
| 				required: false,
 | |
| 				multiline: true,
 | |
| 				treatAsMfm: true,
 | |
| 				label: i18n.ts.description,
 | |
| 				default: clip.value.description,
 | |
| 			},
 | |
| 			isPublic: {
 | |
| 				type: 'boolean',
 | |
| 				label: i18n.ts.public,
 | |
| 				default: clip.value.isPublic,
 | |
| 			},
 | |
| 		});
 | |
| 		if (canceled) return;
 | |
| 
 | |
| 		os.apiWithDialog('clips/update', {
 | |
| 			clipId: clip.value.id,
 | |
| 			...result,
 | |
| 		});
 | |
| 
 | |
| 		clipsCache.delete();
 | |
| 	},
 | |
| }, ...(clip.value.isPublic ? [{
 | |
| 	icon: 'ti ti-share',
 | |
| 	text: i18n.ts.share,
 | |
| 	handler: (ev: MouseEvent): void => {
 | |
| 		const menuItems: MenuItem[] = [];
 | |
| 
 | |
| 		menuItems.push({
 | |
| 			icon: 'ti ti-link',
 | |
| 			text: i18n.ts.copyUrl,
 | |
| 			action: () => {
 | |
| 				copyToClipboard(`${url}/clips/${clip.value!.id}`);
 | |
| 			},
 | |
| 		}, {
 | |
| 			icon: 'ti ti-code',
 | |
| 			text: i18n.ts.embed,
 | |
| 			action: () => {
 | |
| 				genEmbedCode('clips', clip.value!.id);
 | |
| 			},
 | |
| 		});
 | |
| 
 | |
| 		if (isSupportShare()) {
 | |
| 			menuItems.push({
 | |
| 				icon: 'ti ti-share',
 | |
| 				text: i18n.ts.share,
 | |
| 				action: async () => {
 | |
| 					navigator.share({
 | |
| 						title: clip.value!.name,
 | |
| 						text: clip.value!.description ?? '',
 | |
| 						url: `${url}/clips/${clip.value!.id}`,
 | |
| 					});
 | |
| 				},
 | |
| 			});
 | |
| 		}
 | |
| 
 | |
| 		os.popupMenu(menuItems, ev.currentTarget ?? ev.target);
 | |
| 	},
 | |
| }] : []), {
 | |
| 	icon: 'ti ti-trash',
 | |
| 	text: i18n.ts.delete,
 | |
| 	danger: true,
 | |
| 	handler: async (): Promise<void> => {
 | |
| 		const { canceled } = await os.confirm({
 | |
| 			type: 'warning',
 | |
| 			text: i18n.tsx.deleteAreYouSure({ x: clip.value.name }),
 | |
| 		});
 | |
| 		if (canceled) return;
 | |
| 
 | |
| 		await os.apiWithDialog('clips/delete', {
 | |
| 			clipId: clip.value.id,
 | |
| 		});
 | |
| 
 | |
| 		clipsCache.delete();
 | |
| 	},
 | |
| }] : null);
 | |
| 
 | |
| definePage(() => ({
 | |
| 	title: clip.value ? clip.value.name : i18n.ts.clip,
 | |
| 	icon: 'ti ti-paperclip',
 | |
| }));
 | |
| </script>
 | |
| 
 | |
| <style lang="scss" module>
 | |
| .description {
 | |
| 	padding: 16px;
 | |
| }
 | |
| 
 | |
| .user {
 | |
| 	--height: 32px;
 | |
| 	padding: 16px;
 | |
| 	border-top: solid 0.5px var(--MI_THEME-divider);
 | |
| 	line-height: var(--height);
 | |
| }
 | |
| 
 | |
| .avatar {
 | |
| 	width: var(--height);
 | |
| 	height: var(--height);
 | |
| }
 | |
| </style>
 |