lint fix
This commit is contained in:
		
							parent
							
								
									4abecb2bc1
								
							
						
					
					
						commit
						57331536f2
					
				|  | @ -16,10 +16,8 @@ import type { EmojisRepository, MiRole, MiUser } from '@/models/_.js'; | |||
| import { bindThis } from '@/decorators.js'; | ||||
| import { MemoryKVCache, RedisSingleCache } from '@/misc/cache.js'; | ||||
| import { UtilityService } from '@/core/UtilityService.js'; | ||||
| import { query } from '@/misc/prelude/url.js'; | ||||
| import type { Serialized } from '@/types.js'; | ||||
| import { ModerationLogService } from '@/core/ModerationLogService.js'; | ||||
| 
 | ||||
| const parseEmojiStrRegexp = /^(\w+)(?:@([\w.-]+))?$/; | ||||
| 
 | ||||
| @Injectable() | ||||
|  | @ -31,12 +29,6 @@ export class CustomEmojiService implements OnApplicationShutdown { | |||
| 		@Inject(DI.redis) | ||||
| 		private redisClient: Redis.Redis, | ||||
| 
 | ||||
| 		@Inject(DI.config) | ||||
| 		private config: Config, | ||||
| 
 | ||||
| 		@Inject(DI.db) | ||||
| 		private db: DataSource, | ||||
| 
 | ||||
| 		@Inject(DI.emojisRepository) | ||||
| 		private emojisRepository: EmojisRepository, | ||||
| 
 | ||||
|  | @ -119,7 +111,7 @@ export class CustomEmojiService implements OnApplicationShutdown { | |||
| 		license?: string | null; | ||||
| 		isSensitive?: boolean; | ||||
| 		localOnly?: boolean; | ||||
|         draft: boolean; | ||||
| 		draft: boolean; | ||||
| 		roleIdsThatCanBeUsedThisEmojiAsReaction?: MiRole['id'][]; | ||||
| 	}, moderator?: MiUser): Promise<void> { | ||||
| 		const emoji = await this.emojisRepository.findOneByOrFail({ id: id }); | ||||
|  | @ -134,7 +126,7 @@ export class CustomEmojiService implements OnApplicationShutdown { | |||
| 			license: data.license, | ||||
| 			isSensitive: data.isSensitive, | ||||
| 			localOnly: data.localOnly, | ||||
|             draft: data.draft, | ||||
| 			draft: data.draft, | ||||
| 			roleIdsThatCanBeUsedThisEmojiAsReaction: data.roleIdsThatCanBeUsedThisEmojiAsReaction ?? undefined, | ||||
| 		}); | ||||
| 
 | ||||
|  |  | |||
|  | @ -19,7 +19,13 @@ import { FederatedInstanceService } from '@/core/FederatedInstanceService.js'; | |||
| import { WebhookService } from '@/core/WebhookService.js'; | ||||
| import { NotificationService } from '@/core/NotificationService.js'; | ||||
| import { DI } from '@/di-symbols.js'; | ||||
| import type { FollowingsRepository, FollowRequestsRepository, InstancesRepository, UserProfilesRepository, UsersRepository } from '@/models/_.js'; | ||||
| import type { | ||||
| 	FollowingsRepository, | ||||
| 	FollowRequestsRepository, | ||||
| 	InstancesRepository, | ||||
| 	UserProfilesRepository, | ||||
| 	UsersRepository, | ||||
| } from '@/models/_.js'; | ||||
| import { UserEntityService } from '@/core/entities/UserEntityService.js'; | ||||
| import { ApRendererService } from '@/core/activitypub/ApRendererService.js'; | ||||
| import { bindThis } from '@/decorators.js'; | ||||
|  | @ -51,25 +57,18 @@ export class UserFollowingService implements OnModuleInit { | |||
| 
 | ||||
| 	constructor( | ||||
| 		private moduleRef: ModuleRef, | ||||
| 
 | ||||
| 		@Inject(DI.config) | ||||
| 		private config: Config, | ||||
| 
 | ||||
| 		@Inject(DI.usersRepository) | ||||
| 		private usersRepository: UsersRepository, | ||||
| 
 | ||||
| 		@Inject(DI.userProfilesRepository) | ||||
| 		private userProfilesRepository: UserProfilesRepository, | ||||
| 
 | ||||
| 		@Inject(DI.followingsRepository) | ||||
| 		private followingsRepository: FollowingsRepository, | ||||
| 
 | ||||
| 		@Inject(DI.followRequestsRepository) | ||||
| 		private followRequestsRepository: FollowRequestsRepository, | ||||
| 
 | ||||
| 		@Inject(DI.instancesRepository) | ||||
| 		private instancesRepository: InstancesRepository, | ||||
| 
 | ||||
| 		private cacheService: CacheService, | ||||
| 		private userEntityService: UserEntityService, | ||||
| 		private idService: IdService, | ||||
|  | @ -91,7 +90,9 @@ export class UserFollowingService implements OnModuleInit { | |||
| 	} | ||||
| 
 | ||||
| 	@bindThis | ||||
| 	public async follow(_follower: { id: MiUser['id'] }, _followee: { id: MiUser['id'] }, requestId?: string, silent = false): Promise<void> { | ||||
| 	public async follow(_follower: { id: MiUser['id'] }, _followee: { | ||||
| 		id: MiUser['id'] | ||||
| 	}, requestId?: string, silent = false): Promise<void> { | ||||
| 		const [follower, followee] = await Promise.all([ | ||||
| 			this.usersRepository.findOneByOrFail({ id: _follower.id }), | ||||
| 			this.usersRepository.findOneByOrFail({ id: _followee.id }), | ||||
|  | @ -184,10 +185,18 @@ export class UserFollowingService implements OnModuleInit { | |||
| 	@bindThis | ||||
| 	private async insertFollowingDoc( | ||||
| 		followee: { | ||||
| 			id: MiUser['id']; host: MiUser['host']; uri: MiUser['host']; inbox: MiUser['inbox']; sharedInbox: MiUser['sharedInbox'] | ||||
| 			id: MiUser['id']; | ||||
| 			host: MiUser['host']; | ||||
| 			uri: MiUser['host']; | ||||
| 			inbox: MiUser['inbox']; | ||||
| 			sharedInbox: MiUser['sharedInbox'] | ||||
| 		}, | ||||
| 		follower: { | ||||
| 			id: MiUser['id']; host: MiUser['host']; uri: MiUser['host']; inbox: MiUser['inbox']; sharedInbox: MiUser['sharedInbox'] | ||||
| 			id: MiUser['id']; | ||||
| 			host: MiUser['host']; | ||||
| 			uri: MiUser['host']; | ||||
| 			inbox: MiUser['inbox']; | ||||
| 			sharedInbox: MiUser['sharedInbox'] | ||||
| 		}, | ||||
| 		silent = false, | ||||
| 	): Promise<void> { | ||||
|  | @ -233,8 +242,7 @@ export class UserFollowingService implements OnModuleInit { | |||
| 			}); | ||||
| 
 | ||||
| 			// 通知を作成
 | ||||
| 			this.notificationService.createNotification(follower.id, 'followRequestAccepted', { | ||||
| 			}, followee.id); | ||||
| 			this.notificationService.createNotification(follower.id, 'followRequestAccepted', {}, followee.id); | ||||
| 		} | ||||
| 
 | ||||
| 		if (alreadyFollowed) return; | ||||
|  | @ -306,18 +314,25 @@ export class UserFollowingService implements OnModuleInit { | |||
| 			}); | ||||
| 
 | ||||
| 			// 通知を作成
 | ||||
| 			this.notificationService.createNotification(followee.id, 'follow', { | ||||
| 			}, follower.id); | ||||
| 			this.notificationService.createNotification(followee.id, 'follow', {}, follower.id); | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	@bindThis | ||||
| 	public async unfollow( | ||||
| 		follower: { | ||||
| 			id: MiUser['id']; host: MiUser['host']; uri: MiUser['host']; inbox: MiUser['inbox']; sharedInbox: MiUser['sharedInbox']; | ||||
| 			id: MiUser['id']; | ||||
| 			host: MiUser['host']; | ||||
| 			uri: MiUser['host']; | ||||
| 			inbox: MiUser['inbox']; | ||||
| 			sharedInbox: MiUser['sharedInbox']; | ||||
| 		}, | ||||
| 		followee: { | ||||
| 			id: MiUser['id']; host: MiUser['host']; uri: MiUser['host']; inbox: MiUser['inbox']; sharedInbox: MiUser['sharedInbox']; | ||||
| 			id: MiUser['id']; | ||||
| 			host: MiUser['host']; | ||||
| 			uri: MiUser['host']; | ||||
| 			inbox: MiUser['inbox']; | ||||
| 			sharedInbox: MiUser['sharedInbox']; | ||||
| 		}, | ||||
| 		silent = false, | ||||
| 	): Promise<void> { | ||||
|  | @ -446,10 +461,18 @@ export class UserFollowingService implements OnModuleInit { | |||
| 	@bindThis | ||||
| 	public async createFollowRequest( | ||||
| 		follower: { | ||||
| 			id: MiUser['id']; host: MiUser['host']; uri: MiUser['host']; inbox: MiUser['inbox']; sharedInbox: MiUser['sharedInbox']; | ||||
| 			id: MiUser['id']; | ||||
| 			host: MiUser['host']; | ||||
| 			uri: MiUser['host']; | ||||
| 			inbox: MiUser['inbox']; | ||||
| 			sharedInbox: MiUser['sharedInbox']; | ||||
| 		}, | ||||
| 		followee: { | ||||
| 			id: MiUser['id']; host: MiUser['host']; uri: MiUser['host']; inbox: MiUser['inbox']; sharedInbox: MiUser['sharedInbox']; | ||||
| 			id: MiUser['id']; | ||||
| 			host: MiUser['host']; | ||||
| 			uri: MiUser['host']; | ||||
| 			inbox: MiUser['inbox']; | ||||
| 			sharedInbox: MiUser['sharedInbox']; | ||||
| 		}, | ||||
| 		requestId?: string, | ||||
| 	): Promise<void> { | ||||
|  | @ -541,7 +564,11 @@ export class UserFollowingService implements OnModuleInit { | |||
| 	@bindThis | ||||
| 	public async acceptFollowRequest( | ||||
| 		followee: { | ||||
| 			id: MiUser['id']; host: MiUser['host']; uri: MiUser['host']; inbox: MiUser['inbox']; sharedInbox: MiUser['sharedInbox']; | ||||
| 			id: MiUser['id']; | ||||
| 			host: MiUser['host']; | ||||
| 			uri: MiUser['host']; | ||||
| 			inbox: MiUser['inbox']; | ||||
| 			sharedInbox: MiUser['sharedInbox']; | ||||
| 		}, | ||||
| 		follower: MiUser, | ||||
| 	): Promise<void> { | ||||
|  | @ -569,7 +596,11 @@ export class UserFollowingService implements OnModuleInit { | |||
| 	@bindThis | ||||
| 	public async acceptAllFollowRequests( | ||||
| 		user: { | ||||
| 			id: MiUser['id']; host: MiUser['host']; uri: MiUser['host']; inbox: MiUser['inbox']; sharedInbox: MiUser['sharedInbox']; | ||||
| 			id: MiUser['id']; | ||||
| 			host: MiUser['host']; | ||||
| 			uri: MiUser['host']; | ||||
| 			inbox: MiUser['inbox']; | ||||
| 			sharedInbox: MiUser['sharedInbox']; | ||||
| 		}, | ||||
| 	): Promise<void> { | ||||
| 		const requests = await this.followRequestsRepository.findBy({ | ||||
|  |  | |||
|  | @ -1,6 +1,6 @@ | |||
| import { Inject, Injectable } from '@nestjs/common'; | ||||
|  import { Endpoint } from '@/server/api/endpoint-base.js'; | ||||
| import type { DriveFilesRepository } from '@/models/index.js'; | ||||
| import { Endpoint } from '@/server/api/endpoint-base.js'; | ||||
| import type { DriveFilesRepository } from '@/models/_.js'; | ||||
| import { DI } from '@/di-symbols.js'; | ||||
| import { CustomEmojiService } from '@/core/CustomEmojiService.js'; | ||||
| import { ModerationLogService } from '@/core/ModerationLogService.js'; | ||||
|  |  | |||
|  | @ -41,17 +41,21 @@ export const paramDef = { | |||
| 			nullable: true, | ||||
| 			description: 'Use `null` to reset the category.', | ||||
| 		}, | ||||
| 		aliases: { type: 'array', items: { | ||||
| 			type: 'string', | ||||
| 		} }, | ||||
| 		aliases: { | ||||
| 			type: 'array', items: { | ||||
| 				type: 'string', | ||||
| 			}, | ||||
| 		}, | ||||
| 		license: { type: 'string', nullable: true }, | ||||
| 		isSensitive: { type: 'boolean' }, | ||||
| 		localOnly: { type: 'boolean' }, | ||||
| 		roleIdsThatCanBeUsedThisEmojiAsReaction: { type: 'array', items: { | ||||
| 			type: 'string', | ||||
| 		} }, | ||||
| 		roleIdsThatCanBeUsedThisEmojiAsReaction: { | ||||
| 			type: 'array', items: { | ||||
| 				type: 'string', | ||||
| 			}, | ||||
| 		}, | ||||
| 	}, | ||||
| 	required: ['name','fileId', 'draft'], | ||||
| 	required: ['name', 'fileId', 'draft'], | ||||
| } as const; | ||||
| 
 | ||||
| // TODO: ロジックをサービスに切り出す
 | ||||
|  | @ -61,9 +65,7 @@ export default class extends Endpoint<typeof meta, typeof paramDef> { // eslint- | |||
| 	constructor( | ||||
| 		@Inject(DI.driveFilesRepository) | ||||
| 		private driveFilesRepository: DriveFilesRepository, | ||||
| 
 | ||||
| 		private customEmojiService: CustomEmojiService, | ||||
| 
 | ||||
| 		private emojiEntityService: EmojiEntityService, | ||||
| 	) { | ||||
| 		super(meta, paramDef, async (ps, me) => { | ||||
|  | @ -82,7 +84,7 @@ export default class extends Endpoint<typeof meta, typeof paramDef> { // eslint- | |||
| 				license: ps.license ?? null, | ||||
| 				isSensitive: ps.isSensitive ?? false, | ||||
| 				localOnly: ps.localOnly ?? false, | ||||
|                 draft: false, | ||||
| 				draft: false, | ||||
| 				roleIdsThatCanBeUsedThisEmojiAsReaction: ps.roleIdsThatCanBeUsedThisEmojiAsReaction ?? [], | ||||
| 			}, me); | ||||
| 
 | ||||
|  |  | |||
|  | @ -57,7 +57,7 @@ export const paramDef = { | |||
| 		} }, | ||||
| 		draft: { type: 'boolean' }, | ||||
| 	}, | ||||
| 	required: ['id', 'name', 'aliases'], | ||||
| 	required: ['id', 'name', 'draft', 'aliases'], | ||||
| } as const; | ||||
| 
 | ||||
| @Injectable() | ||||
|  | @ -94,7 +94,7 @@ export default class extends Endpoint<typeof meta, typeof paramDef> { // eslint- | |||
| 				isSensitive: ps.isSensitive, | ||||
| 				localOnly: ps.localOnly, | ||||
| 				roleIdsThatCanBeUsedThisEmojiAsReaction: ps.roleIdsThatCanBeUsedThisEmojiAsReaction, | ||||
|             draft: ps.draft, | ||||
| 				draft: ps.draft, | ||||
| 			}, me); | ||||
| 		}); | ||||
| 	} | ||||
|  |  | |||
|  | @ -7,7 +7,7 @@ SPDX-License-Identifier: AGPL-3.0-only | |||
| <MkModalWindow | ||||
| 	ref="dialog" | ||||
| 	:width="400" | ||||
|     :with-ok-button="false " | ||||
| 	:withOkButton="false " | ||||
| 	@close="dialog.close()" | ||||
| 	@closed="$emit('closed')" | ||||
| > | ||||
|  | @ -51,7 +51,7 @@ SPDX-License-Identifier: AGPL-3.0-only | |||
| 					<template #label>{{ i18n.ts.rolesThatCanBeUsedThisEmojiAsReaction }}</template> | ||||
| 					<template #suffix>{{ rolesThatCanBeUsedThisEmojiAsReaction.length === 0 ? i18n.ts.all : rolesThatCanBeUsedThisEmojiAsReaction.length }}</template> | ||||
| 
 | ||||
| 					<div class="_gaps" v-if="!isRequest"> | ||||
| 					<div v-if="!isRequest" class="_gaps"> | ||||
| 						<MkButton rounded @click="addRole"><i class="ti ti-plus"></i> {{ i18n.ts.add }}</MkButton> | ||||
| 
 | ||||
| 						<div v-for="role in rolesThatCanBeUsedThisEmojiAsReaction" :key="role.id" :class="$style.roleItem"> | ||||
|  | @ -73,19 +73,19 @@ SPDX-License-Identifier: AGPL-3.0-only | |||
| 		</MkSpacer> | ||||
| 		<div :class="$style.footer"> | ||||
| 			<div :class="$style.footerButtons"> | ||||
|                 <MkButton v-if="!isRequest" danger rounded style="margin: 0 auto;" @click="del()"><i class="ti ti-trash"></i> {{ i18n.ts.delete }}</MkButton> | ||||
| 				<MkButton v-if="!isRequest" danger rounded style="margin: 0 auto;" @click="del()"><i class="ti ti-trash"></i> {{ i18n.ts.delete }}</MkButton> | ||||
| 				<MkButton v-if="validation" primary rounded style="margin: 0 auto;" @click="done"><i class="ti ti-check"></i> {{ props.emoji ? i18n.ts.update : i18n.ts.create }}</MkButton> | ||||
| 				<MkButton v-else rounded style="margin: 0 auto;"><i class="ti ti-check"></i> {{ props.emoji ? i18n.ts.update : i18n.ts.create }}</MkButton> | ||||
| 			</div> | ||||
| 
 | ||||
|         </div> | ||||
| 		</div> | ||||
| 	</div> | ||||
| </MkModalWindow> | ||||
| </template> | ||||
| 
 | ||||
| <script lang="ts" setup> | ||||
| import { computed, watch,ref } from 'vue'; | ||||
| import { computed, watch } from 'vue'; | ||||
| import * as Misskey from 'misskey-js'; | ||||
| import { DriveFile } from 'misskey-js/built/entities.js'; | ||||
| import MkModalWindow from '@/components/MkModalWindow.vue'; | ||||
| import MkButton from '@/components/MkButton.vue'; | ||||
| import MkInput from '@/components/MkInput.vue'; | ||||
|  | @ -114,7 +114,7 @@ let localOnly = $ref(props.emoji ? props.emoji.localOnly : false); | |||
| let roleIdsThatCanBeUsedThisEmojiAsReaction = $ref(props.emoji ? props.emoji.roleIdsThatCanBeUsedThisEmojiAsReaction : []); | ||||
| let rolesThatCanBeUsedThisEmojiAsReaction = $ref([]); | ||||
| let file = $ref<Misskey.entities.DriveFile>(); | ||||
| let chooseFile: MkDriveFile|null = $ref(null); | ||||
| let chooseFile: DriveFile|null = $ref(null); | ||||
| let draft = $ref(props.emoji ? props.emoji.draft : false); | ||||
| let isRequest = $ref(props.isRequest); | ||||
| 
 | ||||
|  | @ -124,8 +124,8 @@ watch($$(roleIdsThatCanBeUsedThisEmojiAsReaction), async () => { | |||
| 
 | ||||
| const imgUrl = computed(() => file ? file.url : props.emoji ? `/emoji/${props.emoji.name}.webp` : null); | ||||
| const validation = computed(() => { | ||||
|     return name.match(/^[a-zA-Z0-9_]+$/) && imgUrl.value != null; | ||||
| }) | ||||
| 	return name.match(/^[a-zA-Z0-9_]+$/) && imgUrl.value != null; | ||||
| }); | ||||
| const emit = defineEmits<{ | ||||
| 	(ev: 'done', v: { deleted?: boolean; updated?: any; created?: any }): void, | ||||
| 	(ev: 'closed'): void | ||||
|  | @ -164,27 +164,27 @@ async function add() { | |||
| 	dialog.close(); | ||||
| } | ||||
| async function changeImage(ev) { | ||||
|     file = await selectFile(ev.currentTarget ?? ev.target, null); | ||||
|     const candidate = file.name.replace(/\.(.+)$/, ''); | ||||
|     if (candidate.match(/^[a-z0-9_]+$/)) { | ||||
|         name = candidate; | ||||
|     } | ||||
| 	file = await selectFile(ev.currentTarget ?? ev.target, null); | ||||
| 	const candidate = file.name.replace(/\.(.+)$/, ''); | ||||
| 	if (candidate.match(/^[a-z0-9_]+$/)) { | ||||
| 		name = candidate; | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| async function addRole() { | ||||
|     const roles = await os.api('admin/roles/list'); | ||||
|     const currentRoleIds = rolesThatCanBeUsedThisEmojiAsReaction.map(x => x.id); | ||||
| 	const roles = await os.api('admin/roles/list'); | ||||
| 	const currentRoleIds = rolesThatCanBeUsedThisEmojiAsReaction.map(x => x.id); | ||||
| 
 | ||||
|     const { canceled, result: role } = await os.select({ | ||||
|         items: roles.filter(r => r.isPublic).filter(r => !currentRoleIds.includes(r.id)).map(r => ({ text: r.name, value: r })), | ||||
|     }); | ||||
|     if (canceled) return; | ||||
| 	const { canceled, result: role } = await os.select({ | ||||
| 		items: roles.filter(r => r.isPublic).filter(r => !currentRoleIds.includes(r.id)).map(r => ({ text: r.name, value: r })), | ||||
| 	}); | ||||
| 	if (canceled) return; | ||||
| 
 | ||||
|     rolesThatCanBeUsedThisEmojiAsReaction.push(role); | ||||
| 	rolesThatCanBeUsedThisEmojiAsReaction.push(role); | ||||
| } | ||||
| 
 | ||||
| async function removeRole(role, ev) { | ||||
|     rolesThatCanBeUsedThisEmojiAsReaction = rolesThatCanBeUsedThisEmojiAsReaction.filter(x => x.id !== role.id); | ||||
| 	rolesThatCanBeUsedThisEmojiAsReaction = rolesThatCanBeUsedThisEmojiAsReaction.filter(x => x.id !== role.id); | ||||
| } | ||||
| async function update() { | ||||
| 	await os.apiWithDialog('admin/emoji/update', { | ||||
|  | @ -208,8 +208,7 @@ async function update() { | |||
| 		}, | ||||
| 	}); | ||||
| 
 | ||||
| 		dialog.close(); | ||||
| 
 | ||||
| 	dialog.close(); | ||||
| } | ||||
| async function done() { | ||||
| 	const params = { | ||||
|  | @ -223,35 +222,35 @@ async function done() { | |||
| 		roleIdsThatCanBeUsedThisEmojiAsReaction: rolesThatCanBeUsedThisEmojiAsReaction.map(x => x.id), | ||||
| 	}; | ||||
| 
 | ||||
|     if (file) { | ||||
|         params.fileId = file.id; | ||||
|     } | ||||
|     console.log(props.emoji) | ||||
|     if (props.emoji) { | ||||
|         await os.apiWithDialog('admin/emoji/update', { | ||||
|             id: props.emoji.id, | ||||
|             ...params, | ||||
|         }); | ||||
| 	if (file) { | ||||
| 		params.fileId = file.id; | ||||
| 	} | ||||
| 	console.log(props.emoji); | ||||
| 	if (props.emoji) { | ||||
| 		await os.apiWithDialog('admin/emoji/update', { | ||||
| 			id: props.emoji.id, | ||||
| 			...params, | ||||
| 		}); | ||||
| 
 | ||||
|         emit('done', { | ||||
|             updated: { | ||||
|                 id: props.emoji.id, | ||||
|                 ...params, | ||||
|             }, | ||||
|         }); | ||||
| 		emit('done', { | ||||
| 			updated: { | ||||
| 				id: props.emoji.id, | ||||
| 				...params, | ||||
| 			}, | ||||
| 		}); | ||||
| 
 | ||||
|         dialog.close(); | ||||
|     } else { | ||||
|         const created = isRequest | ||||
| 		dialog.close(); | ||||
| 	} else { | ||||
| 		const created = isRequest | ||||
| 		 ? await os.apiWithDialog('admin/emoji/add-draft', params) | ||||
| 		 : await os.apiWithDialog('admin/emoji/add', params); | ||||
| 
 | ||||
|         emit('done', { | ||||
|             created: created, | ||||
|         }); | ||||
| 		emit('done', { | ||||
| 			created: created, | ||||
| 		}); | ||||
| 
 | ||||
|         dialog.close(); | ||||
|     } | ||||
| 		dialog.close(); | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| function chooseFileFrom(ev) { | ||||
|  |  | |||
|  | @ -25,7 +25,7 @@ const props = defineProps<{ | |||
| 
 | ||||
| const customEmojiName = computed(() => (props.name[0] === ':' ? props.name.substring(1, props.name.length - 1) : props.name).replace('@.', '')); | ||||
| const isLocal = computed(() => !props.host && (customEmojiName.value.endsWith('@.') || !customEmojiName.value.includes('@'))); | ||||
| const isDraft = computed(() => customEmojisNameMap.value.get(customEmojiName.value)?.draft ?? false); | ||||
| const isDraft = computed(() => customEmojisMap.get(customEmojiName.value)?.draft ?? false); | ||||
| 
 | ||||
| const rawUrl = computed(() => { | ||||
| 	if (props.url) { | ||||
|  |  | |||
|  | @ -3,7 +3,7 @@ | |||
|  * SPDX-License-Identifier: AGPL-3.0-only | ||||
|  */ | ||||
| 
 | ||||
| import { shallowRef, computed, markRaw, triggerRef,watch } from 'vue'; | ||||
| import { shallowRef, computed, markRaw, triggerRef, watch } from 'vue'; | ||||
| import * as Misskey from 'misskey-js'; | ||||
| import { api, apiGet } from '@/os.js'; | ||||
| import { useStream } from '@/stream.js'; | ||||
|  |  | |||
|  | @ -12,7 +12,6 @@ import * as os from '@/os.js'; | |||
| import { i18n } from '@/i18n.js'; | ||||
| import { ui } from '@/config.js'; | ||||
| import { unisonReload } from '@/scripts/unison-reload.js'; | ||||
| import {defaultStore} from "@/store.js"; | ||||
| 
 | ||||
| export const navbarItemDef = reactive({ | ||||
| 	notifications: { | ||||
|  |  | |||
|  | @ -5,10 +5,22 @@ SPDX-License-Identifier: AGPL-3.0-only | |||
| 
 | ||||
| <template> | ||||
| <MkStickyContainer> | ||||
| 	<template #header><MkPageHeader v-model:tab="tab" :actions="headerActions" :tabs="headerTabs"/></template> | ||||
| 	<template #header> | ||||
| 		<MkPageHeader v-model:tab="tab" :actions="headerActions" :tabs="headerTabs"/> | ||||
| 	</template> | ||||
| 	<MkSpacer v-if="tab === 'emojis'" :contentMax="1000" :marginMin="20"> | ||||
| 		<MkButton v-if="$i && ($i.isModerator || $i.policies.canManageCustomEmojis)" primary link to="/custom-emojis-manager">{{ i18n.ts.manageCustomEmojis }}</MkButton> | ||||
|         <MkButton v-if="$i && (!$i.isModerator ||  !$i.policies.canManageCustomEmojis ||  $i.policies.canRequestCustomEmojis)" primary @click="edit" style='margin-top: 8px;' >{{ i18n.ts.requestCustomEmojis }}</MkButton> | ||||
| 		<MkButton | ||||
| 			v-if="$i && ($i.isModerator || $i.policies.canManageCustomEmojis)" primary link | ||||
| 			to="/custom-emojis-manager" | ||||
| 		> | ||||
| 			{{ i18n.ts.manageCustomEmojis }} | ||||
| 		</MkButton> | ||||
| 		<MkButton | ||||
| 			v-if="$i && (!$i.isModerator || !$i.policies.canManageCustomEmojis || $i.policies.canRequestCustomEmojis)" | ||||
| 			primary style="margin-top: 8px;" @click="edit" | ||||
| 		> | ||||
| 			{{ i18n.ts.requestCustomEmojis }} | ||||
| 		</MkButton> | ||||
| 
 | ||||
| 		<div class="query" style="margin-top: 10px;"> | ||||
| 			<MkInput v-model="q" class="" :placeholder="i18n.ts.search"> | ||||
|  | @ -16,10 +28,10 @@ SPDX-License-Identifier: AGPL-3.0-only | |||
| 			</MkInput> | ||||
| 
 | ||||
| 			<!-- たくさんあると邪魔 | ||||
| 			<div class="tags"> | ||||
| 				<span class="tag _button" v-for="tag in customEmojiTags" :class="{ active: selectedTags.has(tag) }" @click="toggleTag(tag)">{{ tag }}</span> | ||||
| 			</div> | ||||
| 			--> | ||||
| 				<div class="tags"> | ||||
| 					<span class="tag _button" v-for="tag in customEmojiTags" :class="{ active: selectedTags.has(tag) }" @click="toggleTag(tag)">{{ tag }}</span> | ||||
| 				</div> | ||||
| 				--> | ||||
| 		</div> | ||||
| 
 | ||||
| 		<MkFoldableSection v-if="searchEmojis"> | ||||
|  | @ -30,16 +42,19 @@ SPDX-License-Identifier: AGPL-3.0-only | |||
| 		</MkFoldableSection> | ||||
| 
 | ||||
| 		<MkFoldableSection v-for="category in filteredCategories" v-once :key="category"> | ||||
| 			<template  #header>{{ category || i18n.ts.other }}</template> | ||||
| 			<template #header>{{ category || i18n.ts.other }}</template> | ||||
| 			<div :class="$style.emojis"> | ||||
| 				<XEmoji v-for="emoji in customEmojis.filter(e => e.category === category && !e.draft)" :key="emoji.name" :emoji="emoji" :draft="emoji.draft"/> | ||||
| 				<XEmoji | ||||
| 					v-for="emoji in customEmojis.filter(e => e.category === category && !e.draft)" :key="emoji.name" | ||||
| 					:emoji="emoji" :draft="emoji.draft" | ||||
| 				/> | ||||
| 			</div> | ||||
| 		</MkFoldableSection> | ||||
| 	</MkSpacer> | ||||
| 
 | ||||
| 	<MkSpacer v-if="tab === 'draft'" :contentMax="1000" :marginMin="20"> | ||||
| 		<div :class="$style.emojis"> | ||||
|             <XEmoji v-for="emoji in draftEmojis" :key="emoji.name" :emoji="emoji" :draft="emoji.draft"/> | ||||
| 			<XEmoji v-for="emoji in draftEmojis" :key="emoji.name" :emoji="emoji" :draft="emoji.draft"/> | ||||
| 		</div> | ||||
| 	</MkSpacer> | ||||
| </MkStickyContainer> | ||||
|  | @ -69,9 +84,9 @@ const headerTabs = $computed(() => [{ | |||
| 	title: i18n.ts.draftEmojis, | ||||
| }]); | ||||
| const filteredCategories = computed(() => { | ||||
|     return customEmojiCategories.value.filter((category: any) => { | ||||
|         return customEmojis.value.some((e: any) => e.category === category && !e.draft); | ||||
|     }); | ||||
| 	return customEmojiCategories.value.filter((category: any) => { | ||||
| 		return customEmojis.value.some((e: any) => e.category === category && !e.draft); | ||||
| 	}); | ||||
| }); | ||||
| definePageMetadata(ref({})); | ||||
| 
 | ||||
|  | @ -87,6 +102,7 @@ let q = $ref(''); | |||
| let searchEmojis = $ref<Misskey.entities.CustomEmoji[]>(null); | ||||
| let selectedTags = $ref(new Set()); | ||||
| const draftEmojis = customEmojis.value.filter(emoji => emoji.draft); | ||||
| 
 | ||||
| function search() { | ||||
| 	if ((q === '' || q == null) && selectedTags.size === 0) { | ||||
| 		searchEmojis = null; | ||||
|  | @ -135,8 +151,8 @@ watch($$(selectedTags), () => { | |||
| }, { deep: true }); | ||||
| 
 | ||||
| definePageMetadata({ | ||||
|     title: i18n.ts.customEmojis, | ||||
|     icon: null, | ||||
| 	title: i18n.ts.customEmojis, | ||||
| 	icon: null, | ||||
| }); | ||||
| </script> | ||||
| 
 | ||||
|  |  | |||
|  | @ -41,7 +41,10 @@ SPDX-License-Identifier: AGPL-3.0-only | |||
| 
 | ||||
| 	<MkPagination v-slot="{items}" ref="instances" :key="host + state" :pagination="pagination"> | ||||
| 		<div :class="$style.items"> | ||||
| 			<MkA v-for="instance in items" :key="instance.id" v-tooltip.mfm="`Status: ${getStatus(instance)}`" :class="$style.item" :to="`/instance-info/${instance.host}`"> | ||||
| 			<MkA | ||||
| 				v-for="instance in items" :key="instance.id" v-tooltip.mfm="`Status: ${getStatus(instance)}`" | ||||
| 				:class="$style.item" :to="`/instance-info/${instance.host}`" | ||||
| 			> | ||||
| 				<MkInstanceCardMini :instance="instance"/> | ||||
| 			</MkA> | ||||
| 		</div> | ||||
|  |  | |||
|  | @ -5,7 +5,9 @@ SPDX-License-Identifier: AGPL-3.0-only | |||
| 
 | ||||
| <template> | ||||
| <MkStickyContainer> | ||||
| 	<template #header><XHeader :actions="headerActions" :tabs="headerTabs"/></template> | ||||
| 	<template #header> | ||||
| 		<XHeader :actions="headerActions" :tabs="headerTabs"/> | ||||
| 	</template> | ||||
| 	<MkSpacer :contentMax="700" :marginMin="16" :marginMax="32"> | ||||
| 		<FormSuspense :p="init"> | ||||
| 			<MkTextarea v-model="blockedHosts"> | ||||
|  | @ -20,7 +22,7 @@ SPDX-License-Identifier: AGPL-3.0-only | |||
| </template> | ||||
| 
 | ||||
| <script lang="ts" setup> | ||||
| import { } from 'vue'; | ||||
| import {} from 'vue'; | ||||
| import XHeader from './_header_.vue'; | ||||
| import MkButton from '@/components/MkButton.vue'; | ||||
| import MkTextarea from '@/components/MkTextarea.vue'; | ||||
|  |  | |||
		Loading…
	
		Reference in New Issue