Merge branch 'develop' into ed25519
				
					
				
			This commit is contained in:
		| @@ -5,7 +5,11 @@ | |||||||
|  |  | ||||||
| ### Client | ### Client | ||||||
| - Enhance: 自分のノートの添付ファイルから直接ファイルの詳細ページに飛べるように | - Enhance: 自分のノートの添付ファイルから直接ファイルの詳細ページに飛べるように | ||||||
|  | - Enhance: 広告がMisskeyと同一ドメインの場合はRouterで遷移するように | ||||||
|  | - Enhance: リアクション・いいねの総数を表示するように | ||||||
|  | - Enhance: リアクション受け入れが「いいねのみ」の場合はリアクション絵文字一覧を表示しないように | ||||||
| - Fix: 一部のページ内リンクが正しく動作しない問題を修正 | - Fix: 一部のページ内リンクが正しく動作しない問題を修正 | ||||||
|  | - Fix: 周年の実績が閏年を考慮しない問題を修正 | ||||||
|  |  | ||||||
| ### Server | ### Server | ||||||
| - | - | ||||||
|   | |||||||
							
								
								
									
										4
									
								
								locales/index.d.ts
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								locales/index.d.ts
									
									
									
									
										vendored
									
									
								
							| @@ -8909,6 +8909,10 @@ export interface Locale extends ILocale { | |||||||
|          * {n}人がリアクションしました |          * {n}人がリアクションしました | ||||||
|          */ |          */ | ||||||
|         "reactedBySomeUsers": ParameterizedString<"n">; |         "reactedBySomeUsers": ParameterizedString<"n">; | ||||||
|  |         /** | ||||||
|  |          * {n}人がいいねしました | ||||||
|  |          */ | ||||||
|  |         "likedBySomeUsers": ParameterizedString<"n">; | ||||||
|         /** |         /** | ||||||
|          * {n}人がリノートしました |          * {n}人がリノートしました | ||||||
|          */ |          */ | ||||||
|   | |||||||
| @@ -2355,6 +2355,7 @@ _notification: | |||||||
|   sendTestNotification: "テスト通知を送信する" |   sendTestNotification: "テスト通知を送信する" | ||||||
|   notificationWillBeDisplayedLikeThis: "通知はこのように表示されます" |   notificationWillBeDisplayedLikeThis: "通知はこのように表示されます" | ||||||
|   reactedBySomeUsers: "{n}人がリアクションしました" |   reactedBySomeUsers: "{n}人がリアクションしました" | ||||||
|  |   likedBySomeUsers: "{n}人がいいねしました" | ||||||
|   renotedBySomeUsers: "{n}人がリノートしました" |   renotedBySomeUsers: "{n}人がリノートしました" | ||||||
|   followedBySomeUsers: "{n}人にフォローされました" |   followedBySomeUsers: "{n}人にフォローされました" | ||||||
|   flushNotification: "通知の履歴をリセットする" |   flushNotification: "通知の履歴をリセットする" | ||||||
|   | |||||||
| @@ -19,7 +19,7 @@ | |||||||
| 		"watch": "node watch.mjs", | 		"watch": "node watch.mjs", | ||||||
| 		"restart": "pnpm build && pnpm start", | 		"restart": "pnpm build && pnpm start", | ||||||
| 		"dev": "nodemon -w src -e ts,js,mjs,cjs,json --exec \"cross-env NODE_ENV=development pnpm run restart\"", | 		"dev": "nodemon -w src -e ts,js,mjs,cjs,json --exec \"cross-env NODE_ENV=development pnpm run restart\"", | ||||||
| 		"typecheck": "tsc --noEmit", | 		"typecheck": "tsc --noEmit && tsc -p test --noEmit", | ||||||
| 		"eslint": "eslint --quiet \"src/**/*.ts\"", | 		"eslint": "eslint --quiet \"src/**/*.ts\"", | ||||||
| 		"lint": "pnpm typecheck && pnpm eslint", | 		"lint": "pnpm typecheck && pnpm eslint", | ||||||
| 		"jest": "cross-env NODE_ENV=test node --experimental-vm-modules --experimental-import-meta-resolve node_modules/jest/bin/jest.js --forceExit --config jest.config.unit.cjs", | 		"jest": "cross-env NODE_ENV=test node --experimental-vm-modules --experimental-import-meta-resolve node_modules/jest/bin/jest.js --forceExit --config jest.config.unit.cjs", | ||||||
|   | |||||||
| @@ -333,6 +333,7 @@ export class NoteEntityService implements OnModuleInit { | |||||||
| 			visibleUserIds: note.visibility === 'specified' ? note.visibleUserIds : undefined, | 			visibleUserIds: note.visibility === 'specified' ? note.visibleUserIds : undefined, | ||||||
| 			renoteCount: note.renoteCount, | 			renoteCount: note.renoteCount, | ||||||
| 			repliesCount: note.repliesCount, | 			repliesCount: note.repliesCount, | ||||||
|  | 			reactionCount: Object.values(note.reactions).reduce((a, b) => a + b, 0), | ||||||
| 			reactions: this.reactionService.convertLegacyReactions(note.reactions), | 			reactions: this.reactionService.convertLegacyReactions(note.reactions), | ||||||
| 			reactionEmojis: this.customEmojiService.populateEmojis(reactionEmojiNames, host), | 			reactionEmojis: this.customEmojiService.populateEmojis(reactionEmojiNames, host), | ||||||
| 			reactionAndUserPairCache: opts.withReactionAndUserPairCache ? note.reactionAndUserPairCache : undefined, | 			reactionAndUserPairCache: opts.withReactionAndUserPairCache ? note.reactionAndUserPairCache : undefined, | ||||||
|   | |||||||
| @@ -223,6 +223,10 @@ export const packedNoteSchema = { | |||||||
| 				}], | 				}], | ||||||
| 			}, | 			}, | ||||||
| 		}, | 		}, | ||||||
|  | 		reactionCount: { | ||||||
|  | 			type: 'number', | ||||||
|  | 			optional: false, nullable: false, | ||||||
|  | 		}, | ||||||
| 		renoteCount: { | 		renoteCount: { | ||||||
| 			type: 'number', | 			type: 'number', | ||||||
| 			optional: false, nullable: false, | 			optional: false, nullable: false, | ||||||
|   | |||||||
| @@ -86,8 +86,8 @@ | |||||||
| 	//#endregion | 	//#endregion | ||||||
|  |  | ||||||
| 	//#region Script | 	//#region Script | ||||||
| 	function importAppScript() { | 	async function importAppScript() { | ||||||
| 		import(`/vite/${CLIENT_ENTRY}`) | 		await import(`/vite/${CLIENT_ENTRY}`) | ||||||
| 			.catch(async e => { | 			.catch(async e => { | ||||||
| 				console.error(e); | 				console.error(e); | ||||||
| 				renderError('APP_IMPORT', e); | 				renderError('APP_IMPORT', e); | ||||||
|   | |||||||
| @@ -472,7 +472,7 @@ describe('Note', () => { | |||||||
| 						priority: 0, | 						priority: 0, | ||||||
| 						value: true, | 						value: true, | ||||||
| 					}, | 					}, | ||||||
| 				}, | 				} as any, | ||||||
| 			}, alice); | 			}, alice); | ||||||
|  |  | ||||||
| 			assert.strictEqual(res.status, 200); | 			assert.strictEqual(res.status, 200); | ||||||
| @@ -784,7 +784,7 @@ describe('Note', () => { | |||||||
| 						priority: 1, | 						priority: 1, | ||||||
| 						value: 0, | 						value: 0, | ||||||
| 					}, | 					}, | ||||||
| 				}, | 				} as any, | ||||||
| 			}, alice); | 			}, alice); | ||||||
|  |  | ||||||
| 			assert.strictEqual(res.status, 200); | 			assert.strictEqual(res.status, 200); | ||||||
| @@ -838,7 +838,7 @@ describe('Note', () => { | |||||||
| 						priority: 1, | 						priority: 1, | ||||||
| 						value: 0, | 						value: 0, | ||||||
| 					}, | 					}, | ||||||
| 				}, | 				} as any, | ||||||
| 			}, alice); | 			}, alice); | ||||||
|  |  | ||||||
| 			assert.strictEqual(res.status, 200); | 			assert.strictEqual(res.status, 200); | ||||||
| @@ -894,7 +894,7 @@ describe('Note', () => { | |||||||
| 						priority: 1, | 						priority: 1, | ||||||
| 						value: 1, | 						value: 1, | ||||||
| 					}, | 					}, | ||||||
| 				}, | 				} as any, | ||||||
| 			}, alice); | 			}, alice); | ||||||
|  |  | ||||||
| 			assert.strictEqual(res.status, 200); | 			assert.strictEqual(res.status, 200); | ||||||
|   | |||||||
| @@ -890,17 +890,35 @@ describe('Timelines', () => { | |||||||
|  |  | ||||||
| 			const list = await api('users/lists/create', { name: 'list' }, alice).then(res => res.body); | 			const list = await api('users/lists/create', { name: 'list' }, alice).then(res => res.body); | ||||||
| 			await api('users/lists/push', { listId: list.id, userId: bob.id }, alice); | 			await api('users/lists/push', { listId: list.id, userId: bob.id }, alice); | ||||||
|  | 			await api('users/lists/update-membership', { listId: list.id, userId: bob.id, withReplies: false }, alice); | ||||||
| 			await sleep(1000); | 			await sleep(1000); | ||||||
| 			const aliceNote = await post(alice, { text: 'hi' }); | 			const aliceNote = await post(alice, { text: 'hi' }); | ||||||
| 			const bobNote = await post(bob, { text: 'hi', replyId: aliceNote.id }); | 			const bobNote = await post(bob, { text: 'hi', replyId: aliceNote.id }); | ||||||
|  |  | ||||||
| 			await waitForPushToTl(); | 			await waitForPushToTl(); | ||||||
|  |  | ||||||
| 			const res = await api('notes/user-list-timeline', { listId: list.id, withReplies: false }, alice); | 			const res = await api('notes/user-list-timeline', { listId: list.id }, alice); | ||||||
|  |  | ||||||
| 			assert.strictEqual(res.body.some((note: any) => note.id === bobNote.id), true); | 			assert.strictEqual(res.body.some((note: any) => note.id === bobNote.id), true); | ||||||
| 		}); | 		}); | ||||||
|  |  | ||||||
|  | 		test.concurrent('withReplies: false でリスインしているフォローしていないユーザーの他人への返信が含まれない', async () => { | ||||||
|  | 			const [alice, bob, carol] = await Promise.all([signup(), signup(), signup()]); | ||||||
|  |  | ||||||
|  | 			const list = await api('users/lists/create', { name: 'list' }, alice).then(res => res.body); | ||||||
|  | 			await api('users/lists/push', { listId: list.id, userId: bob.id }, alice); | ||||||
|  | 			await api('users/lists/update-membership', { listId: list.id, userId: bob.id, withReplies: false }, alice); | ||||||
|  | 			await sleep(1000); | ||||||
|  | 			const carolNote = await post(carol, { text: 'hi' }); | ||||||
|  | 			const bobNote = await post(bob, { text: 'hi', replyId: carolNote.id }); | ||||||
|  |  | ||||||
|  | 			await waitForPushToTl(); | ||||||
|  |  | ||||||
|  | 			const res = await api('notes/user-list-timeline', { listId: list.id }, alice); | ||||||
|  |  | ||||||
|  | 			assert.strictEqual(res.body.some((note: any) => note.id === bobNote.id), false); | ||||||
|  | 		}); | ||||||
|  |  | ||||||
| 		test.concurrent('withReplies: true でリスインしているフォローしていないユーザーの他人への返信が含まれる', async () => { | 		test.concurrent('withReplies: true でリスインしているフォローしていないユーザーの他人への返信が含まれる', async () => { | ||||||
| 			const [alice, bob, carol] = await Promise.all([signup(), signup(), signup()]); | 			const [alice, bob, carol] = await Promise.all([signup(), signup(), signup()]); | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										7
									
								
								packages/backend/test/global.d.ts
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								packages/backend/test/global.d.ts
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,7 @@ | |||||||
|  | /* | ||||||
|  |  * SPDX-FileCopyrightText: syuilo and misskey-project | ||||||
|  |  * SPDX-License-Identifier: AGPL-3.0-only | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | // eslint-disable-next-line @typescript-eslint/no-explicit-any | ||||||
|  | type FIXME = any; | ||||||
| @@ -4,10 +4,10 @@ | |||||||
|  */ |  */ | ||||||
|  |  | ||||||
| import Ajv from 'ajv'; | import Ajv from 'ajv'; | ||||||
| import { Schema } from '@/misc/schema'; | import { Schema } from '@/misc/json-schema.js'; | ||||||
|  |  | ||||||
| export const getValidator = (paramDef: Schema) => { | export const getValidator = (paramDef: Schema) => { | ||||||
| 	const ajv = new Ajv({ | 	const ajv = new Ajv.default({ | ||||||
| 		useDefaults: true, | 		useDefaults: true, | ||||||
| 	}); | 	}); | ||||||
| 	ajv.addFormat('misskey:id', /^[a-zA-Z0-9]+$/); | 	ajv.addFormat('misskey:id', /^[a-zA-Z0-9]+$/); | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ | |||||||
| 		"noImplicitAny": true, | 		"noImplicitAny": true, | ||||||
| 		"noImplicitReturns": true, | 		"noImplicitReturns": true, | ||||||
| 		"noUnusedParameters": false, | 		"noUnusedParameters": false, | ||||||
| 		"noUnusedLocals": true, | 		"noUnusedLocals": false, | ||||||
| 		"noFallthroughCasesInSwitch": true, | 		"noFallthroughCasesInSwitch": true, | ||||||
| 		"declaration": false, | 		"declaration": false, | ||||||
| 		"sourceMap": true, | 		"sourceMap": true, | ||||||
| @@ -18,6 +18,7 @@ | |||||||
| 		"strict": true, | 		"strict": true, | ||||||
| 		"strictNullChecks": true, | 		"strictNullChecks": true, | ||||||
| 		"strictPropertyInitialization": false, | 		"strictPropertyInitialization": false, | ||||||
|  | 		"skipLibCheck": true, | ||||||
| 		"experimentalDecorators": true, | 		"experimentalDecorators": true, | ||||||
| 		"emitDecoratorMetadata": true, | 		"emitDecoratorMetadata": true, | ||||||
| 		"resolveJsonModule": true, | 		"resolveJsonModule": true, | ||||||
|   | |||||||
| @@ -90,7 +90,8 @@ describe('RelayService', () => { | |||||||
|  |  | ||||||
| 		expect(queueService.deliver).toHaveBeenCalled(); | 		expect(queueService.deliver).toHaveBeenCalled(); | ||||||
| 		expect(queueService.deliver.mock.lastCall![1]?.type).toBe('Undo'); | 		expect(queueService.deliver.mock.lastCall![1]?.type).toBe('Undo'); | ||||||
| 		expect(queueService.deliver.mock.lastCall![1]?.object.type).toBe('Follow'); | 		expect(typeof queueService.deliver.mock.lastCall![1]?.object).toBe('object'); | ||||||
|  | 		expect((queueService.deliver.mock.lastCall![1]?.object as any).type).toBe('Follow'); | ||||||
| 		expect(queueService.deliver.mock.lastCall![2]).toBe('https://example.com'); | 		expect(queueService.deliver.mock.lastCall![2]).toBe('https://example.com'); | ||||||
| 		//expect(queueService.deliver.mock.lastCall![0].username).toBe('relay.actor'); | 		//expect(queueService.deliver.mock.lastCall![0].username).toBe('relay.actor'); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -187,14 +187,26 @@ export async function mainBoot() { | |||||||
| 		if ($i.followersCount >= 500) claimAchievement('followers500'); | 		if ($i.followersCount >= 500) claimAchievement('followers500'); | ||||||
| 		if ($i.followersCount >= 1000) claimAchievement('followers1000'); | 		if ($i.followersCount >= 1000) claimAchievement('followers1000'); | ||||||
|  |  | ||||||
| 		if (Date.now() - new Date($i.createdAt).getTime() > 1000 * 60 * 60 * 24 * 365) { | 		const createdAt = new Date($i.createdAt); | ||||||
| 			claimAchievement('passedSinceAccountCreated1'); | 		const createdAtThreeYearsLater = new Date($i.createdAt); | ||||||
| 		} | 		createdAtThreeYearsLater.setFullYear(createdAtThreeYearsLater.getFullYear() + 3); | ||||||
| 		if (Date.now() - new Date($i.createdAt).getTime() > 1000 * 60 * 60 * 24 * 365 * 2) { | 		if (now >= createdAtThreeYearsLater) { | ||||||
| 			claimAchievement('passedSinceAccountCreated2'); |  | ||||||
| 		} |  | ||||||
| 		if (Date.now() - new Date($i.createdAt).getTime() > 1000 * 60 * 60 * 24 * 365 * 3) { |  | ||||||
| 			claimAchievement('passedSinceAccountCreated3'); | 			claimAchievement('passedSinceAccountCreated3'); | ||||||
|  | 			claimAchievement('passedSinceAccountCreated2'); | ||||||
|  | 			claimAchievement('passedSinceAccountCreated1'); | ||||||
|  | 		} else { | ||||||
|  | 			const createdAtTwoYearsLater = new Date($i.createdAt); | ||||||
|  | 			createdAtTwoYearsLater.setFullYear(createdAtTwoYearsLater.getFullYear() + 2); | ||||||
|  | 			if (now >= createdAtTwoYearsLater) { | ||||||
|  | 				claimAchievement('passedSinceAccountCreated2'); | ||||||
|  | 				claimAchievement('passedSinceAccountCreated1'); | ||||||
|  | 			} else { | ||||||
|  | 				const createdAtOneYearLater = new Date($i.createdAt); | ||||||
|  | 				createdAtOneYearLater.setFullYear(createdAtOneYearLater.getFullYear() + 1); | ||||||
|  | 				if (now >= createdAtOneYearLater) { | ||||||
|  | 					claimAchievement('passedSinceAccountCreated1'); | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		if (claimedAchievements.length >= 30) { | 		if (claimedAchievements.length >= 30) { | ||||||
| @@ -229,7 +241,7 @@ export async function mainBoot() { | |||||||
|  |  | ||||||
| 		const latestDonationInfoShownAt = miLocalStorage.getItem('latestDonationInfoShownAt'); | 		const latestDonationInfoShownAt = miLocalStorage.getItem('latestDonationInfoShownAt'); | ||||||
| 		const neverShowDonationInfo = miLocalStorage.getItem('neverShowDonationInfo'); | 		const neverShowDonationInfo = miLocalStorage.getItem('neverShowDonationInfo'); | ||||||
| 		if (neverShowDonationInfo !== 'true' && (new Date($i.createdAt).getTime() < (Date.now() - (1000 * 60 * 60 * 24 * 3))) && !location.pathname.startsWith('/miauth')) { | 		if (neverShowDonationInfo !== 'true' && (createdAt.getTime() < (Date.now() - (1000 * 60 * 60 * 24 * 3))) && !location.pathname.startsWith('/miauth')) { | ||||||
| 			if (latestDonationInfoShownAt == null || (new Date(latestDonationInfoShownAt).getTime() < (Date.now() - (1000 * 60 * 60 * 24 * 30)))) { | 			if (latestDonationInfoShownAt == null || (new Date(latestDonationInfoShownAt).getTime() < (Date.now() - (1000 * 60 * 60 * 24 * 30)))) { | ||||||
| 				popup(defineAsyncComponent(() => import('@/components/MkDonation.vue')), {}, {}, 'closed'); | 				popup(defineAsyncComponent(() => import('@/components/MkDonation.vue')), {}, {}, 'closed'); | ||||||
| 			} | 			} | ||||||
|   | |||||||
| @@ -93,7 +93,7 @@ SPDX-License-Identifier: AGPL-3.0-only | |||||||
| 				</div> | 				</div> | ||||||
| 				<MkA v-if="appearNote.channel && !inChannel" :class="$style.channel" :to="`/channels/${appearNote.channel.id}`"><i class="ti ti-device-tv"></i> {{ appearNote.channel.name }}</MkA> | 				<MkA v-if="appearNote.channel && !inChannel" :class="$style.channel" :to="`/channels/${appearNote.channel.id}`"><i class="ti ti-device-tv"></i> {{ appearNote.channel.name }}</MkA> | ||||||
| 			</div> | 			</div> | ||||||
| 			<MkReactionsViewer :note="appearNote" :maxNumber="16" @mockUpdateMyReaction="emitUpdReaction"> | 			<MkReactionsViewer v-if="appearNote.reactionAcceptance !== 'likeOnly'" :note="appearNote" :maxNumber="16" @mockUpdateMyReaction="emitUpdReaction"> | ||||||
| 				<template #more> | 				<template #more> | ||||||
| 					<div :class="$style.reactionOmitted">{{ i18n.ts.more }}</div> | 					<div :class="$style.reactionOmitted">{{ i18n.ts.more }}</div> | ||||||
| 				</template> | 				</template> | ||||||
| @@ -101,7 +101,7 @@ SPDX-License-Identifier: AGPL-3.0-only | |||||||
| 			<footer :class="$style.footer"> | 			<footer :class="$style.footer"> | ||||||
| 				<button :class="$style.footerButton" class="_button" @click="reply()"> | 				<button :class="$style.footerButton" class="_button" @click="reply()"> | ||||||
| 					<i class="ti ti-arrow-back-up"></i> | 					<i class="ti ti-arrow-back-up"></i> | ||||||
| 					<p v-if="appearNote.repliesCount > 0" :class="$style.footerButtonCount">{{ appearNote.repliesCount }}</p> | 					<p v-if="appearNote.repliesCount > 0" :class="$style.footerButtonCount">{{ number(appearNote.repliesCount) }}</p> | ||||||
| 				</button> | 				</button> | ||||||
| 				<button | 				<button | ||||||
| 					v-if="canRenote" | 					v-if="canRenote" | ||||||
| @@ -111,17 +111,17 @@ SPDX-License-Identifier: AGPL-3.0-only | |||||||
| 					@mousedown="renote()" | 					@mousedown="renote()" | ||||||
| 				> | 				> | ||||||
| 					<i class="ti ti-repeat"></i> | 					<i class="ti ti-repeat"></i> | ||||||
| 					<p v-if="appearNote.renoteCount > 0" :class="$style.footerButtonCount">{{ appearNote.renoteCount }}</p> | 					<p v-if="appearNote.renoteCount > 0" :class="$style.footerButtonCount">{{ number(appearNote.renoteCount) }}</p> | ||||||
| 				</button> | 				</button> | ||||||
| 				<button v-else :class="$style.footerButton" class="_button" disabled> | 				<button v-else :class="$style.footerButton" class="_button" disabled> | ||||||
| 					<i class="ti ti-ban"></i> | 					<i class="ti ti-ban"></i> | ||||||
| 				</button> | 				</button> | ||||||
| 				<button v-if="appearNote.myReaction == null" ref="reactButton" :class="$style.footerButton" class="_button" @mousedown="react()"> | 				<button ref="reactButton" :class="$style.footerButton" class="_button" @click="toggleReact()"> | ||||||
| 					<i v-if="appearNote.reactionAcceptance === 'likeOnly'" class="ti ti-heart"></i> | 					<i v-if="appearNote.reactionAcceptance === 'likeOnly' && appearNote.myReaction != null" class="ti ti-heart-filled" style="color: var(--eventReactionHeart);"></i> | ||||||
|  | 					<i v-else-if="appearNote.myReaction != null" class="ti ti-minus" style="color: var(--accent);"></i> | ||||||
|  | 					<i v-else-if="appearNote.reactionAcceptance === 'likeOnly'" class="ti ti-heart"></i> | ||||||
| 					<i v-else class="ti ti-plus"></i> | 					<i v-else class="ti ti-plus"></i> | ||||||
| 				</button> | 					<p v-if="appearNote.reactionCount > 0" :class="$style.footerButtonCount">{{ number(appearNote.reactionCount) }}</p> | ||||||
| 				<button v-if="appearNote.myReaction != null" ref="reactButton" :class="$style.footerButton" class="_button" @click="undoReact(appearNote)"> |  | ||||||
| 					<i class="ti ti-minus"></i> |  | ||||||
| 				</button> | 				</button> | ||||||
| 				<button v-if="defaultStore.state.showClipButtonInNoteFooter" ref="clipButton" :class="$style.footerButton" class="_button" @mousedown="clip()"> | 				<button v-if="defaultStore.state.showClipButtonInNoteFooter" ref="clipButton" :class="$style.footerButton" class="_button" @mousedown="clip()"> | ||||||
| 					<i class="ti ti-paperclip"></i> | 					<i class="ti ti-paperclip"></i> | ||||||
| @@ -175,6 +175,7 @@ import { pleaseLogin } from '@/scripts/please-login.js'; | |||||||
| import { focusPrev, focusNext } from '@/scripts/focus.js'; | import { focusPrev, focusNext } from '@/scripts/focus.js'; | ||||||
| import { checkWordMute } from '@/scripts/check-word-mute.js'; | import { checkWordMute } from '@/scripts/check-word-mute.js'; | ||||||
| import { userPage } from '@/filters/user.js'; | import { userPage } from '@/filters/user.js'; | ||||||
|  | import number from '@/filters/number.js'; | ||||||
| import * as os from '@/os.js'; | import * as os from '@/os.js'; | ||||||
| import * as sound from '@/scripts/sound.js'; | import * as sound from '@/scripts/sound.js'; | ||||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||||
| @@ -420,6 +421,14 @@ function undoReact(targetNote: Misskey.entities.Note): void { | |||||||
| 	}); | 	}); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | function toggleReact() { | ||||||
|  | 	if (appearNote.value.myReaction == null) { | ||||||
|  | 		react(); | ||||||
|  | 	} else { | ||||||
|  | 		undoReact(appearNote.value); | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
| function onContextmenu(ev: MouseEvent): void { | function onContextmenu(ev: MouseEvent): void { | ||||||
| 	if (props.mock) { | 	if (props.mock) { | ||||||
| 		return; | 		return; | ||||||
|   | |||||||
| @@ -106,10 +106,10 @@ SPDX-License-Identifier: AGPL-3.0-only | |||||||
| 					<MkTime :time="appearNote.createdAt" mode="detail" colored/> | 					<MkTime :time="appearNote.createdAt" mode="detail" colored/> | ||||||
| 				</MkA> | 				</MkA> | ||||||
| 			</div> | 			</div> | ||||||
| 			<MkReactionsViewer ref="reactionsViewer" :note="appearNote"/> | 			<MkReactionsViewer v-if="appearNote.reactionAcceptance !== 'likeOnly'" ref="reactionsViewer" :note="appearNote"/> | ||||||
| 			<button class="_button" :class="$style.noteFooterButton" @click="reply()"> | 			<button class="_button" :class="$style.noteFooterButton" @click="reply()"> | ||||||
| 				<i class="ti ti-arrow-back-up"></i> | 				<i class="ti ti-arrow-back-up"></i> | ||||||
| 				<p v-if="appearNote.repliesCount > 0" :class="$style.noteFooterButtonCount">{{ appearNote.repliesCount }}</p> | 				<p v-if="appearNote.repliesCount > 0" :class="$style.noteFooterButtonCount">{{ number(appearNote.repliesCount) }}</p> | ||||||
| 			</button> | 			</button> | ||||||
| 			<button | 			<button | ||||||
| 				v-if="canRenote" | 				v-if="canRenote" | ||||||
| @@ -119,17 +119,17 @@ SPDX-License-Identifier: AGPL-3.0-only | |||||||
| 				@mousedown="renote()" | 				@mousedown="renote()" | ||||||
| 			> | 			> | ||||||
| 				<i class="ti ti-repeat"></i> | 				<i class="ti ti-repeat"></i> | ||||||
| 				<p v-if="appearNote.renoteCount > 0" :class="$style.noteFooterButtonCount">{{ appearNote.renoteCount }}</p> | 				<p v-if="appearNote.renoteCount > 0" :class="$style.noteFooterButtonCount">{{ number(appearNote.renoteCount) }}</p> | ||||||
| 			</button> | 			</button> | ||||||
| 			<button v-else class="_button" :class="$style.noteFooterButton" disabled> | 			<button v-else class="_button" :class="$style.noteFooterButton" disabled> | ||||||
| 				<i class="ti ti-ban"></i> | 				<i class="ti ti-ban"></i> | ||||||
| 			</button> | 			</button> | ||||||
| 			<button v-if="appearNote.myReaction == null" ref="reactButton" :class="$style.noteFooterButton" class="_button" @mousedown="react()"> | 			<button ref="reactButton" :class="$style.noteFooterButton" class="_button" @click="toggleReact()"> | ||||||
| 				<i v-if="appearNote.reactionAcceptance === 'likeOnly'" class="ti ti-heart"></i> | 				<i v-if="appearNote.reactionAcceptance === 'likeOnly' && appearNote.myReaction != null" class="ti ti-heart-filled" style="color: var(--eventReactionHeart);"></i> | ||||||
|  | 				<i v-else-if="appearNote.myReaction != null" class="ti ti-minus" style="color: var(--accent);"></i> | ||||||
|  | 				<i v-else-if="appearNote.reactionAcceptance === 'likeOnly'" class="ti ti-heart"></i> | ||||||
| 				<i v-else class="ti ti-plus"></i> | 				<i v-else class="ti ti-plus"></i> | ||||||
| 			</button> | 				<p v-if="appearNote.reactionCount > 0" :class="$style.noteFooterButtonCount">{{ number(appearNote.reactionCount) }}</p> | ||||||
| 			<button v-if="appearNote.myReaction != null" ref="reactButton" class="_button" :class="[$style.noteFooterButton, $style.reacted]" @click="undoReact(appearNote)"> |  | ||||||
| 				<i class="ti ti-minus"></i> |  | ||||||
| 			</button> | 			</button> | ||||||
| 			<button v-if="defaultStore.state.showClipButtonInNoteFooter" ref="clipButton" class="_button" :class="$style.noteFooterButton" @mousedown="clip()"> | 			<button v-if="defaultStore.state.showClipButtonInNoteFooter" ref="clipButton" class="_button" :class="$style.noteFooterButton" @mousedown="clip()"> | ||||||
| 				<i class="ti ti-paperclip"></i> | 				<i class="ti ti-paperclip"></i> | ||||||
| @@ -209,6 +209,7 @@ import { pleaseLogin } from '@/scripts/please-login.js'; | |||||||
| import { checkWordMute } from '@/scripts/check-word-mute.js'; | import { checkWordMute } from '@/scripts/check-word-mute.js'; | ||||||
| import { userPage } from '@/filters/user.js'; | import { userPage } from '@/filters/user.js'; | ||||||
| import { notePage } from '@/filters/note.js'; | import { notePage } from '@/filters/note.js'; | ||||||
|  | import number from '@/filters/number.js'; | ||||||
| import * as os from '@/os.js'; | import * as os from '@/os.js'; | ||||||
| import { misskeyApi } from '@/scripts/misskey-api.js'; | import { misskeyApi } from '@/scripts/misskey-api.js'; | ||||||
| import * as sound from '@/scripts/sound.js'; | import * as sound from '@/scripts/sound.js'; | ||||||
| @@ -401,14 +402,22 @@ function react(viaKeyboard = false): void { | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
| function undoReact(note): void { | function undoReact(targetNote: Misskey.entities.Note): void { | ||||||
| 	const oldReaction = note.myReaction; | 	const oldReaction = targetNote.myReaction; | ||||||
| 	if (!oldReaction) return; | 	if (!oldReaction) return; | ||||||
| 	misskeyApi('notes/reactions/delete', { | 	misskeyApi('notes/reactions/delete', { | ||||||
| 		noteId: note.id, | 		noteId: targetNote.id, | ||||||
| 	}); | 	}); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | function toggleReact() { | ||||||
|  | 	if (appearNote.value.myReaction == null) { | ||||||
|  | 		react(); | ||||||
|  | 	} else { | ||||||
|  | 		undoReact(appearNote.value); | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
| function onContextmenu(ev: MouseEvent): void { | function onContextmenu(ev: MouseEvent): void { | ||||||
| 	const isLink = (el: HTMLElement): boolean => { | 	const isLink = (el: HTMLElement): boolean => { | ||||||
| 		if (el.tagName === 'A') return true; | 		if (el.tagName === 'A') return true; | ||||||
|   | |||||||
| @@ -8,6 +8,7 @@ SPDX-License-Identifier: AGPL-3.0-only | |||||||
| 	<div :class="$style.head"> | 	<div :class="$style.head"> | ||||||
| 		<MkAvatar v-if="['pollEnded', 'note'].includes(notification.type) && notification.note" :class="$style.icon" :user="notification.note.user" link preview/> | 		<MkAvatar v-if="['pollEnded', 'note'].includes(notification.type) && notification.note" :class="$style.icon" :user="notification.note.user" link preview/> | ||||||
| 		<MkAvatar v-else-if="['roleAssigned', 'achievementEarned'].includes(notification.type)" :class="$style.icon" :user="$i" link preview/> | 		<MkAvatar v-else-if="['roleAssigned', 'achievementEarned'].includes(notification.type)" :class="$style.icon" :user="$i" link preview/> | ||||||
|  | 		<div v-else-if="notification.type === 'reaction:grouped' && notification.note.reactionAcceptance === 'likeOnly'" :class="[$style.icon, $style.icon_reactionGroupHeart]"><i class="ti ti-heart" style="line-height: 1;"></i></div> | ||||||
| 		<div v-else-if="notification.type === 'reaction:grouped'" :class="[$style.icon, $style.icon_reactionGroup]"><i class="ti ti-plus" style="line-height: 1;"></i></div> | 		<div v-else-if="notification.type === 'reaction:grouped'" :class="[$style.icon, $style.icon_reactionGroup]"><i class="ti ti-plus" style="line-height: 1;"></i></div> | ||||||
| 		<div v-else-if="notification.type === 'renote:grouped'" :class="[$style.icon, $style.icon_renoteGroup]"><i class="ti ti-repeat" style="line-height: 1;"></i></div> | 		<div v-else-if="notification.type === 'renote:grouped'" :class="[$style.icon, $style.icon_renoteGroup]"><i class="ti ti-repeat" style="line-height: 1;"></i></div> | ||||||
| 		<img v-else-if="notification.type === 'test'" :class="$style.icon" :src="infoImageUrl"/> | 		<img v-else-if="notification.type === 'test'" :class="$style.icon" :src="infoImageUrl"/> | ||||||
| @@ -57,6 +58,7 @@ SPDX-License-Identifier: AGPL-3.0-only | |||||||
| 			<span v-else-if="notification.type === 'achievementEarned'">{{ i18n.ts._notification.achievementEarned }}</span> | 			<span v-else-if="notification.type === 'achievementEarned'">{{ i18n.ts._notification.achievementEarned }}</span> | ||||||
| 			<span v-else-if="notification.type === 'test'">{{ i18n.ts._notification.testNotification }}</span> | 			<span v-else-if="notification.type === 'test'">{{ i18n.ts._notification.testNotification }}</span> | ||||||
| 			<MkA v-else-if="notification.type === 'follow' || notification.type === 'mention' || notification.type === 'reply' || notification.type === 'renote' || notification.type === 'quote' || notification.type === 'reaction' || notification.type === 'receiveFollowRequest' || notification.type === 'followRequestAccepted'" v-user-preview="notification.user.id" :class="$style.headerName" :to="userPage(notification.user)"><MkUserName :user="notification.user"/></MkA> | 			<MkA v-else-if="notification.type === 'follow' || notification.type === 'mention' || notification.type === 'reply' || notification.type === 'renote' || notification.type === 'quote' || notification.type === 'reaction' || notification.type === 'receiveFollowRequest' || notification.type === 'followRequestAccepted'" v-user-preview="notification.user.id" :class="$style.headerName" :to="userPage(notification.user)"><MkUserName :user="notification.user"/></MkA> | ||||||
|  | 			<span v-else-if="notification.type === 'reaction:grouped' && notification.note.reactionAcceptance === 'likeOnly'">{{ i18n.tsx._notification.likedBySomeUsers({ n: notification.reactions.length }) }}</span> | ||||||
| 			<span v-else-if="notification.type === 'reaction:grouped'">{{ i18n.tsx._notification.reactedBySomeUsers({ n: notification.reactions.length }) }}</span> | 			<span v-else-if="notification.type === 'reaction:grouped'">{{ i18n.tsx._notification.reactedBySomeUsers({ n: notification.reactions.length }) }}</span> | ||||||
| 			<span v-else-if="notification.type === 'renote:grouped'">{{ i18n.tsx._notification.renotedBySomeUsers({ n: notification.users.length }) }}</span> | 			<span v-else-if="notification.type === 'renote:grouped'">{{ i18n.tsx._notification.renotedBySomeUsers({ n: notification.users.length }) }}</span> | ||||||
| 			<span v-else-if="notification.type === 'app'">{{ notification.header }}</span> | 			<span v-else-if="notification.type === 'app'">{{ notification.header }}</span> | ||||||
| @@ -201,6 +203,7 @@ const rejectFollowRequest = () => { | |||||||
| } | } | ||||||
|  |  | ||||||
| .icon_reactionGroup, | .icon_reactionGroup, | ||||||
|  | .icon_reactionGroupHeart, | ||||||
| .icon_renoteGroup { | .icon_renoteGroup { | ||||||
| 	display: grid; | 	display: grid; | ||||||
| 	align-items: center; | 	align-items: center; | ||||||
| @@ -213,11 +216,15 @@ const rejectFollowRequest = () => { | |||||||
| } | } | ||||||
|  |  | ||||||
| .icon_reactionGroup { | .icon_reactionGroup { | ||||||
| 	background: #e99a0b; | 	background: var(--eventReaction); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .icon_reactionGroupHeart { | ||||||
|  | 	background: var(--eventReactionHeart); | ||||||
| } | } | ||||||
|  |  | ||||||
| .icon_renoteGroup { | .icon_renoteGroup { | ||||||
| 	background: #36d298; | 	background: var(--eventRenote); | ||||||
| } | } | ||||||
|  |  | ||||||
| .icon_app { | .icon_app { | ||||||
| @@ -246,49 +253,49 @@ const rejectFollowRequest = () => { | |||||||
|  |  | ||||||
| .t_follow, .t_followRequestAccepted, .t_receiveFollowRequest { | .t_follow, .t_followRequestAccepted, .t_receiveFollowRequest { | ||||||
| 	padding: 3px; | 	padding: 3px; | ||||||
| 	background: #36aed2; | 	background: var(--eventFollow); | ||||||
| 	pointer-events: none; | 	pointer-events: none; | ||||||
| } | } | ||||||
|  |  | ||||||
| .t_renote { | .t_renote { | ||||||
| 	padding: 3px; | 	padding: 3px; | ||||||
| 	background: #36d298; | 	background: var(--eventRenote); | ||||||
| 	pointer-events: none; | 	pointer-events: none; | ||||||
| } | } | ||||||
|  |  | ||||||
| .t_quote { | .t_quote { | ||||||
| 	padding: 3px; | 	padding: 3px; | ||||||
| 	background: #36d298; | 	background: var(--eventRenote); | ||||||
| 	pointer-events: none; | 	pointer-events: none; | ||||||
| } | } | ||||||
|  |  | ||||||
| .t_reply { | .t_reply { | ||||||
| 	padding: 3px; | 	padding: 3px; | ||||||
| 	background: #007aff; | 	background: var(--eventReply); | ||||||
| 	pointer-events: none; | 	pointer-events: none; | ||||||
| } | } | ||||||
|  |  | ||||||
| .t_mention { | .t_mention { | ||||||
| 	padding: 3px; | 	padding: 3px; | ||||||
| 	background: #88a6b7; | 	background: var(--eventOther); | ||||||
| 	pointer-events: none; | 	pointer-events: none; | ||||||
| } | } | ||||||
|  |  | ||||||
| .t_pollEnded { | .t_pollEnded { | ||||||
| 	padding: 3px; | 	padding: 3px; | ||||||
| 	background: #88a6b7; | 	background: var(--eventOther); | ||||||
| 	pointer-events: none; | 	pointer-events: none; | ||||||
| } | } | ||||||
|  |  | ||||||
| .t_achievementEarned { | .t_achievementEarned { | ||||||
| 	padding: 3px; | 	padding: 3px; | ||||||
| 	background: #cb9a11; | 	background: var(--eventAchievement); | ||||||
| 	pointer-events: none; | 	pointer-events: none; | ||||||
| } | } | ||||||
|  |  | ||||||
| .t_roleAssigned { | .t_roleAssigned { | ||||||
| 	padding: 3px; | 	padding: 3px; | ||||||
| 	background: #88a6b7; | 	background: var(--eventOther); | ||||||
| 	pointer-events: none; | 	pointer-events: none; | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -63,6 +63,7 @@ const exampleNote = reactive<Misskey.entities.Note>({ | |||||||
| 	reactionAcceptance: null, | 	reactionAcceptance: null, | ||||||
| 	renoteCount: 0, | 	renoteCount: 0, | ||||||
| 	repliesCount: 1, | 	repliesCount: 1, | ||||||
|  | 	reactionCount: 0, | ||||||
| 	reactions: {}, | 	reactions: {}, | ||||||
| 	reactionEmojis: {}, | 	reactionEmojis: {}, | ||||||
| 	fileIds: [], | 	fileIds: [], | ||||||
|   | |||||||
| @@ -68,6 +68,7 @@ const exampleCWNote = reactive<Misskey.entities.Note>({ | |||||||
| 	reactionAcceptance: null, | 	reactionAcceptance: null, | ||||||
| 	renoteCount: 0, | 	renoteCount: 0, | ||||||
| 	repliesCount: 1, | 	repliesCount: 1, | ||||||
|  | 	reactionCount: 0, | ||||||
| 	reactions: {}, | 	reactions: {}, | ||||||
| 	reactionEmojis: {}, | 	reactionEmojis: {}, | ||||||
| 	fileIds: [], | 	fileIds: [], | ||||||
|   | |||||||
| @@ -58,6 +58,7 @@ const exampleNote = reactive<Misskey.entities.Note>({ | |||||||
| 	reactionAcceptance: null, | 	reactionAcceptance: null, | ||||||
| 	renoteCount: 0, | 	renoteCount: 0, | ||||||
| 	repliesCount: 1, | 	repliesCount: 1, | ||||||
|  | 	reactionCount: 0, | ||||||
| 	reactions: {}, | 	reactions: {}, | ||||||
| 	reactionEmojis: {}, | 	reactionEmojis: {}, | ||||||
| 	fileIds: ['0000000002'], | 	fileIds: ['0000000002'], | ||||||
|   | |||||||
| @@ -14,10 +14,20 @@ SPDX-License-Identifier: AGPL-3.0-only | |||||||
| 			[$style.form_vertical]: chosen.place === 'vertical', | 			[$style.form_vertical]: chosen.place === 'vertical', | ||||||
| 		}]" | 		}]" | ||||||
| 	> | 	> | ||||||
| 		<a :href="chosen.url" target="_blank" :class="$style.link"> | 		<component | ||||||
|  | 			:is="self ? 'MkA' : 'a'" | ||||||
|  | 			:class="$style.link" | ||||||
|  | 			v-bind="self ? { | ||||||
|  | 				to: chosen.url.substring(local.length), | ||||||
|  | 			} : { | ||||||
|  | 				href: chosen.url, | ||||||
|  | 				rel: 'nofollow noopener', | ||||||
|  | 				target: '_blank', | ||||||
|  | 			}" | ||||||
|  | 		> | ||||||
| 			<img :src="chosen.imageUrl" :class="$style.img"> | 			<img :src="chosen.imageUrl" :class="$style.img"> | ||||||
| 			<button class="_button" :class="$style.i" @click.prevent.stop="toggleMenu"><i :class="$style.iIcon" class="ti ti-info-circle"></i></button> | 			<button class="_button" :class="$style.i" @click.prevent.stop="toggleMenu"><i :class="$style.iIcon" class="ti ti-info-circle"></i></button> | ||||||
| 		</a> | 		</component> | ||||||
| 	</div> | 	</div> | ||||||
| 	<div v-else :class="$style.menu"> | 	<div v-else :class="$style.menu"> | ||||||
| 		<div :class="$style.menuContainer"> | 		<div :class="$style.menuContainer"> | ||||||
| @@ -32,10 +42,10 @@ SPDX-License-Identifier: AGPL-3.0-only | |||||||
| </template> | </template> | ||||||
|  |  | ||||||
| <script lang="ts" setup> | <script lang="ts" setup> | ||||||
| import { ref } from 'vue'; | import { ref, computed } from 'vue'; | ||||||
| import { i18n } from '@/i18n.js'; | import { i18n } from '@/i18n.js'; | ||||||
| import { instance } from '@/instance.js'; | import { instance } from '@/instance.js'; | ||||||
| import { host } from '@/config.js'; | import { url as local, host } from '@/config.js'; | ||||||
| import MkButton from '@/components/MkButton.vue'; | import MkButton from '@/components/MkButton.vue'; | ||||||
| import { defaultStore } from '@/store.js'; | import { defaultStore } from '@/store.js'; | ||||||
| import * as os from '@/os.js'; | import * as os from '@/os.js'; | ||||||
| @@ -96,6 +106,9 @@ const choseAd = (): Ad | null => { | |||||||
| }; | }; | ||||||
|  |  | ||||||
| const chosen = ref(choseAd()); | const chosen = ref(choseAd()); | ||||||
|  |  | ||||||
|  | const self = computed(() => chosen.value?.url.startsWith(local)); | ||||||
|  |  | ||||||
| const shouldHide = ref(!defaultStore.state.forceShowAds && $i && $i.policies.canHideAds && (props.specify == null)); | const shouldHide = ref(!defaultStore.state.forceShowAds && $i && $i.policies.canHideAds && (props.specify == null)); | ||||||
|  |  | ||||||
| function reduceFrequency(): void { | function reduceFrequency(): void { | ||||||
|   | |||||||
| @@ -35,6 +35,7 @@ export function useNoteCapture(props: { | |||||||
| 				const currentCount = (note.value.reactions || {})[reaction] || 0; | 				const currentCount = (note.value.reactions || {})[reaction] || 0; | ||||||
|  |  | ||||||
| 				note.value.reactions[reaction] = currentCount + 1; | 				note.value.reactions[reaction] = currentCount + 1; | ||||||
|  | 				note.value.reactionCount += 1; | ||||||
|  |  | ||||||
| 				if ($i && (body.userId === $i.id)) { | 				if ($i && (body.userId === $i.id)) { | ||||||
| 					note.value.myReaction = reaction; | 					note.value.myReaction = reaction; | ||||||
| @@ -49,6 +50,7 @@ export function useNoteCapture(props: { | |||||||
| 				const currentCount = (note.value.reactions || {})[reaction] || 0; | 				const currentCount = (note.value.reactions || {})[reaction] || 0; | ||||||
|  |  | ||||||
| 				note.value.reactions[reaction] = Math.max(0, currentCount - 1); | 				note.value.reactions[reaction] = Math.max(0, currentCount - 1); | ||||||
|  | 				note.value.reactionCount = Math.max(0, note.value.reactionCount - 1); | ||||||
| 				if (note.value.reactions[reaction] === 0) delete note.value.reactions[reaction]; | 				if (note.value.reactions[reaction] === 0) delete note.value.reactions[reaction]; | ||||||
|  |  | ||||||
| 				if ($i && (body.userId === $i.id)) { | 				if ($i && (body.userId === $i.id)) { | ||||||
|   | |||||||
| @@ -22,6 +22,13 @@ | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	//--ad: rgb(255 169 0 / 10%); | 	//--ad: rgb(255 169 0 / 10%); | ||||||
|  | 	--eventFollow: #36aed2; | ||||||
|  | 	--eventRenote: #36d298; | ||||||
|  | 	--eventReply: #007aff; | ||||||
|  | 	--eventReactionHeart: #dd2e44; | ||||||
|  | 	--eventReaction: #e99a0b; | ||||||
|  | 	--eventAchievement: #cb9a11; | ||||||
|  | 	--eventOther: #88a6b7; | ||||||
| } | } | ||||||
|  |  | ||||||
| ::selection { | ::selection { | ||||||
|   | |||||||
| @@ -3987,6 +3987,7 @@ export type components = { | |||||||
|       reactions: { |       reactions: { | ||||||
|         [key: string]: number; |         [key: string]: number; | ||||||
|       }; |       }; | ||||||
|  |       reactionCount: number; | ||||||
|       renoteCount: number; |       renoteCount: number; | ||||||
|       repliesCount: number; |       repliesCount: number; | ||||||
|       uri?: string; |       uri?: string; | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user
	 tamaina
					tamaina