Compare commits
	
		
			49 Commits
		
	
	
		
			2024.7.0-r
			...
			2024.8.0-a
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 2ab5ee81b1 | ||
|   | ef950a345b | ||
|   | bfaf938609 | ||
|   | d3cdc08802 | ||
|   | 571566d476 | ||
|   | 748a7e8f6a | ||
|   | 6db3c50e32 | ||
|   | 26322048db | ||
|   | a8810af8d9 | ||
|   | 45d88574c3 | ||
|   | b68b2ee8c6 | ||
|   | 86dd4abadc | ||
|   | cd210001e6 | ||
|   | 41936c16c4 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 4d757865f4 | ||
|   | 2a2bbcd1bc | ||
|   | 94b8c00c66 | ||
|   | ab7bbd4e57 | ||
|   | 93fc06d18b | ||
|   | 0aaf74ee22 | ||
|   | 046f2435b2 | ||
|   | 37c9d91ba0 | ||
|   | 93c569c2cd | ||
|   | cb10156f01 | ||
|   | 1532d5f390 | ||
|   | 7e3dedb045 | ||
|   | 01a815f8a7 | ||
|   | f50941389d | ||
|   | 0d508db8a7 | ||
|   | f244d42500 | ||
|   | 820becb4e4 | ||
|   | 6e3e7d7df1 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 008a66d73f | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 59e2e43a68 | ||
|   | 1a521a44c0 | ||
|   | d6ba12e24c | ||
|   | 4b04b2989b | ||
|   | d63b854f96 | ||
|   | 9dacc20d67 | ||
|   | 3137c104f2 | ||
|   | 63f9c271ca | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 400ae6ef01 | ||
|   | 8b163cd3fb | ||
|   | 676c599e48 | ||
|   | fccc5b6d62 | ||
|   | 0bb5ac0fca | ||
|   | c7354c5e30 | ||
|   | 5c42a0e439 | ||
|   | 8f40f932e4 | 
							
								
								
									
										2
									
								
								.github/workflows/release-edit-with-push.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/release-edit-with-push.yml
									
									
									
									
										vendored
									
									
								
							| @@ -6,7 +6,7 @@ on: | ||||
|       - develop | ||||
|     paths: | ||||
|       - 'CHANGELOG.md' | ||||
|       # - .github/workflows/release-edit-with-push.yml | ||||
|  | ||||
| env: | ||||
|   GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} | ||||
|  | ||||
|   | ||||
							
								
								
									
										8
									
								
								.github/workflows/release-with-dispatch.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										8
									
								
								.github/workflows/release-with-dispatch.yml
									
									
									
									
										vendored
									
									
								
							| @@ -17,6 +17,10 @@ on: | ||||
|         type: boolean | ||||
|         description: 'MERGE RELEASE BRANCH TO MAIN' | ||||
|         default: false | ||||
|       start-rc: | ||||
|         type: boolean | ||||
|         description: 'Start Release Candidate' | ||||
|         default: false | ||||
|  | ||||
| env: | ||||
|   GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} | ||||
| @@ -79,6 +83,9 @@ jobs: | ||||
|       package_jsons_to_rewrite: ${{ vars.PACKAGE_JSONS_TO_REWRITE }} | ||||
|       use_external_app_to_release: ${{ vars.USE_RELEASE_APP == 'true' }} | ||||
|       indent: ${{ vars.INDENT }} | ||||
|       draft_prerelease_channel: alpha | ||||
|       ready_start_prerelease_channel: beta | ||||
|       prerelease_channel: ${{ inputs.start-rc && 'rc' || '' }} | ||||
|     secrets: | ||||
|       RELEASE_APP_ID: ${{ secrets.RELEASE_APP_ID }} | ||||
|       RELEASE_APP_PRIVATE_KEY: ${{ secrets.RELEASE_APP_PRIVATE_KEY }} | ||||
| @@ -122,6 +129,7 @@ jobs: | ||||
|       use_external_app_to_release: ${{ vars.USE_RELEASE_APP == 'true' }} | ||||
|       indent: ${{ vars.INDENT }} | ||||
|       stable_branch: ${{ vars.STABLE_BRANCH }} | ||||
|       draft_prerelease_channel: alpha | ||||
|     secrets: | ||||
|       RELEASE_APP_ID: ${{ secrets.RELEASE_APP_ID }} | ||||
|       RELEASE_APP_PRIVATE_KEY: ${{ secrets.RELEASE_APP_PRIVATE_KEY }} | ||||
|   | ||||
							
								
								
									
										2
									
								
								.github/workflows/release-with-ready.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/release-with-ready.yml
									
									
									
									
										vendored
									
									
								
							| @@ -39,6 +39,8 @@ jobs: | ||||
|       package_jsons_to_rewrite: ${{ vars.PACKAGE_JSONS_TO_REWRITE }} | ||||
|       use_external_app_to_release: ${{ vars.USE_RELEASE_APP == 'true' }} | ||||
|       indent: ${{ vars.INDENT }} | ||||
|       draft_prerelease_channel: alpha | ||||
|       ready_start_prerelease_channel: beta | ||||
|     secrets: | ||||
|       RELEASE_APP_ID: ${{ secrets.RELEASE_APP_ID }} | ||||
|       RELEASE_APP_PRIVATE_KEY: ${{ secrets.RELEASE_APP_PRIVATE_KEY }} | ||||
|   | ||||
							
								
								
									
										5
									
								
								.github/workflows/storybook.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										5
									
								
								.github/workflows/storybook.yml
									
									
									
									
										vendored
									
									
								
							| @@ -7,6 +7,11 @@ on: | ||||
|       - develop | ||||
|       - dev/storybook8 # for testing | ||||
|   pull_request_target: | ||||
|     branches-ignore: | ||||
|       # Since pull requests targets master mostly is the "develop" branch. | ||||
|       # Storybook CI is checked on the "push" event of "develop" branch so it would cause a duplicate build. | ||||
|       # This is a waste of chromatic build quota, so we don't run storybook CI on pull requests targets master. | ||||
|       - master | ||||
|  | ||||
| jobs: | ||||
|   build: | ||||
|   | ||||
							
								
								
									
										2
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -35,6 +35,8 @@ coverage | ||||
| !/.config/example.yml | ||||
| !/.config/docker_example.yml | ||||
| !/.config/docker_example.env | ||||
| docker-compose.yml | ||||
| compose.yml | ||||
| .devcontainer/compose.yml | ||||
| !/.devcontainer/compose.yml | ||||
|  | ||||
|   | ||||
							
								
								
									
										39
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						
									
										39
									
								
								CHANGELOG.md
									
									
									
									
									
								
							| @@ -1,3 +1,33 @@ | ||||
| ## 2024.8.0 | ||||
|  | ||||
| ### General | ||||
| - Enhance: モデレーターはすべてのユーザーのフォロー・フォロワーの一覧を見られるように | ||||
| - Enhance: アカウントの削除のモデレーションログを残すように | ||||
| - Fix: リモートユーザのフォロー・フォロワーの一覧が非公開設定の場合も表示できてしまう問題を修正 | ||||
|  | ||||
| ### Client | ||||
| - Enhance: 「自分のPlay」ページにおいてPlayが非公開かどうかが一目でわかるように | ||||
| - Fix: Play編集時に公開範囲が「パブリック」にリセットされる問題を修正 | ||||
| - Fix: ページ遷移に失敗することがある問題を修正 | ||||
| - Fix: iOSでユーザー名などがリンクとして誤検知される現象を抑制 | ||||
| - Fix: mCaptchaを使用していてもbotプロテクションに関する警告が消えないのを修正 | ||||
| - Fix: ユーザーのモデレーションページにおいてユーザー名にドットが入っているとシステムアカウントとして表示されてしまう問題を修正 | ||||
|  | ||||
| ### Server | ||||
| - Enhance: 凍結されたアカウントのフォローリクエストを表示しないように | ||||
| - Fix: WSの`readAllNotifications` メッセージが `body` を持たない場合に動作しない問題 #14374 | ||||
|   - 通知ページや通知カラム(デッキ)を開いている状態において、新たに発生した通知が既読されない問題が修正されます。 | ||||
|   - これにより、プッシュ通知が有効な同条件下の環境において、プッシュ通知が常に発生してしまう問題も修正されます。 | ||||
| - Fix: Play各種エンドポイントの返り値に`visibility`が含まれていない問題を修正 | ||||
| - Fix: サーバー情報取得の際にモデレーター限定の情報が取得できないことがあるのを修正   | ||||
|   (Cherry-picked from https://activitypub.software/TransFem-org/Sharkey/-/merge_requests/582) | ||||
| - Fix: 公開範囲がダイレクトのノートをユーザーアクティビティのチャート生成に使用しないように   | ||||
|   (Cherry-picked from https://github.com/MisskeyIO/misskey/pull/679) | ||||
| - Fix: ActivityPubのエンティティタイプ判定で不明なタイプを受け取った場合でも処理を継続するように | ||||
|   - キュー処理のつまりが改善される可能性があります | ||||
| - Fix: リバーシの対局設定の変更が反映されないのを修正 | ||||
| - Fix: 無制限にストリーミングのチャンネルに接続できる問題を修正 | ||||
|  | ||||
| ## 2024.7.0 | ||||
|  | ||||
| ### Note | ||||
| @@ -9,6 +39,8 @@ | ||||
| - Feat: ユーザーのアイコン/バナーの変更可否をロールで設定可能に | ||||
|   - 変更不可となっていても、設定済みのものを解除してデフォルト画像に戻すことは出来ます | ||||
| - Feat: ユーザ作成時にSystemWebhookを送信可能に #14281 | ||||
| - Feat: メディアサイレンスを実装 #13842 | ||||
|   - メディアサイレンスされたサーバーに所属するアカウントによるファイルはすべてセンシティブとして扱われ、カスタム絵文字が使用できないようになります。 | ||||
| - Enhance: 管理画面でアーカイブにしたお知らせを表示・編集できるように | ||||
| - Fix: 配信停止したインスタンス一覧が見れなくなる問題を修正 | ||||
| - Fix: Dockerコンテナの立ち上げ時に`pnpm`のインストールで固まることがある問題 | ||||
| @@ -64,6 +96,11 @@ | ||||
| - Fix: 照会に `#` から始まる文字列を入力してそのハッシュタグのページを表示する際、入力が `#` のみの場合に「指定されたURLに該当するページはありませんでした。」が表示されてしまう問題を修正 | ||||
| - Fix: 照会に `@` から始まる文字列を入力してユーザーを照会する際、入力が `@` のみの場合に「問題が発生しました」が表示されてしまう問題を修正 | ||||
| - Fix: 投稿フォームにノートのURLを貼り付けて"引用として添付"した場合、投稿文を空にすることによるRenote化が出来なかった問題を修正 | ||||
| - Fix: フォロー中のユーザーに関する"TLに他の人への返信を含める"の設定が分かりづらい問題を修正 | ||||
| - Fix: タイムラインページを開いた時、`TLに他の人への返信を含める`がオフのときに`ファイル付きのみ`をオンにできない問題を修正 | ||||
| - Fix: deck uiでタイムラインを切り替えた際にTLの設定項目が更新されず、`TLに他の人への返信を含める`のトグルが表示されない問題を修正 | ||||
| - Fix: ウィジェットのタイムライン選択欄に無効化されたタイムラインが表示される問題を修正 | ||||
| - Fix: サウンドにドライブの音声を使用している際にドライブの音声が再生できなくなると設定が変更できなくなる問題を修正 | ||||
|  | ||||
| ### Server | ||||
| - Feat: レートリミット制限に引っかかったときに`Retry-After`ヘッダーを返すように (#13949) | ||||
| @@ -109,6 +146,8 @@ | ||||
|   - NOTE: `drive_file`の`url`, `uri`, `src`の上限が512から1024に変更されます | ||||
| 	  Migrationではカラム定義の変更のみが行われます。 | ||||
| 		サーバー管理者は各サーバーの必要に応じ`drive_file` `("uri")`に対するインデックスを張りなおすことでより安定しDBの探索が行われる可能性があります。詳細 は [GitHub](https://github.com/misskey-dev/misskey/pull/14323#issuecomment-2257562228)で確認可能です | ||||
| - Fix: 自分のフォロワー限定投稿に対するリプライがホームタイムラインで見えないことが有る問題を修正 | ||||
| - Fix: フォローしていないユーザによるフォロワー限定投稿に対するリプライがソーシャルタイムラインで表示されることがある問題を修正 | ||||
|  | ||||
| ### Misskey.js | ||||
| - Feat: `/drive/files/create` のリクエストに対応(`multipart/form-data`に対応) | ||||
|   | ||||
| @@ -2010,7 +2010,6 @@ _webhookSettings: | ||||
|   createWebhook: "Vytvořit Webhook" | ||||
|   name: "Jméno" | ||||
|   secret: "Tajné" | ||||
|   events: "Události Webhook" | ||||
|   active: "Zapnuto" | ||||
|   _events: | ||||
|     follow: "Při sledování uživatele" | ||||
|   | ||||
| @@ -2191,7 +2191,6 @@ _webhookSettings: | ||||
|   createWebhook: "Webhook erstellen" | ||||
|   name: "Name" | ||||
|   secret: "Secret" | ||||
|   events: "Webhook-Ereignisse" | ||||
|   active: "Aktiviert" | ||||
|   _events: | ||||
|     follow: "Wenn du jemandem folgst" | ||||
|   | ||||
| @@ -167,7 +167,7 @@ emojiUrl: "Emoji URL" | ||||
| addEmoji: "Add an emoji" | ||||
| settingGuide: "Recommended settings" | ||||
| cacheRemoteFiles: "Cache remote files" | ||||
| cacheRemoteFilesDescription: "When this setting is disabled, remote files are loaded directly from the remote instance. Disabling this will decrease storage usage, but increase traffic, as thumbnails will not be generated." | ||||
| cacheRemoteFilesDescription: "When this setting is disabled, remote files are loaded directly from the remote servers. Disabling this will decrease storage usage, but increase traffic, as thumbnails will not be generated." | ||||
| youCanCleanRemoteFilesCache: "You can clear the cache by clicking the 🗑️ button in the file management view." | ||||
| cacheRemoteSensitiveFiles: "Cache sensitive remote files" | ||||
| cacheRemoteSensitiveFilesDescription: "When this setting is disabled, sensitive remote files are loaded directly from the remote instance without caching." | ||||
| @@ -212,6 +212,7 @@ perDay: "Per Day" | ||||
| stopActivityDelivery: "Stop sending activities" | ||||
| blockThisInstance: "Block this instance" | ||||
| silenceThisInstance: "Silence this instance" | ||||
| mediaSilenceThisInstance: "Media-silence this server" | ||||
| operations: "Operations" | ||||
| software: "Software" | ||||
| version: "Version" | ||||
| @@ -232,7 +233,9 @@ clearCachedFilesConfirm: "Are you sure that you want to delete all cached remote | ||||
| blockedInstances: "Blocked Instances" | ||||
| blockedInstancesDescription: "List the hostnames of the instances you want to block separated by linebreaks. Listed instances will no longer be able to communicate with this instance." | ||||
| silencedInstances: "Silenced instances" | ||||
| silencedInstancesDescription: "List the hostnames of the instances that you want to silence. All accounts of the listed instances will be treated as silenced, can only make follow requests, and cannot mention local accounts if not followed. This will not affect blocked instances." | ||||
| silencedInstancesDescription: "List the host names of the servers that you want to silence, separated by a new line. All accounts belonging to the listed servers will be treated as silenced, and can only make follow requests, and cannot mention local accounts if not followed. This will not affect the blocked servers." | ||||
| mediaSilencedInstances: "Media-silenced servers" | ||||
| mediaSilencedInstancesDescription: "List the host names of the servers that you want to media-silence, separated by a new line. All accounts belonging to the listed servers will be treated as sensitive, and can't use custom emojis. This will not affect the blocked servers." | ||||
| muteAndBlock: "Mutes and Blocks" | ||||
| mutedUsers: "Muted users" | ||||
| blockedUsers: "Blocked users" | ||||
| @@ -395,7 +398,7 @@ mcaptcha: "mCaptcha" | ||||
| enableMcaptcha: "Enable mCaptcha" | ||||
| mcaptchaSiteKey: "Site key" | ||||
| mcaptchaSecretKey: "Secret key" | ||||
| mcaptchaInstanceUrl: "mCaptcha instance URL" | ||||
| mcaptchaInstanceUrl: "mCaptcha server URL" | ||||
| recaptcha: "reCAPTCHA" | ||||
| enableRecaptcha: "Enable reCAPTCHA" | ||||
| recaptchaSiteKey: "Site key" | ||||
| @@ -1121,6 +1124,8 @@ preventAiLearning: "Reject usage in Machine Learning (Generative AI)" | ||||
| preventAiLearningDescription: "Requests crawlers to not use posted text or image material etc. in machine learning (Predictive / Generative AI) data sets. This is achieved by adding a \"noai\" HTML-Response flag to the respective content. A complete prevention can however not be achieved through this flag, as it may simply be ignored." | ||||
| options: "Options" | ||||
| specifyUser: "Specific user" | ||||
| lookupConfirm: "Do you want to look up?" | ||||
| openTagPageConfirm: "Do you want to open a hashtag page?" | ||||
| specifyHost: "Specify a host" | ||||
| failedToPreviewUrl: "Could not preview" | ||||
| update: "Update" | ||||
| @@ -1250,7 +1255,7 @@ launchApp: "Launch the app" | ||||
| useNativeUIForVideoAudioPlayer: "Use UI of browser when play video and audio" | ||||
| keepOriginalFilename: "Keep original file name" | ||||
| keepOriginalFilenameDescription: "If you turn off this setting, files names will be replaced with random string automatically when you upload files." | ||||
| noDescription: "There is not the explanation" | ||||
| noDescription: "There is no explanation" | ||||
| alwaysConfirmFollow: "Always confirm when following" | ||||
| inquiry: "Contact" | ||||
| tryAgain: "Please try again later" | ||||
| @@ -1360,7 +1365,7 @@ _initialTutorial: | ||||
|       _exampleNote: | ||||
|         cw: "This will surely make you hungry!" | ||||
|         note: "Just had a chocolate-glazed donut 🍩😋" | ||||
|       useCases: "This is used when following the server guidelines for necessary notes or for self-restriction of spoiler or sensitive text." | ||||
|       useCases: "This is used when following the server guidelines, for necessary notes, or for self-restriction of spoiler or sensitive text." | ||||
|   _howToMakeAttachmentsSensitive: | ||||
|     title: "How to Mark Attachments as Sensitive?" | ||||
|     description: "For attachments that are required by server guidelines or that should not be left intact, add a \"sensitive\" flag." | ||||
| @@ -1962,6 +1967,7 @@ _soundSettings: | ||||
|   driveFileTypeWarnDescription: "Select an audio file" | ||||
|   driveFileDurationWarn: "The audio is too long." | ||||
|   driveFileDurationWarnDescription: "Long audio may disrupt using Misskey. Still continue?" | ||||
|   driveFileError: "It couldn't load the sound. Please change the setting." | ||||
| _ago: | ||||
|   future: "Future" | ||||
|   justNow: "Just now" | ||||
| @@ -2420,7 +2426,7 @@ _webhookSettings: | ||||
|   modifyWebhook: "Modify Webhook" | ||||
|   name: "Name" | ||||
|   secret: "Secret" | ||||
|   events: "Webhook Events" | ||||
|   trigger: "Trigger" | ||||
|   active: "Enabled" | ||||
|   _events: | ||||
|     follow: "When following a user" | ||||
| @@ -2488,7 +2494,7 @@ _moderationLogTypes: | ||||
|   unsetUserAvatar: "Unset this user's avatar" | ||||
|   unsetUserBanner: "Unset this user's banner" | ||||
|   createSystemWebhook: "Create SystemWebhook" | ||||
|   updateSystemWebhook: "Update SystemWebHook" | ||||
|   updateSystemWebhook: "Update SystemWebhook" | ||||
|   deleteSystemWebhook: "Delete SystemWebhook" | ||||
|   createAbuseReportNotificationRecipient: "Create a recipient for abuse reports" | ||||
|   updateAbuseReportNotificationRecipient: "Update recipients for abuse reports" | ||||
|   | ||||
| @@ -2382,7 +2382,6 @@ _webhookSettings: | ||||
|   createWebhook: "Crear Webhook" | ||||
|   name: "Nombre" | ||||
|   secret: "Secreto" | ||||
|   events: "Eventos de webhook" | ||||
|   active: "Activado" | ||||
|   _events: | ||||
|     follow: "Cuando se sigue a alguien" | ||||
|   | ||||
| @@ -1094,6 +1094,8 @@ preservedUsernames: "Noms d'utilisateur·rice réservés" | ||||
| preservedUsernamesDescription: "Énumérez les noms d'utilisateur à réserver, séparés par des nouvelles lignes. Les noms d'utilisateur spécifiés ici ne seront plus utilisables lors de la création d'un compte, sauf la création manuelle par un administrateur. De plus, les comptes existants ne seront pas affectés." | ||||
| createNoteFromTheFile: "Rédiger une note de ce fichier" | ||||
| archive: "Archive" | ||||
| archived: "Archivé" | ||||
| unarchive: "Annuler l'archivage" | ||||
| channelArchiveConfirmTitle: "Voulez-vous vraiment archiver {name} ?" | ||||
| channelArchiveConfirmDescription: "Une fois archivé, le canal n'apparaîtra plus dans la liste des canaux ni dans les résultats de recherche, et la publication des nouvelles notes sera impossible." | ||||
| thisChannelArchived: "Ce canal a été archivé." | ||||
| @@ -1224,7 +1226,10 @@ enableHorizontalSwipe: "Glisser pour changer d'onglet" | ||||
| loading: "Chargement en cours" | ||||
| surrender: "Annuler" | ||||
| gameRetry: "Réessayer" | ||||
| launchApp: "Lancer l'app" | ||||
| inquiry: "Contact" | ||||
| _delivery: | ||||
|   status: "Statut de la diffusion" | ||||
|   stop: "Suspendu·e" | ||||
|   _type: | ||||
|     none: "Publié" | ||||
|   | ||||
| @@ -2403,7 +2403,6 @@ _webhookSettings: | ||||
|   modifyWebhook: "Sunting Webhook" | ||||
|   name: "Nama" | ||||
|   secret: "Secret" | ||||
|   events: "Webhook Events" | ||||
|   active: "Aktif" | ||||
|   _events: | ||||
|     follow: "Ketika mengikuti pengguna" | ||||
|   | ||||
							
								
								
									
										28
									
								
								locales/index.d.ts
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										28
									
								
								locales/index.d.ts
									
									
									
									
										vendored
									
									
								
							| @@ -864,6 +864,10 @@ export interface Locale extends ILocale { | ||||
|      * サーバーをサイレンス | ||||
|      */ | ||||
|     "silenceThisInstance": string; | ||||
|     /** | ||||
|      * サーバーをメディアサイレンス | ||||
|      */ | ||||
|     "mediaSilenceThisInstance": string; | ||||
|     /** | ||||
|      * 操作 | ||||
|      */ | ||||
| @@ -948,6 +952,14 @@ export interface Locale extends ILocale { | ||||
|      * サイレンスしたいサーバーのホストを改行で区切って設定します。サイレンスされたサーバーに所属するアカウントはすべて「サイレンス」として扱われ、フォローがすべてリクエストになります。ブロックしたインスタンスには影響しません。 | ||||
|      */ | ||||
|     "silencedInstancesDescription": string; | ||||
|     /** | ||||
|      * メディアサイレンスしたサーバー | ||||
|      */ | ||||
|     "mediaSilencedInstances": string; | ||||
|     /** | ||||
|      * メディアサイレンスしたいサーバーのホストを改行で区切って設定します。メディアサイレンスされたサーバーに所属するアカウントによるファイルはすべてセンシティブとして扱われ、カスタム絵文字が使用できないようになります。ブロックしたインスタンスには影響しません。 | ||||
|      */ | ||||
|     "mediaSilencedInstancesDescription": string; | ||||
|     /** | ||||
|      * ミュートとブロック | ||||
|      */ | ||||
| @@ -7621,6 +7633,10 @@ export interface Locale extends ILocale { | ||||
|          * 長い音声を使用するとMisskeyの使用に支障をきたす可能性があります。それでも続行しますか? | ||||
|          */ | ||||
|         "driveFileDurationWarnDescription": string; | ||||
|         /** | ||||
|          * 音声が読み込めませんでした。設定を変更してください | ||||
|          */ | ||||
|         "driveFileError": string; | ||||
|     }; | ||||
|     "_ago": { | ||||
|         /** | ||||
| @@ -8969,6 +8985,10 @@ export interface Locale extends ILocale { | ||||
|          * ブロックを追加 | ||||
|          */ | ||||
|         "chooseBlock": string; | ||||
|         /** | ||||
|          * セクションタイトルを入力 | ||||
|          */ | ||||
|         "enterSectionTitle": string; | ||||
|         /** | ||||
|          * 種類を選択 | ||||
|          */ | ||||
| @@ -9386,9 +9406,9 @@ export interface Locale extends ILocale { | ||||
|          */ | ||||
|         "secret": string; | ||||
|         /** | ||||
|          * Webhookを実行するタイミング | ||||
|          * トリガー | ||||
|          */ | ||||
|         "events": string; | ||||
|         "trigger": string; | ||||
|         /** | ||||
|          * 有効 | ||||
|          */ | ||||
| @@ -9663,6 +9683,10 @@ export interface Locale extends ILocale { | ||||
|          * 通報の通知先を削除 | ||||
|          */ | ||||
|         "deleteAbuseReportNotificationRecipient": string; | ||||
|         /** | ||||
|          * アカウントを削除 | ||||
|          */ | ||||
|         "deleteAccount": string; | ||||
|     }; | ||||
|     "_fileViewer": { | ||||
|         /** | ||||
|   | ||||
| @@ -2412,7 +2412,6 @@ _webhookSettings: | ||||
|   modifyWebhook: "Modifica Webhook" | ||||
|   name: "Nome" | ||||
|   secret: "Segreto" | ||||
|   events: "Quando eseguire il Webhook" | ||||
|   active: "Attivo" | ||||
|   _events: | ||||
|     follow: "Quando segui un profilo" | ||||
|   | ||||
| @@ -212,6 +212,7 @@ perDay: "1日ごと" | ||||
| stopActivityDelivery: "アクティビティの配送を停止" | ||||
| blockThisInstance: "このサーバーをブロック" | ||||
| silenceThisInstance: "サーバーをサイレンス" | ||||
| mediaSilenceThisInstance: "サーバーをメディアサイレンス" | ||||
| operations: "操作" | ||||
| software: "ソフトウェア" | ||||
| version: "バージョン" | ||||
| @@ -233,6 +234,8 @@ blockedInstances: "ブロックしたサーバー" | ||||
| blockedInstancesDescription: "ブロックしたいサーバーのホストを改行で区切って設定します。ブロックされたサーバーは、このインスタンスとやり取りできなくなります。" | ||||
| silencedInstances: "サイレンスしたサーバー" | ||||
| silencedInstancesDescription: "サイレンスしたいサーバーのホストを改行で区切って設定します。サイレンスされたサーバーに所属するアカウントはすべて「サイレンス」として扱われ、フォローがすべてリクエストになります。ブロックしたインスタンスには影響しません。" | ||||
| mediaSilencedInstances: "メディアサイレンスしたサーバー" | ||||
| mediaSilencedInstancesDescription: "メディアサイレンスしたいサーバーのホストを改行で区切って設定します。メディアサイレンスされたサーバーに所属するアカウントによるファイルはすべてセンシティブとして扱われ、カスタム絵文字が使用できないようになります。ブロックしたインスタンスには影響しません。" | ||||
| muteAndBlock: "ミュートとブロック" | ||||
| mutedUsers: "ミュートしたユーザー" | ||||
| blockedUsers: "ブロックしたユーザー" | ||||
| @@ -1999,6 +2002,7 @@ _soundSettings: | ||||
|   driveFileTypeWarnDescription: "音声ファイルを選択してください" | ||||
|   driveFileDurationWarn: "音声が長すぎます" | ||||
|   driveFileDurationWarnDescription: "長い音声を使用するとMisskeyの使用に支障をきたす可能性があります。それでも続行しますか?" | ||||
|   driveFileError: "音声が読み込めませんでした。設定を変更してください" | ||||
|  | ||||
| _ago: | ||||
|   future: "未来" | ||||
| @@ -2367,6 +2371,7 @@ _pages: | ||||
|   eyeCatchingImageSet: "アイキャッチ画像を設定" | ||||
|   eyeCatchingImageRemove: "アイキャッチ画像を削除" | ||||
|   chooseBlock: "ブロックを追加" | ||||
|   enterSectionTitle: "セクションタイトルを入力" | ||||
|   selectType: "種類を選択" | ||||
|   contentBlocks: "コンテンツ" | ||||
|   inputBlocks: "入力" | ||||
| @@ -2488,7 +2493,7 @@ _webhookSettings: | ||||
|   modifyWebhook: "Webhookを編集" | ||||
|   name: "名前" | ||||
|   secret: "シークレット" | ||||
|   events: "Webhookを実行するタイミング" | ||||
|   trigger: "トリガー" | ||||
|   active: "有効" | ||||
|   _events: | ||||
|     follow: "フォローしたとき" | ||||
| @@ -2563,6 +2568,7 @@ _moderationLogTypes: | ||||
|   createAbuseReportNotificationRecipient: "通報の通知先を作成" | ||||
|   updateAbuseReportNotificationRecipient: "通報の通知先を更新" | ||||
|   deleteAbuseReportNotificationRecipient: "通報の通知先を削除" | ||||
|   deleteAccount: "アカウントを削除" | ||||
|  | ||||
| _fileViewer: | ||||
|   title: "ファイルの詳細" | ||||
|   | ||||
| @@ -60,6 +60,7 @@ copyFileId: "ファイルIDをコピー" | ||||
| copyFolderId: "フォルダーIDをコピー" | ||||
| copyProfileUrl: "プロフィールURLをコピー" | ||||
| searchUser: "ユーザーを探す" | ||||
| searchThisUsersNotes: "ユーザーのノートを検索" | ||||
| reply: "返事" | ||||
| loadMore: "まだまだあるで!" | ||||
| showMore: "まだまだあるで!" | ||||
| @@ -114,6 +115,8 @@ cantReRenote: "リノート自体はリノートできへんで。" | ||||
| quote: "引用" | ||||
| inChannelRenote: "チャンネルの中でリノート" | ||||
| inChannelQuote: "チャンネル内引用" | ||||
| renoteToChannel: "チャンネルにリノート" | ||||
| renoteToOtherChannel: "他のチャンネルにリノート" | ||||
| pinnedNote: "ピン留めされとるノート" | ||||
| pinned: "ピン留めしとく" | ||||
| you: "あんた" | ||||
| @@ -152,6 +155,7 @@ editList: "リストいじる" | ||||
| selectChannel: "チャンネルを選ぶ" | ||||
| selectAntenna: "アンテナを選ぶ" | ||||
| editAntenna: "アンテナいじる" | ||||
| createAntenna: "アンテナを作成" | ||||
| selectWidget: "ウィジェットを選ぶ" | ||||
| editWidgets: "ウィジェットをいじる" | ||||
| editWidgetsExit: "いじるのをやめる" | ||||
| @@ -178,6 +182,10 @@ addAccount: "アカウントを追加" | ||||
| reloadAccountsList: "アカウントリストの情報を更新" | ||||
| loginFailed: "ログインに失敗してもうた…" | ||||
| showOnRemote: "リモートで見る" | ||||
| continueOnRemote: "リモートで続行" | ||||
| chooseServerOnMisskeyHub: "Misskey Hubからサーバーを選択" | ||||
| specifyServerHost: "サーバーのドメインを直接指定" | ||||
| inputHostName: "ドメインを入力せえや" | ||||
| general: "全般" | ||||
| wallpaper: "壁紙" | ||||
| setWallpaper: "壁紙を設定" | ||||
| @@ -188,6 +196,7 @@ followConfirm: "{name}をフォローしてええか?" | ||||
| proxyAccount: "プロキシアカウント" | ||||
| proxyAccountDescription: "プロキシアカウントは、代わりにフォローしてくれるアカウントや。例えば、551に豚まんが無いときやったり、ユーザーがリモートユーザーをアカウントに入れたとき、リストに入れられたユーザーが誰からもフォローされてないと寂しいやん。寂しいし、アクティビティも配達されへんから、プロキシアカウントがフォローしてくれるで。ええやつやん…" | ||||
| host: "ホスト" | ||||
| selectSelf: "自分を選択" | ||||
| selectUser: "ユーザーを選ぶ" | ||||
| recipient: "宛先" | ||||
| annotation: "注釈" | ||||
| @@ -203,6 +212,7 @@ perDay: "1日ごと" | ||||
| stopActivityDelivery: "アクティビティの配送をやめる" | ||||
| blockThisInstance: "このサーバーをブロックすんで" | ||||
| silenceThisInstance: "サーバーサイレンスすんで?" | ||||
| mediaSilenceThisInstance: "サーバーをメディアサイレンス" | ||||
| operations: "操作" | ||||
| software: "ソフトウェア" | ||||
| version: "バージョン" | ||||
| @@ -224,6 +234,8 @@ blockedInstances: "ブロックしたサーバー" | ||||
| blockedInstancesDescription: "ブロックしたいサーバーのホストを改行で区切って設定してな。ブロックされてもうたサーバーとはもう金輪際やり取りできひんくなるで。" | ||||
| silencedInstances: "サーバーサイレンスされてんねん" | ||||
| silencedInstancesDescription: "サイレンスしたいサーバーのホストを改行で区切って設定すんで。サイレンスされたサーバーに所属するアカウントはすべて「サイレンス」として扱われ、フォローがすべてリクエストになり、フォロワーでないローカルアカウントにはメンションできなくなんねん。ブロックしたインスタンスには影響せーへんで。" | ||||
| mediaSilencedInstances: "メディアサイレンスしたサーバー" | ||||
| mediaSilencedInstancesDescription: "メディアサイレンスしたいサーバーのホストを改行で区切って設定するで。メディアサイレンスされたサーバーに所属するアカウントによるファイルはすべてセンシティブとして扱われてな、カスタム絵文字が使えへんようになるで。ブロックしたインスタンスには影響せえへんで。" | ||||
| muteAndBlock: "ミュートとブロック" | ||||
| mutedUsers: "ミュートしとるユーザー" | ||||
| blockedUsers: "ブロックしとるユーザー" | ||||
| @@ -475,6 +487,7 @@ noMessagesYet: "まだチャットはあらへんで" | ||||
| newMessageExists: "新しいメッセージがきたで" | ||||
| onlyOneFileCanBeAttached: "ごめんな、メッセージに添付できるファイルはひとつだけなんよ。" | ||||
| signinRequired: "ログインしてくれへん?" | ||||
| signinOrContinueOnRemote: "続行するには、お使いのサーバーに移動するか、このサーバーに登録・ログインする必要があるで" | ||||
| invitations: "来てや" | ||||
| invitationCode: "招待コード" | ||||
| checking: "確認しとるで" | ||||
| @@ -1025,6 +1038,7 @@ thisPostMayBeAnnoyingHome: "ホームに投稿" | ||||
| thisPostMayBeAnnoyingCancel: "やめとく" | ||||
| thisPostMayBeAnnoyingIgnore: "このまま投稿" | ||||
| collapseRenotes: "見たことあるリノートは飛ばして表示するで" | ||||
| collapseRenotesDescription: "リアクションやリノートをしたことがあるノートをたたんで表示するで。" | ||||
| internalServerError: "サーバー内部エラー" | ||||
| internalServerErrorDescription: "サーバーでなんか変なこと起こっとるわ。" | ||||
| copyErrorInfo: "エラー情報をコピるで" | ||||
| @@ -1098,6 +1112,8 @@ preservedUsernames: "予約ユーザー名" | ||||
| preservedUsernamesDescription: "予約しとくユーザー名を行ごとに挙げるで。ここで指定されたユーザー名はアカウント作るときに使えへんくなるけど、管理者は例外や。あと、もうあるアカウントも例外やな。" | ||||
| createNoteFromTheFile: "このファイル使うてノート作るで" | ||||
| archive: "アーカイブ" | ||||
| archived: "アーカイブ済み" | ||||
| unarchive: "アーカイブ解除" | ||||
| channelArchiveConfirmTitle: "{name}をアーカイブしてええか?" | ||||
| channelArchiveConfirmDescription: "アーカイブしたら、チャンネル一覧とか検索結果からなくなるし、新しく書き込みもできへんなるで。" | ||||
| thisChannelArchived: "このチャンネル、アーカイブされとるで。" | ||||
| @@ -1108,6 +1124,9 @@ preventAiLearning: "生成AIの学習に使わんといて" | ||||
| preventAiLearningDescription: "他の文章生成AIとか画像生成AIに、投稿したノートとか画像なんかを勝手に使わんように頼むで。具体的にはnoaiフラグをHTMLレスポンスに含めるんやけど、これ聞いてくれるんはAIの気分次第やから、使われる可能性もちょっとはあるな。" | ||||
| options: "オプション" | ||||
| specifyUser: "ユーザー指定" | ||||
| lookupConfirm: "照会するけどええか?" | ||||
| openTagPageConfirm: "ハッシュタグのページを開くんか?" | ||||
| specifyHost: "ホスト指定" | ||||
| failedToPreviewUrl: "プレビューできへん" | ||||
| update: "更新" | ||||
| rolesThatCanBeUsedThisEmojiAsReaction: "ツッコミとして使えるロール" | ||||
| @@ -1239,10 +1258,20 @@ keepOriginalFilenameDescription: "この設定をオフにすると、アップ | ||||
| noDescription: "説明文はあらへんで" | ||||
| alwaysConfirmFollow: "フォローの際常に確認する" | ||||
| inquiry: "問い合わせ" | ||||
| tryAgain: "もう一度試しいや。" | ||||
| confirmWhenRevealingSensitiveMedia: "センシティブなメディアを表示するとき確認する" | ||||
| sensitiveMediaRevealConfirm: "センシティブなメディアやで。表示するんか?" | ||||
| createdLists: "作成したリスト" | ||||
| createdAntennas: "作成したアンテナ" | ||||
| _delivery: | ||||
|   status: "配信状態" | ||||
|   stop: "配信せぇへん" | ||||
|   resume: "配信再開" | ||||
|   _type: | ||||
|     none: "配信しとる" | ||||
|     manuallySuspended: "手動停止中" | ||||
|     goneSuspended: "サーバー削除のため停止中" | ||||
|     autoSuspendedForNotResponding: "サーバー応答せえへんから停止中" | ||||
| _bubbleGame: | ||||
|   howToPlay: "遊び方" | ||||
|   hold: "ホールド" | ||||
| @@ -1368,6 +1397,8 @@ _serverSettings: | ||||
|   fanoutTimelineDescription: "入れると、おのおのタイムラインを取得するときにめちゃめちゃ動きが良うなって、データベースが軽くなるわ。でも、Redisのメモリ使う量が増えるから注意な。サーバーのメモリが足りんときとか、動きが変なときは切れるで。" | ||||
|   fanoutTimelineDbFallback: "データベースにフォールバックする" | ||||
|   fanoutTimelineDbFallbackDescription: "有効にしたら、タイムラインがキャッシュん中に入ってないときにDBにもっかい問い合わせるフォールバック処理ってのをやっとくで。切ったらフォールバック処理をやらんからサーバーはもっと軽くなんねんけど、タイムラインの取得範囲がちょっと減るで。" | ||||
|   inquiryUrl: "問い合わせ先URL" | ||||
|   inquiryUrlDescription: "サーバー運営者へのお問い合わせフォームのURLや、運営者の連絡先等が記載されたWebページのURLを指定するで。" | ||||
| _accountMigration: | ||||
|   moveFrom: "別のアカウントからこのアカウントに引っ越す" | ||||
|   moveFromSub: "別のアカウントへエイリアスを作る" | ||||
| @@ -1684,6 +1715,7 @@ _role: | ||||
|     canManageAvatarDecorations: "アバターを飾るモンの管理" | ||||
|     driveCapacity: "ドライブ容量" | ||||
|     alwaysMarkNsfw: "勝手にファイルにNSFWをくっつける" | ||||
|     canUpdateBioMedia: "アイコンとバナーの更新を許可" | ||||
|     pinMax: "ノートピン留めできる数" | ||||
|     antennaMax: "アンテナ作れる数" | ||||
|     wordMuteMax: "ワードミュートの最大文字数" | ||||
| @@ -1935,6 +1967,7 @@ _soundSettings: | ||||
|   driveFileTypeWarnDescription: "音声ファイルを選びや" | ||||
|   driveFileDurationWarn: "音が長すぎるわ" | ||||
|   driveFileDurationWarnDescription: "長い音使うたらMisskey使うのに良うないかもしれへんで。それでもええか?" | ||||
|   driveFileError: "音声が読み込めへんかったで。設定を変更せえや" | ||||
| _ago: | ||||
|   future: "未来" | ||||
|   justNow: "ついさっき" | ||||
| @@ -2351,6 +2384,7 @@ _deck: | ||||
|   alwaysShowMainColumn: "いつもメインカラムを表示" | ||||
|   columnAlign: "カラムの寄せ" | ||||
|   addColumn: "カラムを追加" | ||||
|   newNoteNotificationSettings: "新着ノート通知の設定" | ||||
|   configureColumn: "カラムの設定" | ||||
|   swapLeft: "左に移動" | ||||
|   swapRight: "右に移動" | ||||
| @@ -2389,9 +2423,10 @@ _drivecleaner: | ||||
|   orderByCreatedAtAsc: "追加日の古い順" | ||||
| _webhookSettings: | ||||
|   createWebhook: "Webhookをつくる" | ||||
|   modifyWebhook: "Webhookを編集" | ||||
|   name: "名前" | ||||
|   secret: "シークレット" | ||||
|   events: "Webhookを投げるタイミング" | ||||
|   trigger: "トリガー" | ||||
|   active: "有効" | ||||
|   _events: | ||||
|     follow: "フォローしたとき~!" | ||||
| @@ -2401,11 +2436,25 @@ _webhookSettings: | ||||
|     renote: "リノートされるとき~!" | ||||
|     reaction: "ツッコまれたとき~!" | ||||
|     mention: "メンションがあるとき~!" | ||||
|   _systemEvents: | ||||
|     abuseReport: "ユーザーから通報があったとき" | ||||
|     abuseReportResolved: "ユーザーからの通報を処理したとき" | ||||
|     userCreated: "ユーザーが作成されたとき" | ||||
|   deleteConfirm: "ほんまにWebhookをほかしてもええんか?" | ||||
| _abuseReport: | ||||
|   _notificationRecipient: | ||||
|     createRecipient: "通報の通知先を追加" | ||||
|     modifyRecipient: "通報の通知先を編集" | ||||
|     recipientType: "通知先の種類" | ||||
|     _recipientType: | ||||
|       mail: "メール" | ||||
|       webhook: "Webhook" | ||||
|       _captions: | ||||
|         mail: "モデレーター権限を持つユーザーのメアドに通知を送るで(通報を受けた時のみ)" | ||||
|         webhook: "指定したSystemWebhookに通知を送るで(通報を受けた時と通報を解決した時にそれぞれ発信)" | ||||
|     keywords: "キーワード" | ||||
|     notifiedUser: "通知先ユーザー" | ||||
|     notifiedWebhook: "使用するWebhook" | ||||
|     deleteConfirm: "通知先を削除してもええか?" | ||||
| _moderationLogTypes: | ||||
|   createRole: "ロールを追加すんで" | ||||
| @@ -2444,6 +2493,8 @@ _moderationLogTypes: | ||||
|   deleteAvatarDecoration: "アイコンデコレーションを削除" | ||||
|   unsetUserAvatar: "この子のアイコン元に戻す" | ||||
|   unsetUserBanner: "この子のバナー元に戻す" | ||||
|   createSystemWebhook: "SystemWebhookを作成" | ||||
|   updateSystemWebhook: "SystemWebhookを更新" | ||||
| _fileViewer: | ||||
|   title: "ファイルの詳しい情報" | ||||
|   type: "ファイルの種類" | ||||
|   | ||||
| @@ -2411,7 +2411,6 @@ _webhookSettings: | ||||
|   modifyWebhook: "Webhook 수정" | ||||
|   name: "이름" | ||||
|   secret: "시크릿" | ||||
|   events: "Webhook을 실행할 타이밍" | ||||
|   active: "활성화" | ||||
|   _events: | ||||
|     follow: "누군가를 팔로우했을 때" | ||||
|   | ||||
| @@ -1544,7 +1544,6 @@ _webhookSettings: | ||||
|   createWebhook: "Stwórz Webhook" | ||||
|   name: "Nazwa" | ||||
|   secret: "Sekret" | ||||
|   events: "Uruchomienie Webhooka" | ||||
|   active: "Właczono" | ||||
|   _events: | ||||
|     follow: "Po zaobserwowaniu użytkownika" | ||||
|   | ||||
							
								
								
									
										1250
									
								
								locales/pt-PT.yml
									
									
									
									
									
								
							
							
						
						
									
										1250
									
								
								locales/pt-PT.yml
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -60,6 +60,7 @@ copyFileId: "คัดลอกไฟล์ ID" | ||||
| copyFolderId: "คัดลอกโฟลเดอร์ ID" | ||||
| copyProfileUrl: "คัดลอกโปรไฟล์ URL" | ||||
| searchUser: "ค้นหาผู้ใช้" | ||||
| searchThisUsersNotes: "ค้นหาโน้ตของผู้ใช้" | ||||
| reply: "ตอบกลับ" | ||||
| loadMore: "แสดงเพิ่มเติม" | ||||
| showMore: "แสดงเพิ่มเติม" | ||||
| @@ -154,6 +155,7 @@ editList: "แก้ไขรายชื่อ" | ||||
| selectChannel: "เลือกช่อง" | ||||
| selectAntenna: "เลือกเสาอากาศ" | ||||
| editAntenna: "แก้ไขเสาอากาศ" | ||||
| createAntenna: "สร้างเสาอากาศ" | ||||
| selectWidget: "เลือกวิดเจ็ต" | ||||
| editWidgets: "แก้ไขวิดเจ็ต" | ||||
| editWidgetsExit: "เรียบร้อย" | ||||
| @@ -194,6 +196,7 @@ followConfirm: "ต้องการติดตาม {name} ใช่ไห | ||||
| proxyAccount: "บัญชีพร็อกซี่" | ||||
| proxyAccountDescription: "บัญชีพร็อกซี คือ บัญชีที่ทำหน้าที่ติดตาม(ผู้ใช้)ระยะไกลภายใต้เงื่อนไขบางประการ ตัวอย่างเช่น เมื่อผู้ใช้ท้องถิ่นเพิ่มผู้ใช้ระยะไกลลงรายชื่อ หากไม่มีใครติดตามผู้ใช้ระยะไกลในรายชื่อนั้น กิจกรรมก็จะไม่ถูกส่งมายังเซิร์ฟเวอร์ ดังนั้นจึงมีบัญชีพร็อกซีไว้ติดตามผู้ใช้ระยะไกลเหล่านั้น" | ||||
| host: "โฮสต์" | ||||
| selectSelf: "เลือกตัวเอง" | ||||
| selectUser: "เลือกผู้ใช้งาน" | ||||
| recipient: "ผู้รับ" | ||||
| annotation: "หมายเหตุประกอบ" | ||||
| @@ -209,6 +212,7 @@ perDay: "ต่อวัน" | ||||
| stopActivityDelivery: "หยุดส่งกิจกรรม" | ||||
| blockThisInstance: "บล็อกเซิร์ฟเวอร์นี้" | ||||
| silenceThisInstance: "ปิดปากเซิร์ฟเวอร์นี้" | ||||
| mediaSilenceThisInstance: "ปิดปากสื่อของเซิร์ฟเวอร์นี้" | ||||
| operations: "ดำเนินการ" | ||||
| software: "ซอฟต์แวร์" | ||||
| version: "เวอร์ชั่น" | ||||
| @@ -230,6 +234,8 @@ blockedInstances: "เซิร์ฟเวอร์ที่ถูกบล็ | ||||
| blockedInstancesDescription: "ระบุโฮสต์ของเซิร์ฟเวอร์ที่ต้องการบล็อก คั่นด้วยการขึ้นบรรทัดใหม่ เซิร์ฟเวอร์ที่ถูกบล็อกจะไม่สามารถติดต่อกับอินสแตนซ์นี้ได้" | ||||
| silencedInstances: "ปิดปากเซิร์ฟเวอร์นี้แล้ว" | ||||
| silencedInstancesDescription: "ระบุโฮสต์ของเซิร์ฟเวอร์ที่ต้องการปิดปาก คั่นด้วยการขึ้นบรรทัดใหม่, บัญชีทั้งหมดของเซิร์ฟเวอร์ดังกล่าวจะถือว่าถูกปิดปากเช่นกัน ทำได้เฉพาะคำขอติดตามเท่านั้น และไม่สามารถกล่าวถึงบัญชีในเซิร์ฟเวอร์นี้ได้หากไม่ได้ถูกติดตามกลับ | สิ่งนี้ไม่มีผลต่ออินสแตนซ์ที่ถูกบล็อก" | ||||
| mediaSilencedInstances: "เซิร์ฟเวอร์ที่ถูกปิดปากสื่อ" | ||||
| mediaSilencedInstancesDescription: "ระบุโฮสต์ของเซิร์ฟเวอร์ที่ต้องการปิดปากสื่อ คั่นด้วยการขึ้นบรรทัดใหม่, ไฟล์ที่ถูกส่งจากบัญชีของเซิร์ฟเวอร์ดังกล่าวจะถือว่าถูกปิดปาก แล้วจะถูกติดเครื่องหมายว่ามีเนื้อหาละเอียดอ่อน และเอโมจิแบบกำหนดเองก็จะใช้ไม่ได้ด้วย | สิ่งนี้ไม่มีผลต่ออินสแตนซ์ที่ถูกบล็อก" | ||||
| muteAndBlock: "ปิดเสียงและบล็อก" | ||||
| mutedUsers: "ผู้ใช้ที่ถูกปิดเสียง" | ||||
| blockedUsers: "ผู้ใช้ที่ถูกบล็อก" | ||||
| @@ -881,7 +887,7 @@ accountDeletionInProgress: "กำลังดำเนินการลบบ | ||||
| usernameInfo: "ชื่อที่ระบุบัญชีของคุณจากผู้อื่นในเซิร์ฟเวอร์นี้ คุณสามารถใช้ตัวอักษร (a~z, A~Z), ตัวเลข (0~9) หรือขีดล่าง (_) ชื่อผู้ใช้ไม่สามารถเปลี่ยนแปลงได้ในภายหลัง" | ||||
| aiChanMode: "โหมด Ai " | ||||
| devMode: "โหมดนักพัฒนา" | ||||
| keepCw: "เก็บคำเตือนเนื้อหา" | ||||
| keepCw: "คงการเตือนเนื้อหาไว้" | ||||
| pubSub: "บัญชี Pub/Sub" | ||||
| lastCommunication: "การสื่อสารครั้งสุดท้ายล่าสุด" | ||||
| resolved: "คลี่คลายแล้ว" | ||||
| @@ -1028,15 +1034,15 @@ achievements: "ความสำเร็จ" | ||||
| gotInvalidResponseError: "การตอบสนองเซิร์ฟเวอร์ไม่ถูกต้อง" | ||||
| gotInvalidResponseErrorDescription: "เซิร์ฟเวอร์อาจไม่สามารถเข้าถึงได้หรืออาจจะกำลังอยู่ในระหว่างปรับปรุง กรุณาลองใหม่อีกครั้งในภายหลังนะคะ" | ||||
| thisPostMayBeAnnoying: "โน้ตนี้อาจจะเป็นการรบกวนผู้อื่นนะคะ" | ||||
| thisPostMayBeAnnoyingHome: "โพสต์ไปยังไทม์ไลน์หลัก" | ||||
| thisPostMayBeAnnoyingCancel: "เลิก" | ||||
| thisPostMayBeAnnoyingIgnore: "โพสต์ยังไงก็แล้วแต่" | ||||
| thisPostMayBeAnnoyingHome: "โพสต์ลงไทม์ไลน์หลักเท่านั้น" | ||||
| thisPostMayBeAnnoyingCancel: "ยกเลิก" | ||||
| thisPostMayBeAnnoyingIgnore: "โพสต์ไปเลย ไม่ต้องปรับการมองเห็น" | ||||
| collapseRenotes: "ยุบรีโน้ตที่คุณเคยเห็นแล้ว" | ||||
| collapseRenotesDescription: "พับย่อโน้ตที่เคยตอบสนองหรือรีโน้ตแล้ว" | ||||
| internalServerError: "เซิร์ฟเวอร์ภายในเกิดข้อผิดพลาด" | ||||
| internalServerErrorDescription: "เกิดข้อผิดพลาดที่ไม่คาดคิดภายในเซิร์ฟเวอร์" | ||||
| copyErrorInfo: "คัดลอกรายละเอียดข้อผิดพลาด" | ||||
| joinThisServer: "ลงทะเบียนบนเซิร์ฟเวอร์นี้" | ||||
| joinThisServer: "ลงทะเบียนในเซิร์ฟเวอร์นี้" | ||||
| exploreOtherServers: "มองหาเซิร์ฟเวอร์อื่น" | ||||
| letsLookAtTimeline: "มาดูไทม์ไลน์กัน" | ||||
| disableFederationConfirm: "ปิดใช้งานสหพันธ์เลยใช่ไหม?" | ||||
| @@ -1099,13 +1105,15 @@ vertical: "แนวตั้ง" | ||||
| horizontal: "แนวนอน" | ||||
| position: "ตำแหน่ง" | ||||
| serverRules: "กฎของเซิร์ฟเวอร์" | ||||
| pleaseConfirmBelowBeforeSignup: "หากต้องการลงทะเบียนบนเซิร์ฟเวอร์นี้ คุณต้องตรวจสอบและยอมรับสิ่งต่อไปนี้" | ||||
| pleaseConfirmBelowBeforeSignup: "หากต้องการลงทะเบียนในเซิร์ฟเวอร์นี้ คุณต้องตรวจสอบและยอมรับสิ่งต่อไปนี้" | ||||
| pleaseAgreeAllToContinue: "คุณต้องยอมรับทุกช่องตรงด้านบนเพื่อดำเนินการต่อค่ะ" | ||||
| continue: "ดำเนินการต่อ" | ||||
| preservedUsernames: "ชื่อผู้ใช้ที่สงวนไว้" | ||||
| preservedUsernamesDescription: "ระบุชื่อผู้ใช้ที่จะสงวนชื่อไว้ คั่นด้วยการขึ้นบรรทัดใหม่ ชื่อผู้ใช้ที่ระบุที่นี่จะไม่สามารถใช้งานได้อีกต่อไปเมื่อสร้างบัญชีใหม่ ยกเว้นเมื่อผู้ดูแลระบบสร้างบัญชี นอกจากนี้ บัญชีที่มีอยู่แล้วจะไม่ได้รับผลกระทบ" | ||||
| createNoteFromTheFile: "เรียบเรียงโน้ตจากไฟล์นี้" | ||||
| archive: "เก็บถาวร" | ||||
| archived: "เก็บถาวรแล้ว" | ||||
| unarchive: "เลิกการเก็บถาวร" | ||||
| channelArchiveConfirmTitle: "ต้องการเก็บถาวรเจ้า {name} ใช่ไหม?" | ||||
| channelArchiveConfirmDescription: "เมื่อเก็บถาวรแล้ว จะไม่ปรากฏในรายการช่องหรือผลการค้นหาอีกต่อไป และจะไม่สามารถโพสต์ใหม่ได้อีกต่อไป" | ||||
| thisChannelArchived: "ช่องนี้ถูกเก็บถาวรแล้วนะ" | ||||
| @@ -1116,6 +1124,9 @@ preventAiLearning: "ปฏิเสธการเรียนรู้ด้ว | ||||
| preventAiLearningDescription: "ส่งคำร้องขอไม่ให้ใช้ ข้อความในโน้ตที่โพสต์, หรือเนื้อหารูปภาพ ฯลฯ ในการเรียนรู้ของเครื่อง(machine learning) / Predictive AI / Generative AI โดยการเพิ่มแฟล็ก “noai” ลง HTML-Response ให้กับเนื้อหาที่เกี่ยวข้อง แต่ทั้งนี้ ไม่ได้ป้องกัน AI จากการเรียนรู้ได้อย่างสมบูรณ์ เนื่องจากมี AI บางตัวเท่านั้นที่จะเคารพคำขอดังกล่าว" | ||||
| options: "ตัวเลือกบทบาท" | ||||
| specifyUser: "ผู้ใช้เฉพาะ" | ||||
| lookupConfirm: "ต้องการเรียกดูข้อมูลใช่ไหม?" | ||||
| openTagPageConfirm: "ต้องการเปิดหน้าแฮชแท็กใช่ไหม?" | ||||
| specifyHost: "ระบุโฮสต์" | ||||
| failedToPreviewUrl: "ไม่สามารถดูตัวอย่างได้" | ||||
| update: "อัปเดต" | ||||
| rolesThatCanBeUsedThisEmojiAsReaction: "บทบาทที่สามารถใช้เอโมจินี้เป็นรีแอคชั่นได้" | ||||
| @@ -1250,6 +1261,8 @@ inquiry: "ติดต่อเรา" | ||||
| tryAgain: "โปรดลองอีกครั้ง" | ||||
| confirmWhenRevealingSensitiveMedia: "ตรวจสอบก่อนแสดงสื่อที่มีเนื้อหาละเอียดอ่อน" | ||||
| sensitiveMediaRevealConfirm: "สื่อนี้มีเนื้อหาละเอียดอ่อน, ต้องการแสดงใช่ไหม?" | ||||
| createdLists: "รายชื่อที่ถูกสร้าง" | ||||
| createdAntennas: "เสาอากาศที่ถูกสร้าง" | ||||
| _delivery: | ||||
|   status: "สถานะการจัดส่ง" | ||||
|   stop: "ระงับการส่ง" | ||||
| @@ -1348,9 +1361,9 @@ _initialTutorial: | ||||
|       localOnly: "การโพสต์ด้วย flag นี้จะไม่รวมโน้ตไปยังเซิร์ฟเวอร์อื่น ผู้ใช้บนเซิร์ฟเวอร์อื่นจะไม่สามารถดูโน้ตเหล่านี้ได้โดยตรง โดยไม่คำนึงถึงการตั้งค่าการแสดงผลข้างต้น" | ||||
|     _cw: | ||||
|       title: "คำเตือนเกี่ยวกับเนื้อหา" | ||||
|       description: "เนื้อหาที่เขียนด้วย “คำอธิบายประกอบ” จะแสดงแทนข้อความหลัก คลิก “ดูเพิ่มเติม” เพื่อแสดงข้อความเต็ม" | ||||
|       description: "เนื้อหาที่เขียนใน “คำอธิบายประกอบ” จะแสดงแทนเนื้อหาหลัก ต้องคลิก “ดูเพิ่มเติม” เพื่อให้เนื้อหาหลักแสดง" | ||||
|       _exampleNote: | ||||
|         cw: "นี่อาจจะทำให้คุณหิวอย่างแน่นอน!" | ||||
|         cw: " ห้ามดู ระวังหิว" | ||||
|         note: "เพิ่งไปกินโดนัทเคลือบช็อคโกแลตมา 🍩😋" | ||||
|       useCases: "ใช้สิ่งนี้เพื่อระบุโน้ตที่ต้องตามแนวทางปฏิบัติของเซิร์ฟเวอร์ หรือเพื่อควบคุมการสปอยล์และข้อความที่ละเอียดอ่อนด้วยตนเอง" | ||||
|   _howToMakeAttachmentsSensitive: | ||||
| @@ -1466,15 +1479,15 @@ _achievements: | ||||
|       title: "มือใหม่ III" | ||||
|       description: "เข้าสู่ระบบเป็นเวลารวม 15 วัน" | ||||
|     _login30: | ||||
|       title: "มิสคิสท์ I" | ||||
|       title: "มิสคิสต์ I" | ||||
|       description: "เข้าสู่ระบบเป็นเวลารวม 30 วัน" | ||||
|     _login60: | ||||
|       title: "มิสคิสท์ II" | ||||
|       title: "มิสคิสต์ II" | ||||
|       description: "เข้าสู่ระบบเป็นเวลารวม 60 วัน" | ||||
|     _login100: | ||||
|       title: "มิสคิสท์ III" | ||||
|       title: "มิสคิสต์ III" | ||||
|       description: "เข้าสู่ระบบเป็นเวลารวม 100 วัน" | ||||
|       flavor: "มิสคิสต์หัวรุนแรง" | ||||
|       flavor: "Violent Misskist (ทำไมเหมือนชื่อหนังสักเรื่องจังเลยนะ)" | ||||
|     _login200: | ||||
|       title: "ลูกค้าประจำ I" | ||||
|       description: "เข้าสู่ระบบเป็นเวลารวม 200 วัน" | ||||
| @@ -1954,6 +1967,7 @@ _soundSettings: | ||||
|   driveFileTypeWarnDescription: "กรุณาเลือกไฟล์เสียง" | ||||
|   driveFileDurationWarn: "เสียงยาวเกินไป" | ||||
|   driveFileDurationWarnDescription: "การใช้เสียงที่ยาว อาจรบกวนการใช้งาน Misskey, ต้องการดำเนินการต่อใช่ไหม?" | ||||
|   driveFileError: "ไม่สามารถโหลดไฟล์เสียงได้ กรุณาเปลี่ยนแปลงการตั้งค่า" | ||||
| _ago: | ||||
|   future: "อนาคต" | ||||
|   justNow: "เมื่อกี๊นี้" | ||||
| @@ -2141,7 +2155,7 @@ _widgets: | ||||
|   serverMetric: "ตัวชี้วัดเซิร์ฟเวอร์" | ||||
|   aiscript: " คอนโซล AiScript" | ||||
|   aiscriptApp: "แอป AiScript" | ||||
|   aichan: "ไอ" | ||||
|   aichan: "藍 (ไอ)" | ||||
|   userList: "รายชื่อผู้ใช้" | ||||
|   _userList: | ||||
|     chooseList: "เลือกรายชื่อ" | ||||
| @@ -2183,7 +2197,7 @@ _visibility: | ||||
|   followersDescription: "เฉพาะผู้ติดตามเท่านั้นที่มองเห็นได้" | ||||
|   specified: "ไดเร็ค" | ||||
|   specifiedDescription: "ทำให้มองเห็นได้เฉพาะผู้ใช้ที่ระบุเท่านั้น" | ||||
|   disableFederation: "ไม่มีสหพันธ์" | ||||
|   disableFederation: "การปิดใช้งานสหพันธ์" | ||||
|   disableFederationDescription: "อย่าส่งข้อมูลไปยังเซิร์ฟเวอร์อื่น" | ||||
| _postForm: | ||||
|   replyPlaceholder: "ตอบกลับโน้ตนี้..." | ||||
| @@ -2412,7 +2426,7 @@ _webhookSettings: | ||||
|   modifyWebhook: "แก้ไข Webhook" | ||||
|   name: "ชื่อ" | ||||
|   secret: "ความลับ" | ||||
|   events: "อีเว้นท์ Webhook" | ||||
|   trigger: "ทริกเกอร์" | ||||
|   active: "เปิดใช้งาน" | ||||
|   _events: | ||||
|     follow: "เมื่อกำลังติดตามผู้ใช้" | ||||
| @@ -2536,7 +2550,7 @@ _externalResourceInstaller: | ||||
|       description: "เกิดปัญหาระหว่างการติดตั้งธีม กรุณาลองอีกครั้ง. รายละเอียดข้อผิดพลาดสามารถดูได้ในคอนโซล Javascript" | ||||
| _dataSaver: | ||||
|   _media: | ||||
|     title: "โหลดมีเดีย" | ||||
|     title: "โหลดสื่อ" | ||||
|     description: "กันไม่ให้ภาพและวิดีโอโหลดโดยอัตโนมัติ แตะรูปภาพ/วิดีโอที่ซ่อนอยู่เพื่อโหลด" | ||||
|   _avatar: | ||||
|     title: "รูปไอคอน" | ||||
| @@ -2616,3 +2630,8 @@ _mediaControls: | ||||
|   pip: "รูปภาพในรูปภาม" | ||||
|   playbackRate: "ความเร็วในการเล่น" | ||||
|   loop: "เล่นวนซ้ำ" | ||||
| _contextMenu: | ||||
|   title: "เมนูเนื้อหา" | ||||
|   app: "แอปพลิเคชัน" | ||||
|   appWithShift: "แอปฟลิเคชันด้วยปุ่มยกแคร่ (Shift)" | ||||
|   native: "UI ของเบราว์เซอร์" | ||||
|   | ||||
| @@ -1918,7 +1918,6 @@ _webhookSettings: | ||||
|   createWebhook: "Tạo Webhook" | ||||
|   name: "Tên" | ||||
|   secret: "Mã bí mật" | ||||
|   events: "Sự kiện Webhook" | ||||
|   active: "Đã bật" | ||||
|   _events: | ||||
|     reaction: "Khi nhận được sự kiện" | ||||
|   | ||||
| @@ -212,6 +212,7 @@ perDay: "每天" | ||||
| stopActivityDelivery: "停止发送活动" | ||||
| blockThisInstance: "阻止此服务器向本服务器推流" | ||||
| silenceThisInstance: "使服务器静音" | ||||
| mediaSilenceThisInstance: "隐藏此服务器的媒体文件" | ||||
| operations: "操作" | ||||
| software: "软件" | ||||
| version: "版本" | ||||
| @@ -230,9 +231,11 @@ clearQueueConfirmText: "未送达的帖子将不会被投递。 通常无需执 | ||||
| clearCachedFiles: "清除缓存" | ||||
| clearCachedFilesConfirm: "确定要清除所有缓存的远程文件?" | ||||
| blockedInstances: "被封锁的服务器" | ||||
| blockedInstancesDescription: "设定要封锁的服务器,以换行来进行分割。被封锁的服务器将无法与本服务器进行交换通讯。子域名也同样会被封锁。" | ||||
| blockedInstancesDescription: "设定要封锁的服务器,以换行分隔。被封锁的服务器将无法与本服务器进行交换通讯。子域名也同样会被封锁。" | ||||
| silencedInstances: "被静音的服务器" | ||||
| silencedInstancesDescription: "设置要静音的服务器,以换行符分隔。被静音的服务器内所有的账户将默认处于「静音」状态,仅能发送关注请求,并且在未关注状态下无法提及本地账户。被阻止的实例不受影响。" | ||||
| silencedInstancesDescription: "设置要静音的服务器,以换行分隔。被静音的服务器内所有的账户将默认处于「静音」状态,仅能发送关注请求,并且在未关注状态下无法提及本地账户。被阻止的实例不受影响。" | ||||
| mediaSilencedInstances: "已隐藏媒体文件的服务器" | ||||
| mediaSilencedInstancesDescription: "设置要隐藏媒体文件的服务器,以换行分隔。被设置为隐藏媒体文件服务器内所有账号的文件均按照「敏感内容」处理,且将无法使用自定义表情符号。被阻止的实例不受影响。" | ||||
| muteAndBlock: "静音/拉黑" | ||||
| mutedUsers: "已静音用户" | ||||
| blockedUsers: "已拉黑的用户" | ||||
| @@ -1121,6 +1124,8 @@ preventAiLearning: "拒绝接受生成式 AI 的学习" | ||||
| preventAiLearningDescription: "要求文章生成 AI 或图像生成 AI 不能够以发布的帖子和图像等内容作为学习对象。这是通过在 HTML 响应中包含 noai 标志来实现的,这不能完全阻止 AI 学习你的发布内容,并不是所有 AI 都会遵守这类请求。" | ||||
| options: "选项" | ||||
| specifyUser: "用户指定" | ||||
| lookupConfirm: "确定查询?" | ||||
| openTagPageConfirm: "确定打开话题标签页面?" | ||||
| specifyHost: "指定主机名" | ||||
| failedToPreviewUrl: "无法预览" | ||||
| update: "更新" | ||||
| @@ -1660,6 +1665,7 @@ _achievements: | ||||
|     _bubbleGameDoubleExplodingHead: | ||||
|       title: "两个🤯" | ||||
|       description: "你合成出了2个游戏里最大的Emoji" | ||||
|       flavor: "" | ||||
| _role: | ||||
|   new: "创建角色" | ||||
|   edit: "编辑角色" | ||||
| @@ -1961,6 +1967,7 @@ _soundSettings: | ||||
|   driveFileTypeWarnDescription: "请选择音频文件" | ||||
|   driveFileDurationWarn: "音频过长" | ||||
|   driveFileDurationWarnDescription: "使用长音频可能会影响 Misskey 的使用。即使这样也要继续吗?" | ||||
|   driveFileError: "无法读取声音。请更改设置。" | ||||
| _ago: | ||||
|   future: "未来" | ||||
|   justNow: "最近" | ||||
| @@ -2419,7 +2426,7 @@ _webhookSettings: | ||||
|   modifyWebhook: "编辑 webhook" | ||||
|   name: "名称" | ||||
|   secret: "密钥" | ||||
|   events: "何时运行 Webhook" | ||||
|   trigger: "触发器" | ||||
|   active: "已启用" | ||||
|   _events: | ||||
|     follow: "关注时" | ||||
|   | ||||
| @@ -212,6 +212,7 @@ perDay: "每日" | ||||
| stopActivityDelivery: "停止發送活動" | ||||
| blockThisInstance: "封鎖此伺服器" | ||||
| silenceThisInstance: "禁言此伺服器" | ||||
| mediaSilenceThisInstance: "將這個伺服器的媒體設為禁言" | ||||
| operations: "操作" | ||||
| software: "軟體" | ||||
| version: "版本" | ||||
| @@ -233,6 +234,8 @@ blockedInstances: "已封鎖的伺服器" | ||||
| blockedInstancesDescription: "請逐行輸入需要封鎖的伺服器。已封鎖的伺服器將無法與本伺服器進行通訊。" | ||||
| silencedInstances: "被禁言的伺服器" | ||||
| silencedInstancesDescription: "設定要禁言的伺服器主機名稱,以換行分隔。隸屬於禁言伺服器的所有帳戶都將被視為「禁言帳戶」,只能發出「追隨請求」,而且無法提及未追隨的本地帳戶。這不會影響已封鎖的實例。" | ||||
| mediaSilencedInstances: "媒體被禁言的伺服器" | ||||
| mediaSilencedInstancesDescription: "設定您想要對媒體設定禁言的伺服器,以換行符號區隔。來自被媒體禁言的伺服器所屬帳戶的所有檔案都會被視為敏感檔案,且自訂表情符號不能使用。被封鎖的伺服器不受影響。" | ||||
| muteAndBlock: "靜音和封鎖" | ||||
| mutedUsers: "被靜音的使用者" | ||||
| blockedUsers: "被封鎖的使用者" | ||||
| @@ -1121,6 +1124,8 @@ preventAiLearning: "拒絕接受生成式AI的訓練" | ||||
| preventAiLearningDescription: "要求站外生成式 AI 不使用您發佈的內容訓練模型。此功能會使伺服器於 HTML 回應新增「noai」標籤,而因為要視乎 AI 會否遵守該標籤,所以此功能無法完全阻止所有 AI 使用您的內容。" | ||||
| options: "選項" | ||||
| specifyUser: "指定使用者" | ||||
| lookupConfirm: "要查詢嗎?" | ||||
| openTagPageConfirm: "要開啟標籤的頁面嗎?" | ||||
| specifyHost: "指定主機" | ||||
| failedToPreviewUrl: "無法預覽" | ||||
| update: "更新" | ||||
| @@ -1962,6 +1967,7 @@ _soundSettings: | ||||
|   driveFileTypeWarnDescription: "請選擇音效檔案" | ||||
|   driveFileDurationWarn: "音效太長了" | ||||
|   driveFileDurationWarnDescription: "使用長音效檔可能會影響 Misskey 的使用體驗。仍要使用此檔案嗎?" | ||||
|   driveFileError: "無法載入語音。請變更設定" | ||||
| _ago: | ||||
|   future: "未來" | ||||
|   justNow: "剛剛" | ||||
| @@ -2420,7 +2426,7 @@ _webhookSettings: | ||||
|   modifyWebhook: "編輯 Webhook" | ||||
|   name: "名字" | ||||
|   secret: "密鑰" | ||||
|   events: "何時運行 Webhook" | ||||
|   trigger: "觸發器" | ||||
|   active: "已啟用" | ||||
|   _events: | ||||
|     follow: "當你追隨時" | ||||
| @@ -2433,6 +2439,7 @@ _webhookSettings: | ||||
|   _systemEvents: | ||||
|     abuseReport: "當使用者檢舉時" | ||||
|     abuseReportResolved: "當處理了使用者的檢舉時" | ||||
|     userCreated: "使用者被新增時" | ||||
|   deleteConfirm: "請問是否要刪除 Webhook?" | ||||
| _abuseReport: | ||||
|   _notificationRecipient: | ||||
| @@ -2626,4 +2633,5 @@ _mediaControls: | ||||
| _contextMenu: | ||||
|   title: "內容功能表" | ||||
|   app: "應用程式" | ||||
|   appWithShift: "Shift 鍵應用程式" | ||||
|   native: "瀏覽器的使用者介面" | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| { | ||||
| 	"name": "misskey", | ||||
| 	"version": "2024.7.0-rc.7", | ||||
| 	"version": "2024.8.0-alpha.1", | ||||
| 	"codename": "nasubi", | ||||
| 	"repository": { | ||||
| 		"type": "git", | ||||
| @@ -61,7 +61,7 @@ | ||||
| 		"glob": "11.0.0" | ||||
| 	}, | ||||
| 	"devDependencies": { | ||||
| 		"@misskey-dev/eslint-plugin": "2.0.2", | ||||
| 		"@misskey-dev/eslint-plugin": "2.0.3", | ||||
| 		"@types/node": "20.14.12", | ||||
| 		"@typescript-eslint/eslint-plugin": "7.17.0", | ||||
| 		"@typescript-eslint/parser": "7.17.0", | ||||
|   | ||||
| @@ -0,0 +1,16 @@ | ||||
| /* | ||||
|  * SPDX-FileCopyrightText: syuilo and misskey-project | ||||
|  * SPDX-License-Identifier: AGPL-3.0-only | ||||
|  */ | ||||
|  | ||||
| export class MediaSilenceForHosts1716197366117 { | ||||
|     name = 'MediaSilenceForHosts1716197366117' | ||||
|  | ||||
|     async up(queryRunner) { | ||||
|         await queryRunner.query(`ALTER TABLE "meta" ADD "mediaSilencedHosts" character varying(1024) array NOT NULL DEFAULT '{}'`); | ||||
|     } | ||||
|  | ||||
|     async down(queryRunner) { | ||||
|         await queryRunner.query(`ALTER TABLE "meta" DROP COLUMN "mediaSilencedHosts"`); | ||||
|     } | ||||
| } | ||||
| @@ -4,12 +4,15 @@ | ||||
|  */ | ||||
|  | ||||
| import { Inject, Injectable } from '@nestjs/common'; | ||||
| import type { UsersRepository } from '@/models/_.js'; | ||||
| import { Not, IsNull } from 'typeorm'; | ||||
| import type { FollowingsRepository, MiUser, UsersRepository } from '@/models/_.js'; | ||||
| import { QueueService } from '@/core/QueueService.js'; | ||||
| import { UserSuspendService } from '@/core/UserSuspendService.js'; | ||||
| import { DI } from '@/di-symbols.js'; | ||||
| import { bindThis } from '@/decorators.js'; | ||||
| import { GlobalEventService } from '@/core/GlobalEventService.js'; | ||||
| import { UserEntityService } from '@/core/entities/UserEntityService.js'; | ||||
| import { ApRendererService } from '@/core/activitypub/ApRendererService.js'; | ||||
| import { ModerationLogService } from '@/core/ModerationLogService.js'; | ||||
|  | ||||
| @Injectable() | ||||
| export class DeleteAccountService { | ||||
| @@ -17,9 +20,14 @@ export class DeleteAccountService { | ||||
| 		@Inject(DI.usersRepository) | ||||
| 		private usersRepository: UsersRepository, | ||||
|  | ||||
| 		private userSuspendService: UserSuspendService, | ||||
| 		@Inject(DI.followingsRepository) | ||||
| 		private followingsRepository: FollowingsRepository, | ||||
|  | ||||
| 		private userEntityService: UserEntityService, | ||||
| 		private apRendererService: ApRendererService, | ||||
| 		private queueService: QueueService, | ||||
| 		private globalEventService: GlobalEventService, | ||||
| 		private moderationLogService: ModerationLogService, | ||||
| 	) { | ||||
| 	} | ||||
|  | ||||
| @@ -27,16 +35,52 @@ export class DeleteAccountService { | ||||
| 	public async deleteAccount(user: { | ||||
| 		id: string; | ||||
| 		host: string | null; | ||||
| 	}): Promise<void> { | ||||
| 	}, moderator?: MiUser): Promise<void> { | ||||
| 		const _user = await this.usersRepository.findOneByOrFail({ id: user.id }); | ||||
| 		if (_user.isRoot) throw new Error('cannot delete a root account'); | ||||
|  | ||||
| 		if (moderator != null) { | ||||
| 			this.moderationLogService.log(moderator, 'deleteAccount', { | ||||
| 				userId: user.id, | ||||
| 				userUsername: _user.username, | ||||
| 				userHost: user.host, | ||||
| 			}); | ||||
| 		} | ||||
|  | ||||
| 		// 物理削除する前にDelete activityを送信する | ||||
| 		await this.userSuspendService.doPostSuspend(user).catch(e => {}); | ||||
| 		if (this.userEntityService.isLocalUser(user)) { | ||||
| 			// 知り得る全SharedInboxにDelete配信 | ||||
| 			const content = this.apRendererService.addContext(this.apRendererService.renderDelete(this.userEntityService.genLocalUserUri(user.id), user)); | ||||
|  | ||||
| 			const queue: string[] = []; | ||||
|  | ||||
| 			const followings = await this.followingsRepository.find({ | ||||
| 				where: [ | ||||
| 					{ followerSharedInbox: Not(IsNull()) }, | ||||
| 					{ followeeSharedInbox: Not(IsNull()) }, | ||||
| 				], | ||||
| 				select: ['followerSharedInbox', 'followeeSharedInbox'], | ||||
| 			}); | ||||
|  | ||||
| 			const inboxes = followings.map(x => x.followerSharedInbox ?? x.followeeSharedInbox); | ||||
|  | ||||
| 			for (const inbox of inboxes) { | ||||
| 				if (inbox != null && !queue.includes(inbox)) queue.push(inbox); | ||||
| 			} | ||||
|  | ||||
| 			for (const inbox of queue) { | ||||
| 				this.queueService.deliver(user, content, inbox, true); | ||||
| 			} | ||||
|  | ||||
| 			this.queueService.createDeleteAccountJob(user, { | ||||
| 				soft: false, | ||||
| 			}); | ||||
| 		} else { | ||||
| 			// リモートユーザーの削除は、完全にDBから物理削除してしまうと再度連合してきてアカウントが復活する可能性があるため、soft指定する | ||||
| 			this.queueService.createDeleteAccountJob(user, { | ||||
| 				soft: true, | ||||
| 			}); | ||||
| 		} | ||||
|  | ||||
| 		await this.usersRepository.update(user.id, { | ||||
| 			isDeleted: true, | ||||
|   | ||||
| @@ -43,6 +43,7 @@ import { RoleService } from '@/core/RoleService.js'; | ||||
| import { correctFilename } from '@/misc/correct-filename.js'; | ||||
| import { isMimeImage } from '@/misc/is-mime-image.js'; | ||||
| import { ModerationLogService } from '@/core/ModerationLogService.js'; | ||||
| import { UtilityService } from '@/core/UtilityService.js'; | ||||
|  | ||||
| type AddFileArgs = { | ||||
| 	/** User who wish to add file */ | ||||
| @@ -127,6 +128,7 @@ export class DriveService { | ||||
| 		private driveChart: DriveChart, | ||||
| 		private perUserDriveChart: PerUserDriveChart, | ||||
| 		private instanceChart: InstanceChart, | ||||
| 		private utilityService: UtilityService, | ||||
| 	) { | ||||
| 		const logger = new Logger('drive', 'blue'); | ||||
| 		this.registerLogger = logger.createSubLogger('register', 'yellow'); | ||||
| @@ -587,6 +589,7 @@ export class DriveService { | ||||
| 			sensitive ?? false | ||||
| 			: false; | ||||
|  | ||||
| 		if (user && this.utilityService.isMediaSilencedHost(instance.mediaSilencedHosts, user.host)) file.isSensitive = true; | ||||
| 		if (info.sensitive && profile!.autoSensitive) file.isSensitive = true; | ||||
| 		if (info.sensitive && instance.setSensitiveFlagAutomatically) file.isSensitive = true; | ||||
| 		if (userRoleNSFW) file.isSensitive = true; | ||||
|   | ||||
| @@ -15,7 +15,7 @@ import isSvg from 'is-svg'; | ||||
| import probeImageSize from 'probe-image-size'; | ||||
| import { type predictionType } from 'nsfwjs'; | ||||
| import { sharpBmp } from '@misskey-dev/sharp-read-bmp'; | ||||
| import { encode } from 'blurhash'; | ||||
| import * as blurhash from 'blurhash'; | ||||
| import { createTempDir } from '@/misc/create-temp.js'; | ||||
| import { AiService } from '@/core/AiService.js'; | ||||
| import { LoggerService } from '@/core/LoggerService.js'; | ||||
| @@ -452,7 +452,7 @@ export class FileInfoService { | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * Calculate average color of image | ||||
| 	 * Calculate blurhash string of image | ||||
| 	 */ | ||||
| 	@bindThis | ||||
| 	private getBlurhash(path: string, type: string): Promise<string> { | ||||
| @@ -467,7 +467,7 @@ export class FileInfoService { | ||||
| 					let hash; | ||||
|  | ||||
| 					try { | ||||
| 						hash = encode(new Uint8ClampedArray(buffer), info.width, info.height, 5, 5); | ||||
| 						hash = blurhash.encode(new Uint8ClampedArray(buffer), info.width, info.height, 5, 5); | ||||
| 					} catch (e) { | ||||
| 						return reject(e); | ||||
| 					} | ||||
|   | ||||
| @@ -9,7 +9,8 @@ import type { ModerationLogsRepository } from '@/models/_.js'; | ||||
| import type { MiUser } from '@/models/User.js'; | ||||
| import { IdService } from '@/core/IdService.js'; | ||||
| import { bindThis } from '@/decorators.js'; | ||||
| import { ModerationLogPayloads, moderationLogTypes } from '@/types.js'; | ||||
| import type { ModerationLogPayloads } from '@/types.js'; | ||||
| import { moderationLogTypes } from '@/types.js'; | ||||
|  | ||||
| @Injectable() | ||||
| export class ModerationLogService { | ||||
|   | ||||
| @@ -364,6 +364,9 @@ export class NoteCreateService implements OnApplicationShutdown { | ||||
| 			mentionedUsers = data.apMentions ?? await this.extractMentionedUsers(user, combinedTokens); | ||||
| 		} | ||||
|  | ||||
| 		// if the host is media-silenced, custom emojis are not allowed | ||||
| 		if (this.utilityService.isMediaSilencedHost(meta.mediaSilencedHosts, user.host)) emojis = []; | ||||
|  | ||||
| 		tags = tags.filter(tag => Array.from(tag).length <= 128).splice(0, 32); | ||||
|  | ||||
| 		if (data.reply && (user.id !== data.reply.userId) && !mentionedUsers.some(u => u.id === data.reply!.userId)) { | ||||
| @@ -506,7 +509,7 @@ export class NoteCreateService implements OnApplicationShutdown { | ||||
| 		const meta = await this.metaService.fetch(); | ||||
|  | ||||
| 		this.notesChart.update(note, true); | ||||
| 		if (meta.enableChartsForRemoteUser || (user.host == null)) { | ||||
| 		if (note.visibility !== 'specified' && (meta.enableChartsForRemoteUser || (user.host == null))) { | ||||
| 			this.perUserNotesChart.update(user, note, true); | ||||
| 		} | ||||
|  | ||||
|   | ||||
| @@ -92,7 +92,7 @@ export class NoteDeleteService { | ||||
| 				this.deliverToConcerned(user, note, content); | ||||
| 			} | ||||
|  | ||||
| 			// also deliever delete activity to cascaded notes | ||||
| 			// also deliver delete activity to cascaded notes | ||||
| 			const federatedLocalCascadingNotes = (cascadingNotes).filter(note => !note.localOnly && note.userHost == null); // filter out local-only notes | ||||
| 			for (const cascadingNote of federatedLocalCascadingNotes) { | ||||
| 				if (!cascadingNote.user) continue; | ||||
|   | ||||
| @@ -105,6 +105,8 @@ export class ReactionService { | ||||
|  | ||||
| 	@bindThis | ||||
| 	public async create(user: { id: MiUser['id']; host: MiUser['host']; isBot: MiUser['isBot'] }, note: MiNote, _reaction?: string | null) { | ||||
| 		const meta = await this.metaService.fetch(); | ||||
|  | ||||
| 		// Check blocking | ||||
| 		if (note.userId !== user.id) { | ||||
| 			const blocked = await this.userBlockingService.checkBlocked(note.userId, user.id); | ||||
| @@ -148,6 +150,11 @@ export class ReactionService { | ||||
| 						if ((note.reactionAcceptance === 'nonSensitiveOnly' || note.reactionAcceptance === 'nonSensitiveOnlyForLocalLikeOnlyForRemote') && emoji.isSensitive) { | ||||
| 							reaction = FALLBACK; | ||||
| 						} | ||||
|  | ||||
| 						// for media silenced host, custom emoji reactions are not allowed | ||||
| 						if (reacterHost != null && this.utilityService.isMediaSilencedHost(meta.mediaSilencedHosts, reacterHost)) { | ||||
| 							reaction = FALLBACK; | ||||
| 						} | ||||
| 					} else { | ||||
| 						// リアクションとして使う権限がない | ||||
| 						reaction = FALLBACK; | ||||
| @@ -220,8 +227,6 @@ export class ReactionService { | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		const meta = await this.metaService.fetch(); | ||||
|  | ||||
| 		if (meta.enableChartsForRemoteUser || (user.host == null)) { | ||||
| 			this.perUserReactionsChart.update(user, note); | ||||
| 		} | ||||
|   | ||||
| @@ -6,6 +6,7 @@ | ||||
| import { Inject, Injectable } from '@nestjs/common'; | ||||
| import * as Redis from 'ioredis'; | ||||
| import { ModuleRef } from '@nestjs/core'; | ||||
| import { reversiUpdateKeys } from 'misskey-js'; | ||||
| import * as Reversi from 'misskey-reversi'; | ||||
| import { IsNull, LessThan, MoreThan } from 'typeorm'; | ||||
| import type { | ||||
| @@ -399,7 +400,33 @@ export class ReversiService implements OnApplicationShutdown, OnModuleInit { | ||||
| 	} | ||||
|  | ||||
| 	@bindThis | ||||
| 	public async updateSettings(gameId: MiReversiGame['id'], user: MiUser, key: string, value: any) { | ||||
| 	public isValidReversiUpdateKey(key: unknown): key is typeof reversiUpdateKeys[number] { | ||||
| 		if (typeof key !== 'string') return false; | ||||
| 		return (reversiUpdateKeys as string[]).includes(key); | ||||
| 	} | ||||
|  | ||||
| 	@bindThis | ||||
| 	public isValidReversiUpdateValue<K extends typeof reversiUpdateKeys[number]>(key: K, value: unknown): value is MiReversiGame[K] { | ||||
| 		switch (key) { | ||||
| 			case 'map': | ||||
| 				return Array.isArray(value) && value.every(row => typeof row === 'string'); | ||||
| 			case 'bw': | ||||
| 				return typeof value === 'string' && ['random', '1', '2'].includes(value); | ||||
| 			case 'isLlotheo': | ||||
| 				return typeof value === 'boolean'; | ||||
| 			case 'canPutEverywhere': | ||||
| 				return typeof value === 'boolean'; | ||||
| 			case 'loopedBoard': | ||||
| 				return typeof value === 'boolean'; | ||||
| 			case 'timeLimitForEachTurn': | ||||
| 				return typeof value === 'number' && value >= 0; | ||||
| 			default: | ||||
| 				return false; | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	@bindThis | ||||
| 	public async updateSettings<K extends typeof reversiUpdateKeys[number]>(gameId: MiReversiGame['id'], user: MiUser, key: K, value: MiReversiGame[K]) { | ||||
| 		const game = await this.get(gameId); | ||||
| 		if (game == null) throw new Error('game not found'); | ||||
| 		if (game.isStarted) return; | ||||
| @@ -407,10 +434,6 @@ export class ReversiService implements OnApplicationShutdown, OnModuleInit { | ||||
| 		if ((game.user1Id === user.id) && game.user1Ready) return; | ||||
| 		if ((game.user2Id === user.id) && game.user2Ready) return; | ||||
|  | ||||
| 		if (!['map', 'bw', 'isLlotheo', 'canPutEverywhere', 'loopedBoard', 'timeLimitForEachTurn'].includes(key)) return; | ||||
|  | ||||
| 		// TODO: より厳格なバリデーション | ||||
|  | ||||
| 		const updatedGame = { | ||||
| 			...game, | ||||
| 			[key]: value, | ||||
|   | ||||
| @@ -5,7 +5,7 @@ | ||||
|  | ||||
| import { Inject, Injectable } from '@nestjs/common'; | ||||
| import { Not, IsNull } from 'typeorm'; | ||||
| import type { FollowingsRepository } from '@/models/_.js'; | ||||
| import type { FollowingsRepository, FollowRequestsRepository, UsersRepository } from '@/models/_.js'; | ||||
| import type { MiUser } from '@/models/User.js'; | ||||
| import { QueueService } from '@/core/QueueService.js'; | ||||
| import { GlobalEventService } from '@/core/GlobalEventService.js'; | ||||
| @@ -13,24 +13,75 @@ import { DI } from '@/di-symbols.js'; | ||||
| import { ApRendererService } from '@/core/activitypub/ApRendererService.js'; | ||||
| import { UserEntityService } from '@/core/entities/UserEntityService.js'; | ||||
| import { bindThis } from '@/decorators.js'; | ||||
| import { RelationshipJobData } from '@/queue/types.js'; | ||||
| import { ModerationLogService } from '@/core/ModerationLogService.js'; | ||||
|  | ||||
| @Injectable() | ||||
| export class UserSuspendService { | ||||
| 	constructor( | ||||
| 		@Inject(DI.usersRepository) | ||||
| 		private usersRepository: UsersRepository, | ||||
|  | ||||
| 		@Inject(DI.followingsRepository) | ||||
| 		private followingsRepository: FollowingsRepository, | ||||
|  | ||||
| 		@Inject(DI.followRequestsRepository) | ||||
| 		private followRequestsRepository: FollowRequestsRepository, | ||||
|  | ||||
| 		private userEntityService: UserEntityService, | ||||
| 		private queueService: QueueService, | ||||
| 		private globalEventService: GlobalEventService, | ||||
| 		private apRendererService: ApRendererService, | ||||
| 		private moderationLogService: ModerationLogService, | ||||
| 	) { | ||||
| 	} | ||||
|  | ||||
| 	@bindThis | ||||
| 	public async doPostSuspend(user: { id: MiUser['id']; host: MiUser['host'] }): Promise<void> { | ||||
| 	public async suspend(user: MiUser, moderator: MiUser): Promise<void> { | ||||
| 		await this.usersRepository.update(user.id, { | ||||
| 			isSuspended: true, | ||||
| 		}); | ||||
|  | ||||
| 		this.moderationLogService.log(moderator, 'suspend', { | ||||
| 			userId: user.id, | ||||
| 			userUsername: user.username, | ||||
| 			userHost: user.host, | ||||
| 		}); | ||||
|  | ||||
| 		(async () => { | ||||
| 			await this.postSuspend(user).catch(e => {}); | ||||
| 			await this.unFollowAll(user).catch(e => {}); | ||||
| 		})(); | ||||
| 	} | ||||
|  | ||||
| 	@bindThis | ||||
| 	public async unsuspend(user: MiUser, moderator: MiUser): Promise<void> { | ||||
| 		await this.usersRepository.update(user.id, { | ||||
| 			isSuspended: false, | ||||
| 		}); | ||||
|  | ||||
| 		this.moderationLogService.log(moderator, 'unsuspend', { | ||||
| 			userId: user.id, | ||||
| 			userUsername: user.username, | ||||
| 			userHost: user.host, | ||||
| 		}); | ||||
|  | ||||
| 		(async () => { | ||||
| 			await this.postUnsuspend(user).catch(e => {}); | ||||
| 		})(); | ||||
| 	} | ||||
|  | ||||
| 	@bindThis | ||||
| 	private async postSuspend(user: { id: MiUser['id']; host: MiUser['host'] }): Promise<void> { | ||||
| 		this.globalEventService.publishInternalEvent('userChangeSuspendedState', { id: user.id, isSuspended: true }); | ||||
|  | ||||
| 		this.followRequestsRepository.delete({ | ||||
| 			followeeId: user.id, | ||||
| 		}); | ||||
| 		this.followRequestsRepository.delete({ | ||||
| 			followerId: user.id, | ||||
| 		}); | ||||
|  | ||||
| 		if (this.userEntityService.isLocalUser(user)) { | ||||
| 			// 知り得る全SharedInboxにDelete配信 | ||||
| 			const content = this.apRendererService.addContext(this.apRendererService.renderDelete(this.userEntityService.genLocalUserUri(user.id), user)); | ||||
| @@ -58,7 +109,7 @@ export class UserSuspendService { | ||||
| 	} | ||||
|  | ||||
| 	@bindThis | ||||
| 	public async doPostUnsuspend(user: MiUser): Promise<void> { | ||||
| 	private async postUnsuspend(user: MiUser): Promise<void> { | ||||
| 		this.globalEventService.publishInternalEvent('userChangeSuspendedState', { id: user.id, isSuspended: false }); | ||||
|  | ||||
| 		if (this.userEntityService.isLocalUser(user)) { | ||||
| @@ -86,4 +137,26 @@ export class UserSuspendService { | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	@bindThis | ||||
| 	private async unFollowAll(follower: MiUser) { | ||||
| 		const followings = await this.followingsRepository.find({ | ||||
| 			where: { | ||||
| 				followerId: follower.id, | ||||
| 				followeeId: Not(IsNull()), | ||||
| 			}, | ||||
| 		}); | ||||
|  | ||||
| 		const jobs: RelationshipJobData[] = []; | ||||
| 		for (const following of followings) { | ||||
| 			if (following.followeeId && following.followerId) { | ||||
| 				jobs.push({ | ||||
| 					from: { id: following.followerId }, | ||||
| 					to: { id: following.followeeId }, | ||||
| 					silent: true, | ||||
| 				}); | ||||
| 			} | ||||
| 		} | ||||
| 		this.queueService.createUnfollowJob(jobs); | ||||
| 	} | ||||
| } | ||||
|   | ||||
| @@ -42,6 +42,12 @@ export class UtilityService { | ||||
| 		return silencedHosts.some(x => `.${host.toLowerCase()}`.endsWith(`.${x}`)); | ||||
| 	} | ||||
|  | ||||
| 	@bindThis | ||||
| 	public isMediaSilencedHost(silencedHosts: string[] | undefined, host: string | null): boolean { | ||||
| 		if (!silencedHosts || host == null) return false; | ||||
| 		return silencedHosts.some(x => host.toLowerCase() === x); | ||||
| 	} | ||||
|  | ||||
| 	@bindThis | ||||
| 	public concatNoteContentsForKeyWordCheck(content: { | ||||
| 		cw?: string | null; | ||||
|   | ||||
| @@ -78,9 +78,10 @@ export class ApNoteService { | ||||
| 	@bindThis | ||||
| 	public validateNote(object: IObject, uri: string): Error | null { | ||||
| 		const expectHost = this.utilityService.extractDbHost(uri); | ||||
| 		const apType = getApType(object); | ||||
|  | ||||
| 		if (!validPost.includes(getApType(object))) { | ||||
| 			return new IdentifiableError('d450b8a9-48e4-4dab-ae36-f4db763fda7c', `invalid Note: invalid object type ${getApType(object)}`); | ||||
| 		if (apType == null || !validPost.includes(apType)) { | ||||
| 			return new IdentifiableError('d450b8a9-48e4-4dab-ae36-f4db763fda7c', `invalid Note: invalid object type ${apType ?? 'undefined'}`); | ||||
| 		} | ||||
|  | ||||
| 		if (object.id && this.utilityService.extractDbHost(object.id) !== expectHost) { | ||||
|   | ||||
| @@ -48,7 +48,7 @@ import type { ApResolverService, Resolver } from '../ApResolverService.js'; | ||||
| import type { ApLoggerService } from '../ApLoggerService.js'; | ||||
| // eslint-disable-next-line @typescript-eslint/consistent-type-imports | ||||
| import type { ApImageService } from './ApImageService.js'; | ||||
| import type { IActor, IObject } from '../type.js'; | ||||
| import type { IActor, ICollection, IObject, IOrderedCollection } from '../type.js'; | ||||
|  | ||||
| const nameLength = 128; | ||||
| const summaryLength = 2048; | ||||
| @@ -296,6 +296,21 @@ export class ApPersonService implements OnModuleInit { | ||||
|  | ||||
| 		const isBot = getApType(object) === 'Service' || getApType(object) === 'Application'; | ||||
|  | ||||
| 		const [followingVisibility, followersVisibility] = await Promise.all( | ||||
| 			[ | ||||
| 				this.isPublicCollection(person.following, resolver), | ||||
| 				this.isPublicCollection(person.followers, resolver), | ||||
| 			].map((p): Promise<'public' | 'private'> => p | ||||
| 				.then(isPublic => isPublic ? 'public' : 'private') | ||||
| 				.catch(err => { | ||||
| 					if (!(err instanceof StatusError) || err.isRetryable) { | ||||
| 						this.logger.error('error occurred while fetching following/followers collection', { stack: err }); | ||||
| 					} | ||||
| 					return 'private'; | ||||
| 				}) | ||||
| 			) | ||||
| 		); | ||||
|  | ||||
| 		const bday = person['vcard:bday']?.match(/^\d{4}-\d{2}-\d{2}/); | ||||
|  | ||||
| 		const url = getOneApHrefNullable(person.url); | ||||
| @@ -357,6 +372,8 @@ export class ApPersonService implements OnModuleInit { | ||||
| 					description: _description, | ||||
| 					url, | ||||
| 					fields, | ||||
| 					followingVisibility, | ||||
| 					followersVisibility, | ||||
| 					birthday: bday?.[0] ?? null, | ||||
| 					location: person['vcard:Address'] ?? null, | ||||
| 					userHost: host, | ||||
| @@ -464,6 +481,23 @@ export class ApPersonService implements OnModuleInit { | ||||
|  | ||||
| 		const tags = extractApHashtags(person.tag).map(normalizeForSearch).splice(0, 32); | ||||
|  | ||||
| 		const [followingVisibility, followersVisibility] = await Promise.all( | ||||
| 			[ | ||||
| 				this.isPublicCollection(person.following, resolver), | ||||
| 				this.isPublicCollection(person.followers, resolver), | ||||
| 			].map((p): Promise<'public' | 'private' | undefined> => p | ||||
| 				.then(isPublic => isPublic ? 'public' : 'private') | ||||
| 				.catch(err => { | ||||
| 					if (!(err instanceof StatusError) || err.isRetryable) { | ||||
| 						this.logger.error('error occurred while fetching following/followers collection', { stack: err }); | ||||
| 						// Do not update the visibiility on transient errors. | ||||
| 						return undefined; | ||||
| 					} | ||||
| 					return 'private'; | ||||
| 				}) | ||||
| 			) | ||||
| 		); | ||||
|  | ||||
| 		const bday = person['vcard:bday']?.match(/^\d{4}-\d{2}-\d{2}/); | ||||
|  | ||||
| 		const url = getOneApHrefNullable(person.url); | ||||
| @@ -532,6 +566,8 @@ export class ApPersonService implements OnModuleInit { | ||||
| 			url, | ||||
| 			fields, | ||||
| 			description: _description, | ||||
| 			followingVisibility, | ||||
| 			followersVisibility, | ||||
| 			birthday: bday?.[0] ?? null, | ||||
| 			location: person['vcard:Address'] ?? null, | ||||
| 		}); | ||||
| @@ -703,4 +739,16 @@ export class ApPersonService implements OnModuleInit { | ||||
|  | ||||
| 		return 'ok'; | ||||
| 	} | ||||
|  | ||||
| 	@bindThis | ||||
| 	private async isPublicCollection(collection: string | ICollection | IOrderedCollection | undefined, resolver: Resolver): Promise<boolean> { | ||||
| 		if (collection) { | ||||
| 			const resolved = await resolver.resolveCollection(collection); | ||||
| 			if (resolved.first || (resolved as ICollection).items || (resolved as IOrderedCollection).orderedItems) { | ||||
| 				return true; | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		return false; | ||||
| 	} | ||||
| } | ||||
|   | ||||
| @@ -60,11 +60,14 @@ export function getApId(value: string | IObject): string { | ||||
|  | ||||
| /** | ||||
|  * Get ActivityStreams Object type | ||||
|  * | ||||
|  * タイプ判定ができなかった場合に、あえてエラーではなくnullを返すようにしている。 | ||||
|  * 詳細: https://github.com/misskey-dev/misskey/issues/14239 | ||||
|  */ | ||||
| export function getApType(value: IObject): string { | ||||
| export function getApType(value: IObject): string | null { | ||||
| 	if (typeof value.type === 'string') return value.type; | ||||
| 	if (Array.isArray(value.type) && typeof value.type[0] === 'string') return value.type[0]; | ||||
| 	throw new Error('cannot detect type'); | ||||
| 	return null; | ||||
| } | ||||
|  | ||||
| export function getOneApHrefNullable(value: ApObject | undefined): string | undefined { | ||||
| @@ -97,19 +100,23 @@ export interface IActivity extends IObject { | ||||
| export interface ICollection extends IObject { | ||||
| 	type: 'Collection'; | ||||
| 	totalItems: number; | ||||
| 	items: ApObject; | ||||
| 	first?: IObject | string; | ||||
| 	items?: ApObject; | ||||
| } | ||||
|  | ||||
| export interface IOrderedCollection extends IObject { | ||||
| 	type: 'OrderedCollection'; | ||||
| 	totalItems: number; | ||||
| 	orderedItems: ApObject; | ||||
| 	first?: IObject | string; | ||||
| 	orderedItems?: ApObject; | ||||
| } | ||||
|  | ||||
| export const validPost = ['Note', 'Question', 'Article', 'Audio', 'Document', 'Image', 'Page', 'Video', 'Event']; | ||||
|  | ||||
| export const isPost = (object: IObject): object is IPost => | ||||
| 	validPost.includes(getApType(object)); | ||||
| export const isPost = (object: IObject): object is IPost => { | ||||
| 	const type = getApType(object); | ||||
| 	return type != null && validPost.includes(type); | ||||
| }; | ||||
|  | ||||
| export interface IPost extends IObject { | ||||
| 	type: 'Note' | 'Question' | 'Article' | 'Audio' | 'Document' | 'Image' | 'Page' | 'Video' | 'Event'; | ||||
| @@ -156,8 +163,10 @@ export const isTombstone = (object: IObject): object is ITombstone => | ||||
|  | ||||
| export const validActor = ['Person', 'Service', 'Group', 'Organization', 'Application']; | ||||
|  | ||||
| export const isActor = (object: IObject): object is IActor => | ||||
| 	validActor.includes(getApType(object)); | ||||
| export const isActor = (object: IObject): object is IActor => { | ||||
| 	const type = getApType(object); | ||||
| 	return type != null && validActor.includes(type); | ||||
| }; | ||||
|  | ||||
| export interface IActor extends IObject { | ||||
| 	type: 'Person' | 'Service' | 'Organization' | 'Group' | 'Application'; | ||||
| @@ -240,12 +249,16 @@ export interface IKey extends IObject { | ||||
| 	publicKeyPem: string | Buffer; | ||||
| } | ||||
|  | ||||
| export const validDocumentTypes = ['Audio', 'Document', 'Image', 'Page', 'Video']; | ||||
|  | ||||
| export interface IApDocument extends IObject { | ||||
| 	type: 'Audio' | 'Document' | 'Image' | 'Page' | 'Video'; | ||||
| } | ||||
|  | ||||
| export const isDocument = (object: IObject): object is IApDocument => | ||||
| 	['Audio', 'Document', 'Image', 'Page', 'Video'].includes(getApType(object)); | ||||
| export const isDocument = (object: IObject): object is IApDocument => { | ||||
| 	const type = getApType(object); | ||||
| 	return type != null && validDocumentTypes.includes(type); | ||||
| }; | ||||
|  | ||||
| export interface IApImage extends IApDocument { | ||||
| 	type: 'Image'; | ||||
| @@ -323,7 +336,10 @@ export const isAccept = (object: IObject): object is IAccept => getApType(object | ||||
| export const isReject = (object: IObject): object is IReject => getApType(object) === 'Reject'; | ||||
| export const isAdd = (object: IObject): object is IAdd => getApType(object) === 'Add'; | ||||
| export const isRemove = (object: IObject): object is IRemove => getApType(object) === 'Remove'; | ||||
| export const isLike = (object: IObject): object is ILike => getApType(object) === 'Like' || getApType(object) === 'EmojiReaction' || getApType(object) === 'EmojiReact'; | ||||
| export const isLike = (object: IObject): object is ILike => { | ||||
| 	const type = getApType(object); | ||||
| 	return type != null && ['Like', 'EmojiReaction', 'EmojiReact'].includes(type); | ||||
| }; | ||||
| export const isAnnounce = (object: IObject): object is IAnnounce => getApType(object) === 'Announce'; | ||||
| export const isBlock = (object: IObject): object is IBlock => getApType(object) === 'Block'; | ||||
| export const isFlag = (object: IObject): object is IFlag => getApType(object) === 'Flag'; | ||||
|   | ||||
| @@ -49,6 +49,7 @@ export class FlashEntityService { | ||||
| 			title: flash.title, | ||||
| 			summary: flash.summary, | ||||
| 			script: flash.script, | ||||
| 			visibility: flash.visibility, | ||||
| 			likedCount: flash.likedCount, | ||||
| 			isLiked: meId ? await this.flashLikesRepository.exists({ where: { flashId: flash.id, userId: meId } }) : undefined, | ||||
| 		}); | ||||
|   | ||||
| @@ -50,6 +50,7 @@ export class InstanceEntityService { | ||||
| 			maintainerName: instance.maintainerName, | ||||
| 			maintainerEmail: instance.maintainerEmail, | ||||
| 			isSilenced: this.utilityService.isSilencedHost(meta.silencedHosts, instance.host), | ||||
| 			isMediaSilenced: this.utilityService.isMediaSilencedHost(meta.mediaSilencedHosts, instance.host), | ||||
| 			iconUrl: instance.iconUrl, | ||||
| 			faviconUrl: instance.faviconUrl, | ||||
| 			themeColor: instance.themeColor, | ||||
| @@ -62,8 +63,9 @@ export class InstanceEntityService { | ||||
| 	@bindThis | ||||
| 	public packMany( | ||||
| 		instances: MiInstance[], | ||||
| 		me?: { id: MiUser['id']; } | null | undefined, | ||||
| 	) { | ||||
| 		return Promise.all(instances.map(x => this.pack(x))); | ||||
| 		return Promise.all(instances.map(x => this.pack(x, me))); | ||||
| 	} | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -454,12 +454,12 @@ export class UserEntityService implements OnModuleInit { | ||||
| 		} | ||||
|  | ||||
| 		const followingCount = profile == null ? null : | ||||
| 			(profile.followingVisibility === 'public') || isMe ? user.followingCount : | ||||
| 			(profile.followingVisibility === 'public') || isMe || iAmModerator ? user.followingCount : | ||||
| 			(profile.followingVisibility === 'followers') && (relation && relation.isFollowing) ? user.followingCount : | ||||
| 			null; | ||||
|  | ||||
| 		const followersCount = profile == null ? null : | ||||
| 			(profile.followersVisibility === 'public') || isMe ? user.followersCount : | ||||
| 			(profile.followersVisibility === 'public') || isMe || iAmModerator ? user.followersCount : | ||||
| 			(profile.followersVisibility === 'followers') && (relation && relation.isFollowing) ? user.followersCount : | ||||
| 			null; | ||||
|  | ||||
|   | ||||
| @@ -6,3 +6,7 @@ | ||||
| export type JsonValue = JsonArray | JsonObject | string | number | boolean | null; | ||||
| export type JsonObject = {[K in string]?: JsonValue}; | ||||
| export type JsonArray = JsonValue[]; | ||||
|  | ||||
| export function isJsonObject(value: JsonValue | undefined): value is JsonObject { | ||||
| 	return typeof value === 'object' && value !== null && !Array.isArray(value); | ||||
| } | ||||
|   | ||||
| @@ -86,6 +86,11 @@ export class MiMeta { | ||||
| 	}) | ||||
| 	public silencedHosts: string[]; | ||||
|  | ||||
| 	@Column('varchar', { | ||||
| 		length: 1024, array: true, default: '{}', | ||||
| 	}) | ||||
| 	public mediaSilencedHosts: string[]; | ||||
|  | ||||
| 	@Column('varchar', { | ||||
| 		length: 1024, | ||||
| 		nullable: true, | ||||
|   | ||||
| @@ -88,6 +88,10 @@ export const packedFederationInstanceSchema = { | ||||
| 			type: 'boolean', | ||||
| 			optional: false, nullable: false, | ||||
| 		}, | ||||
| 		isMediaSilenced: { | ||||
| 			type: 'boolean', | ||||
| 			optional: false, nullable: false, | ||||
| 		}, | ||||
| 		iconUrl: { | ||||
| 			type: 'string', | ||||
| 			optional: false, nullable: true, | ||||
|   | ||||
| @@ -44,6 +44,11 @@ export const packedFlashSchema = { | ||||
| 			type: 'string', | ||||
| 			optional: false, nullable: false, | ||||
| 		}, | ||||
| 		visibility: { | ||||
| 			type: 'string', | ||||
| 			optional: false, nullable: false, | ||||
| 			enum: ['private', 'public'], | ||||
| 		}, | ||||
| 		likedCount: { | ||||
| 			type: 'number', | ||||
| 			optional: false, nullable: true, | ||||
|   | ||||
| @@ -7,9 +7,9 @@ import { Inject, Injectable } from '@nestjs/common'; | ||||
| import { Endpoint } from '@/server/api/endpoint-base.js'; | ||||
| import type { UsersRepository } from '@/models/_.js'; | ||||
| import { QueueService } from '@/core/QueueService.js'; | ||||
| import { UserSuspendService } from '@/core/UserSuspendService.js'; | ||||
| import { DI } from '@/di-symbols.js'; | ||||
| import { UserEntityService } from '@/core/entities/UserEntityService.js'; | ||||
| import { DeleteAccountService } from '@/core/DeleteAccountService.js'; | ||||
|  | ||||
| export const meta = { | ||||
| 	tags: ['admin'], | ||||
| @@ -33,9 +33,7 @@ export default class extends Endpoint<typeof meta, typeof paramDef> { // eslint- | ||||
| 		@Inject(DI.usersRepository) | ||||
| 		private usersRepository: UsersRepository, | ||||
|  | ||||
| 		private userEntityService: UserEntityService, | ||||
| 		private queueService: QueueService, | ||||
| 		private userSuspendService: UserSuspendService, | ||||
| 		private deleteAccoountService: DeleteAccountService, | ||||
| 	) { | ||||
| 		super(meta, paramDef, async (ps, me) => { | ||||
| 			const user = await this.usersRepository.findOneBy({ id: ps.userId }); | ||||
| @@ -48,22 +46,7 @@ export default class extends Endpoint<typeof meta, typeof paramDef> { // eslint- | ||||
| 				throw new Error('cannot delete a root account'); | ||||
| 			} | ||||
|  | ||||
| 			if (this.userEntityService.isLocalUser(user)) { | ||||
| 				// 物理削除する前にDelete activityを送信する | ||||
| 				await this.userSuspendService.doPostSuspend(user).catch(err => {}); | ||||
|  | ||||
| 				this.queueService.createDeleteAccountJob(user, { | ||||
| 					soft: false, | ||||
| 				}); | ||||
| 			} else { | ||||
| 				this.queueService.createDeleteAccountJob(user, { | ||||
| 					soft: true, // リモートユーザーの削除は、完全にDBから物理削除してしまうと再度連合してきてアカウントが復活する可能性があるため、soft指定する | ||||
| 				}); | ||||
| 			} | ||||
|  | ||||
| 			await this.usersRepository.update(user.id, { | ||||
| 				isDeleted: true, | ||||
| 			}); | ||||
| 			await this.deleteAccoountService.deleteAccount(user); | ||||
| 		}); | ||||
| 	} | ||||
| } | ||||
|   | ||||
| @@ -128,6 +128,16 @@ export const meta = { | ||||
| 					nullable: false, | ||||
| 				}, | ||||
| 			}, | ||||
| 			mediaSilencedHosts: { | ||||
| 				type: 'array', | ||||
| 				optional: false, | ||||
| 				nullable: false, | ||||
| 				items: { | ||||
| 					type: 'string', | ||||
| 					optional: false, | ||||
| 					nullable: false, | ||||
| 				}, | ||||
| 			}, | ||||
| 			pinnedUsers: { | ||||
| 				type: 'array', | ||||
| 				optional: false, nullable: false, | ||||
| @@ -552,6 +562,7 @@ export default class extends Endpoint<typeof meta, typeof paramDef> { // eslint- | ||||
| 				hiddenTags: instance.hiddenTags, | ||||
| 				blockedHosts: instance.blockedHosts, | ||||
| 				silencedHosts: instance.silencedHosts, | ||||
| 				mediaSilencedHosts: instance.mediaSilencedHosts, | ||||
| 				sensitiveWords: instance.sensitiveWords, | ||||
| 				prohibitedWords: instance.prohibitedWords, | ||||
| 				preservedUsernames: instance.preservedUsernames, | ||||
|   | ||||
| @@ -3,18 +3,12 @@ | ||||
|  * SPDX-License-Identifier: AGPL-3.0-only | ||||
|  */ | ||||
|  | ||||
| import { IsNull, Not } from 'typeorm'; | ||||
| import { Inject, Injectable } from '@nestjs/common'; | ||||
| import { Endpoint } from '@/server/api/endpoint-base.js'; | ||||
| import type { UsersRepository, FollowingsRepository } from '@/models/_.js'; | ||||
| import type { MiUser } from '@/models/User.js'; | ||||
| import type { RelationshipJobData } from '@/queue/types.js'; | ||||
| import { ModerationLogService } from '@/core/ModerationLogService.js'; | ||||
| import type { UsersRepository } from '@/models/_.js'; | ||||
| import { UserSuspendService } from '@/core/UserSuspendService.js'; | ||||
| import { DI } from '@/di-symbols.js'; | ||||
| import { bindThis } from '@/decorators.js'; | ||||
| import { RoleService } from '@/core/RoleService.js'; | ||||
| import { QueueService } from '@/core/QueueService.js'; | ||||
|  | ||||
| export const meta = { | ||||
| 	tags: ['admin'], | ||||
| @@ -38,13 +32,8 @@ export default class extends Endpoint<typeof meta, typeof paramDef> { // eslint- | ||||
| 		@Inject(DI.usersRepository) | ||||
| 		private usersRepository: UsersRepository, | ||||
|  | ||||
| 		@Inject(DI.followingsRepository) | ||||
| 		private followingsRepository: FollowingsRepository, | ||||
|  | ||||
| 		private userSuspendService: UserSuspendService, | ||||
| 		private roleService: RoleService, | ||||
| 		private moderationLogService: ModerationLogService, | ||||
| 		private queueService: QueueService, | ||||
| 	) { | ||||
| 		super(meta, paramDef, async (ps, me) => { | ||||
| 			const user = await this.usersRepository.findOneBy({ id: ps.userId }); | ||||
| @@ -57,42 +46,7 @@ export default class extends Endpoint<typeof meta, typeof paramDef> { // eslint- | ||||
| 				throw new Error('cannot suspend moderator account'); | ||||
| 			} | ||||
|  | ||||
| 			await this.usersRepository.update(user.id, { | ||||
| 				isSuspended: true, | ||||
| 			await this.userSuspendService.suspend(user, me); | ||||
| 		}); | ||||
|  | ||||
| 			this.moderationLogService.log(me, 'suspend', { | ||||
| 				userId: user.id, | ||||
| 				userUsername: user.username, | ||||
| 				userHost: user.host, | ||||
| 			}); | ||||
|  | ||||
| 			(async () => { | ||||
| 				await this.userSuspendService.doPostSuspend(user).catch(e => {}); | ||||
| 				await this.unFollowAll(user).catch(e => {}); | ||||
| 			})(); | ||||
| 		}); | ||||
| 	} | ||||
|  | ||||
| 	@bindThis | ||||
| 	private async unFollowAll(follower: MiUser) { | ||||
| 		const followings = await this.followingsRepository.find({ | ||||
| 			where: { | ||||
| 				followerId: follower.id, | ||||
| 				followeeId: Not(IsNull()), | ||||
| 			}, | ||||
| 		}); | ||||
|  | ||||
| 		const jobs: RelationshipJobData[] = []; | ||||
| 		for (const following of followings) { | ||||
| 			if (following.followeeId && following.followerId) { | ||||
| 				jobs.push({ | ||||
| 					from: { id: following.followerId }, | ||||
| 					to: { id: following.followeeId }, | ||||
| 					silent: true, | ||||
| 				}); | ||||
| 			} | ||||
| 		} | ||||
| 		this.queueService.createUnfollowJob(jobs); | ||||
| 	} | ||||
| } | ||||
|   | ||||
| @@ -6,7 +6,6 @@ | ||||
| import { Inject, Injectable } from '@nestjs/common'; | ||||
| import { Endpoint } from '@/server/api/endpoint-base.js'; | ||||
| import type { UsersRepository } from '@/models/_.js'; | ||||
| import { ModerationLogService } from '@/core/ModerationLogService.js'; | ||||
| import { UserSuspendService } from '@/core/UserSuspendService.js'; | ||||
| import { DI } from '@/di-symbols.js'; | ||||
|  | ||||
| @@ -33,7 +32,6 @@ export default class extends Endpoint<typeof meta, typeof paramDef> { // eslint- | ||||
| 		private usersRepository: UsersRepository, | ||||
|  | ||||
| 		private userSuspendService: UserSuspendService, | ||||
| 		private moderationLogService: ModerationLogService, | ||||
| 	) { | ||||
| 		super(meta, paramDef, async (ps, me) => { | ||||
| 			const user = await this.usersRepository.findOneBy({ id: ps.userId }); | ||||
| @@ -42,17 +40,7 @@ export default class extends Endpoint<typeof meta, typeof paramDef> { // eslint- | ||||
| 				throw new Error('user not found'); | ||||
| 			} | ||||
|  | ||||
| 			await this.usersRepository.update(user.id, { | ||||
| 				isSuspended: false, | ||||
| 			}); | ||||
|  | ||||
| 			this.moderationLogService.log(me, 'unsuspend', { | ||||
| 				userId: user.id, | ||||
| 				userUsername: user.username, | ||||
| 				userHost: user.host, | ||||
| 			}); | ||||
|  | ||||
| 			this.userSuspendService.doPostUnsuspend(user); | ||||
| 			await this.userSuspendService.unsuspend(user, me); | ||||
| 		}); | ||||
| 	} | ||||
| } | ||||
|   | ||||
| @@ -150,6 +150,13 @@ export const paramDef = { | ||||
| 				type: 'string', | ||||
| 			}, | ||||
| 		}, | ||||
| 		mediaSilencedHosts: { | ||||
| 			type: 'array', | ||||
| 			nullable: true, | ||||
| 			items: { | ||||
| 				type: 'string', | ||||
| 			}, | ||||
| 		}, | ||||
| 		summalyProxy: { | ||||
| 			type: 'string', nullable: true, | ||||
| 			description: '[Deprecated] Use "urlPreviewSummaryProxyUrl" instead.', | ||||
| @@ -203,6 +210,14 @@ export default class extends Endpoint<typeof meta, typeof paramDef> { // eslint- | ||||
| 					return h !== '' && h !== lv && !set.blockedHosts?.includes(h); | ||||
| 				}); | ||||
| 			} | ||||
| 			if (Array.isArray(ps.mediaSilencedHosts)) { | ||||
| 				let lastValue = ''; | ||||
| 				set.mediaSilencedHosts = ps.mediaSilencedHosts.sort().filter((h) => { | ||||
| 					const lv = lastValue; | ||||
| 					lastValue = h; | ||||
| 					return h !== '' && h !== lv && !set.blockedHosts?.includes(h); | ||||
| 				}); | ||||
| 			} | ||||
| 			if (ps.themeColor !== undefined) { | ||||
| 				set.themeColor = ps.themeColor; | ||||
| 			} | ||||
|   | ||||
| @@ -170,7 +170,7 @@ export default class extends Endpoint<typeof meta, typeof paramDef> { // eslint- | ||||
|  | ||||
| 			const instances = await query.limit(ps.limit).offset(ps.offset).getMany(); | ||||
|  | ||||
| 			return await this.instanceEntityService.packMany(instances); | ||||
| 			return await this.instanceEntityService.packMany(instances, me); | ||||
| 		}); | ||||
| 	} | ||||
| } | ||||
|   | ||||
| @@ -107,9 +107,9 @@ export default class extends Endpoint<typeof meta, typeof paramDef> { // eslint- | ||||
| 			const gotPubCount = topPubInstances.map(x => x.followingCount).reduce((a, b) => a + b, 0); | ||||
|  | ||||
| 			return await awaitAll({ | ||||
| 				topSubInstances: this.instanceEntityService.packMany(topSubInstances), | ||||
| 				topSubInstances: this.instanceEntityService.packMany(topSubInstances, me), | ||||
| 				otherFollowersCount: Math.max(0, allSubCount - gotSubCount), | ||||
| 				topPubInstances: this.instanceEntityService.packMany(topPubInstances), | ||||
| 				topPubInstances: this.instanceEntityService.packMany(topPubInstances, me), | ||||
| 				otherFollowingCount: Math.max(0, allPubCount - gotPubCount), | ||||
| 			}); | ||||
| 		}); | ||||
|   | ||||
| @@ -143,6 +143,12 @@ export default class extends Endpoint<typeof meta, typeof paramDef> { // eslint- | ||||
| 				]; | ||||
| 			} | ||||
|  | ||||
| 			const [ | ||||
| 				followings, | ||||
| 			] = await Promise.all([ | ||||
| 				this.cacheService.userFollowingsCache.fetch(me.id), | ||||
| 			]); | ||||
|  | ||||
| 			const redisTimeline = await this.fanoutTimelineEndpointService.timeline({ | ||||
| 				untilId, | ||||
| 				sinceId, | ||||
| @@ -153,6 +159,13 @@ export default class extends Endpoint<typeof meta, typeof paramDef> { // eslint- | ||||
| 				useDbFallback: serverSettings.enableFanoutTimelineDbFallback, | ||||
| 				alwaysIncludeMyNotes: true, | ||||
| 				excludePureRenotes: !ps.withRenotes, | ||||
| 				noteFilter: note => { | ||||
| 					if (note.reply && note.reply.visibility === 'followers') { | ||||
| 						if (!Object.hasOwn(followings, note.reply.userId) && note.reply.userId !== me.id) return false; | ||||
| 					} | ||||
|  | ||||
| 					return true; | ||||
| 				}, | ||||
| 				dbFallback: async (untilId, sinceId, limit) => await this.getFromDb({ | ||||
| 					untilId, | ||||
| 					sinceId, | ||||
|   | ||||
| @@ -114,7 +114,7 @@ export default class extends Endpoint<typeof meta, typeof paramDef> { // eslint- | ||||
| 				excludePureRenotes: !ps.withRenotes, | ||||
| 				noteFilter: note => { | ||||
| 					if (note.reply && note.reply.visibility === 'followers') { | ||||
| 						if (!Object.hasOwn(followings, note.reply.userId)) return false; | ||||
| 						if (!Object.hasOwn(followings, note.reply.userId) && note.reply.userId !== me.id) return false; | ||||
| 					} | ||||
|  | ||||
| 					return true; | ||||
|   | ||||
| @@ -11,6 +11,7 @@ import { QueryService } from '@/core/QueryService.js'; | ||||
| import { FollowingEntityService } from '@/core/entities/FollowingEntityService.js'; | ||||
| import { UtilityService } from '@/core/UtilityService.js'; | ||||
| import { DI } from '@/di-symbols.js'; | ||||
| import { RoleService } from '@/core/RoleService.js'; | ||||
| import { ApiError } from '../../error.js'; | ||||
|  | ||||
| export const meta = { | ||||
| @@ -81,6 +82,7 @@ export default class extends Endpoint<typeof meta, typeof paramDef> { // eslint- | ||||
| 		private utilityService: UtilityService, | ||||
| 		private followingEntityService: FollowingEntityService, | ||||
| 		private queryService: QueryService, | ||||
| 		private roleService: RoleService, | ||||
| 	) { | ||||
| 		super(meta, paramDef, async (ps, me) => { | ||||
| 			const user = await this.usersRepository.findOneBy(ps.userId != null | ||||
| @@ -93,6 +95,7 @@ export default class extends Endpoint<typeof meta, typeof paramDef> { // eslint- | ||||
|  | ||||
| 			const profile = await this.userProfilesRepository.findOneByOrFail({ userId: user.id }); | ||||
|  | ||||
| 			if (profile.followersVisibility !== 'public' && !await this.roleService.isModerator(me)) { | ||||
| 				if (profile.followersVisibility === 'private') { | ||||
| 					if (me == null || (me.id !== user.id)) { | ||||
| 						throw new ApiError(meta.errors.forbidden); | ||||
| @@ -112,6 +115,7 @@ export default class extends Endpoint<typeof meta, typeof paramDef> { // eslint- | ||||
| 						} | ||||
| 					} | ||||
| 				} | ||||
| 			} | ||||
|  | ||||
| 			const query = this.queryService.makePaginationQuery(this.followingsRepository.createQueryBuilder('following'), ps.sinceId, ps.untilId) | ||||
| 				.andWhere('following.followeeId = :userId', { userId: user.id }) | ||||
|   | ||||
| @@ -12,6 +12,7 @@ import { QueryService } from '@/core/QueryService.js'; | ||||
| import { FollowingEntityService } from '@/core/entities/FollowingEntityService.js'; | ||||
| import { UtilityService } from '@/core/UtilityService.js'; | ||||
| import { DI } from '@/di-symbols.js'; | ||||
| import { RoleService } from '@/core/RoleService.js'; | ||||
| import { ApiError } from '../../error.js'; | ||||
|  | ||||
| export const meta = { | ||||
| @@ -90,6 +91,7 @@ export default class extends Endpoint<typeof meta, typeof paramDef> { // eslint- | ||||
| 		private utilityService: UtilityService, | ||||
| 		private followingEntityService: FollowingEntityService, | ||||
| 		private queryService: QueryService, | ||||
| 		private roleService: RoleService, | ||||
| 	) { | ||||
| 		super(meta, paramDef, async (ps, me) => { | ||||
| 			const user = await this.usersRepository.findOneBy(ps.userId != null | ||||
| @@ -102,6 +104,7 @@ export default class extends Endpoint<typeof meta, typeof paramDef> { // eslint- | ||||
|  | ||||
| 			const profile = await this.userProfilesRepository.findOneByOrFail({ userId: user.id }); | ||||
|  | ||||
| 			if (profile.followingVisibility !== 'public' && !await this.roleService.isModerator(me)) { | ||||
| 				if (profile.followingVisibility === 'private') { | ||||
| 					if (me == null || (me.id !== user.id)) { | ||||
| 						throw new ApiError(meta.errors.forbidden); | ||||
| @@ -121,6 +124,7 @@ export default class extends Endpoint<typeof meta, typeof paramDef> { // eslint- | ||||
| 						} | ||||
| 					} | ||||
| 				} | ||||
| 			} | ||||
|  | ||||
| 			const query = this.queryService.makePaginationQuery(this.followingsRepository.createQueryBuilder('following'), ps.sinceId, ps.untilId) | ||||
| 				.andWhere('following.followerId = :userId', { userId: user.id }) | ||||
|   | ||||
| @@ -14,11 +14,14 @@ import { CacheService } from '@/core/CacheService.js'; | ||||
| import { MiFollowing, MiUserProfile } from '@/models/_.js'; | ||||
| import type { StreamEventEmitter, GlobalEvents } from '@/core/GlobalEventService.js'; | ||||
| import { ChannelFollowingService } from '@/core/ChannelFollowingService.js'; | ||||
| import type { JsonObject } from '@/misc/json-value.js'; | ||||
| import { isJsonObject } from '@/misc/json-value.js'; | ||||
| import type { JsonObject, JsonValue } from '@/misc/json-value.js'; | ||||
| import type { ChannelsService } from './ChannelsService.js'; | ||||
| import type { EventEmitter } from 'events'; | ||||
| import type Channel from './channel.js'; | ||||
|  | ||||
| const MAX_CHANNELS_PER_CONNECTION = 32; | ||||
|  | ||||
| /** | ||||
|  * Main stream connection | ||||
|  */ | ||||
| @@ -112,8 +115,6 @@ export default class Connection { | ||||
|  | ||||
| 		const { type, body } = obj; | ||||
|  | ||||
| 		if (typeof body !== 'object' || body === null || Array.isArray(body)) return; | ||||
|  | ||||
| 		switch (type) { | ||||
| 			case 'readNotification': this.onReadNotification(body); break; | ||||
| 			case 'subNote': this.onSubscribeNote(body); break; | ||||
| @@ -154,7 +155,8 @@ export default class Connection { | ||||
| 	} | ||||
|  | ||||
| 	@bindThis | ||||
| 	private readNote(body: JsonObject) { | ||||
| 	private readNote(body: JsonValue | undefined) { | ||||
| 		if (!isJsonObject(body)) return; | ||||
| 		const id = body.id; | ||||
|  | ||||
| 		const note = this.cachedNotes.find(n => n.id === id); | ||||
| @@ -166,7 +168,7 @@ export default class Connection { | ||||
| 	} | ||||
|  | ||||
| 	@bindThis | ||||
| 	private onReadNotification(payload: JsonObject) { | ||||
| 	private onReadNotification(payload: JsonValue | undefined) { | ||||
| 		this.notificationService.readAllNotification(this.user!.id); | ||||
| 	} | ||||
|  | ||||
| @@ -174,7 +176,8 @@ export default class Connection { | ||||
| 	 * 投稿購読要求時 | ||||
| 	 */ | ||||
| 	@bindThis | ||||
| 	private onSubscribeNote(payload: JsonObject) { | ||||
| 	private onSubscribeNote(payload: JsonValue | undefined) { | ||||
| 		if (!isJsonObject(payload)) return; | ||||
| 		if (!payload.id || typeof payload.id !== 'string') return; | ||||
|  | ||||
| 		const current = this.subscribingNotes[payload.id] ?? 0; | ||||
| @@ -190,7 +193,8 @@ export default class Connection { | ||||
| 	 * 投稿購読解除要求時 | ||||
| 	 */ | ||||
| 	@bindThis | ||||
| 	private onUnsubscribeNote(payload: JsonObject) { | ||||
| 	private onUnsubscribeNote(payload: JsonValue | undefined) { | ||||
| 		if (!isJsonObject(payload)) return; | ||||
| 		if (!payload.id || typeof payload.id !== 'string') return; | ||||
|  | ||||
| 		const current = this.subscribingNotes[payload.id]; | ||||
| @@ -216,12 +220,13 @@ export default class Connection { | ||||
| 	 * チャンネル接続要求時 | ||||
| 	 */ | ||||
| 	@bindThis | ||||
| 	private onChannelConnectRequested(payload: JsonObject) { | ||||
| 	private onChannelConnectRequested(payload: JsonValue | undefined) { | ||||
| 		if (!isJsonObject(payload)) return; | ||||
| 		const { channel, id, params, pong } = payload; | ||||
| 		if (typeof id !== 'string') return; | ||||
| 		if (typeof channel !== 'string') return; | ||||
| 		if (typeof pong !== 'boolean' && typeof pong !== 'undefined' && pong !== null) return; | ||||
| 		if (typeof params !== 'undefined' && (typeof params !== 'object' || params === null || Array.isArray(params))) return; | ||||
| 		if (typeof params !== 'undefined' && !isJsonObject(params)) return; | ||||
| 		this.connectChannel(id, params, channel, pong ?? undefined); | ||||
| 	} | ||||
|  | ||||
| @@ -229,7 +234,8 @@ export default class Connection { | ||||
| 	 * チャンネル切断要求時 | ||||
| 	 */ | ||||
| 	@bindThis | ||||
| 	private onChannelDisconnectRequested(payload: JsonObject) { | ||||
| 	private onChannelDisconnectRequested(payload: JsonValue | undefined) { | ||||
| 		if (!isJsonObject(payload)) return; | ||||
| 		const { id } = payload; | ||||
| 		if (typeof id !== 'string') return; | ||||
| 		this.disconnectChannel(id); | ||||
| @@ -251,6 +257,10 @@ export default class Connection { | ||||
| 	 */ | ||||
| 	@bindThis | ||||
| 	public connectChannel(id: string, params: JsonObject | undefined, channel: string, pong = false) { | ||||
| 		if (this.channels.length >= MAX_CHANNELS_PER_CONNECTION) { | ||||
| 			return; | ||||
| 		} | ||||
|  | ||||
| 		const channelService = this.channelsService.getChannelService(channel); | ||||
|  | ||||
| 		if (channelService.requireCredential && this.user == null) { | ||||
| @@ -297,7 +307,8 @@ export default class Connection { | ||||
| 	 * @param data メッセージ | ||||
| 	 */ | ||||
| 	@bindThis | ||||
| 	private onChannelMessageRequested(data: JsonObject) { | ||||
| 	private onChannelMessageRequested(data: JsonValue | undefined) { | ||||
| 		if (!isJsonObject(data)) return; | ||||
| 		if (typeof data.id !== 'string') return; | ||||
| 		if (typeof data.type !== 'string') return; | ||||
| 		if (typeof data.body === 'undefined') return; | ||||
|   | ||||
| @@ -60,7 +60,7 @@ class HomeTimelineChannel extends Channel { | ||||
| 			const reply = note.reply; | ||||
| 			if (this.following[note.userId]?.withReplies) { | ||||
| 				// 自分のフォローしていないユーザーの visibility: followers な投稿への返信は弾く | ||||
| 				if (reply.visibility === 'followers' && !Object.hasOwn(this.following, reply.userId)) return; | ||||
| 				if (reply.visibility === 'followers' && !Object.hasOwn(this.following, reply.userId) && reply.userId !== this.user!.id) return; | ||||
| 			} else { | ||||
| 				// 「チャンネル接続主への返信」でもなければ、「チャンネル接続主が行った返信」でもなければ、「投稿者の投稿者自身への返信」でもない場合 | ||||
| 				if (reply.userId !== this.user!.id && !isMe && reply.userId !== note.userId) return; | ||||
| @@ -73,7 +73,7 @@ class HomeTimelineChannel extends Channel { | ||||
| 			if (note.renote.reply) { | ||||
| 				const reply = note.renote.reply; | ||||
| 				// 自分のフォローしていないユーザーの visibility: followers な投稿への返信のリノートは弾く | ||||
| 				if (reply.visibility === 'followers' && !Object.hasOwn(this.following, reply.userId)) return; | ||||
| 				if (reply.visibility === 'followers' && !Object.hasOwn(this.following, reply.userId) && reply.userId !== this.user!.id) return; | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
|   | ||||
| @@ -76,14 +76,22 @@ class HybridTimelineChannel extends Channel { | ||||
| 			const reply = note.reply; | ||||
| 			if ((this.following[note.userId]?.withReplies ?? false) || this.withReplies) { | ||||
| 				// 自分のフォローしていないユーザーの visibility: followers な投稿への返信は弾く | ||||
| 				if (reply.visibility === 'followers' && !Object.hasOwn(this.following, reply.userId)) return; | ||||
| 				if (reply.visibility === 'followers' && !Object.hasOwn(this.following, reply.userId) && reply.userId !== this.user!.id) return; | ||||
| 			} else { | ||||
| 				// 「チャンネル接続主への返信」でもなければ、「チャンネル接続主が行った返信」でもなければ、「投稿者の投稿者自身への返信」でもない場合 | ||||
| 				if (reply.userId !== this.user!.id && !isMe && reply.userId !== note.userId) return; | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		if (isRenotePacked(note) && !isQuotePacked(note) && !this.withRenotes) return; | ||||
| 		// 純粋なリノート(引用リノートでないリノート)の場合 | ||||
| 		if (isRenotePacked(note) && !isQuotePacked(note) && note.renote) { | ||||
| 			if (!this.withRenotes) return; | ||||
| 			if (note.renote.reply) { | ||||
| 				const reply = note.renote.reply; | ||||
| 				// 自分のフォローしていないユーザーの visibility: followers な投稿への返信のリノートは弾く | ||||
| 				if (reply.visibility === 'followers' && !Object.hasOwn(this.following, reply.userId) && reply.userId !== this.user!.id) return; | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		if (this.user && note.renoteId && !note.text) { | ||||
| 			if (note.renote && Object.keys(note.renote.reactions).length > 0) { | ||||
|   | ||||
| @@ -6,6 +6,7 @@ | ||||
| import Xev from 'xev'; | ||||
| import { Injectable } from '@nestjs/common'; | ||||
| import { bindThis } from '@/decorators.js'; | ||||
| import { isJsonObject } from '@/misc/json-value.js'; | ||||
| import type { JsonObject, JsonValue } from '@/misc/json-value.js'; | ||||
| import Channel, { type MiChannelService } from '../channel.js'; | ||||
|  | ||||
| @@ -36,7 +37,7 @@ class QueueStatsChannel extends Channel { | ||||
| 	public onMessage(type: string, body: JsonValue) { | ||||
| 		switch (type) { | ||||
| 			case 'requestLog': | ||||
| 				if (typeof body !== 'object' || body === null || Array.isArray(body)) return; | ||||
| 				if (!isJsonObject(body)) return; | ||||
| 				if (typeof body.id !== 'string') return; | ||||
| 				if (typeof body.length !== 'number') return; | ||||
| 				ev.once(`queueStatsLog:${body.id}`, statsLog => { | ||||
|   | ||||
| @@ -9,8 +9,10 @@ import { DI } from '@/di-symbols.js'; | ||||
| import { bindThis } from '@/decorators.js'; | ||||
| import { ReversiService } from '@/core/ReversiService.js'; | ||||
| import { ReversiGameEntityService } from '@/core/entities/ReversiGameEntityService.js'; | ||||
| import { isJsonObject } from '@/misc/json-value.js'; | ||||
| import type { JsonObject, JsonValue } from '@/misc/json-value.js'; | ||||
| import Channel, { type MiChannelService } from '../channel.js'; | ||||
| import { reversiUpdateKeys } from 'misskey-js'; | ||||
|  | ||||
| class ReversiGameChannel extends Channel { | ||||
| 	public readonly chName = 'reversiGame'; | ||||
| @@ -44,16 +46,17 @@ class ReversiGameChannel extends Channel { | ||||
| 				this.ready(body); | ||||
| 				break; | ||||
| 			case 'updateSettings': | ||||
| 				if (typeof body !== 'object' || body === null || Array.isArray(body)) return; | ||||
| 				if (typeof body.key !== 'string') return; | ||||
| 				if (typeof body.value !== 'object' || body.value === null || Array.isArray(body.value)) return; | ||||
| 				if (!isJsonObject(body)) return; | ||||
| 				if (!this.reversiService.isValidReversiUpdateKey(body.key)) return; | ||||
| 				if (!this.reversiService.isValidReversiUpdateValue(body.key, body.value)) return; | ||||
|  | ||||
| 				this.updateSettings(body.key, body.value); | ||||
| 				break; | ||||
| 			case 'cancel': | ||||
| 				this.cancelGame(); | ||||
| 				break; | ||||
| 			case 'putStone': | ||||
| 				if (typeof body !== 'object' || body === null || Array.isArray(body)) return; | ||||
| 				if (!isJsonObject(body)) return; | ||||
| 				if (typeof body.pos !== 'number') return; | ||||
| 				if (typeof body.id !== 'string') return; | ||||
| 				this.putStone(body.pos, body.id); | ||||
| @@ -63,7 +66,7 @@ class ReversiGameChannel extends Channel { | ||||
| 	} | ||||
|  | ||||
| 	@bindThis | ||||
| 	private async updateSettings(key: string, value: JsonObject) { | ||||
| 	private async updateSettings<K extends typeof reversiUpdateKeys[number]>(key: K, value: MiReversiGame[K]) { | ||||
| 		if (this.user == null) return; | ||||
|  | ||||
| 		this.reversiService.updateSettings(this.gameId!, this.user, key, value); | ||||
|   | ||||
| @@ -6,6 +6,7 @@ | ||||
| import Xev from 'xev'; | ||||
| import { Injectable } from '@nestjs/common'; | ||||
| import { bindThis } from '@/decorators.js'; | ||||
| import { isJsonObject } from '@/misc/json-value.js'; | ||||
| import type { JsonObject, JsonValue } from '@/misc/json-value.js'; | ||||
| import Channel, { type MiChannelService } from '../channel.js'; | ||||
|  | ||||
| @@ -36,7 +37,7 @@ class ServerStatsChannel extends Channel { | ||||
| 	public onMessage(type: string, body: JsonValue) { | ||||
| 		switch (type) { | ||||
| 			case 'requestLog': | ||||
| 				if (typeof body !== 'object' || body === null || Array.isArray(body)) return; | ||||
| 				if (!isJsonObject(body)) return; | ||||
| 				ev.once(`serverStatsLog:${body.id}`, statsLog => { | ||||
| 					this.send('statsLog', statsLog); | ||||
| 				}); | ||||
|   | ||||
| @@ -28,6 +28,7 @@ html | ||||
| 		meta(property='og:site_name' content= instanceName || 'Misskey') | ||||
| 		meta(property='instance_url' content= instanceUrl) | ||||
| 		meta(name='viewport' content='width=device-width, initial-scale=1') | ||||
| 		meta(name='format-detection' content='telephone=no,date=no,address=no,email=no,url=no') | ||||
| 		link(rel='icon' href= icon || '/favicon.ico') | ||||
| 		link(rel='apple-touch-icon' href= appleTouchIcon || '/apple-touch-icon.png') | ||||
| 		link(rel='manifest' href='/manifest.json') | ||||
|   | ||||
| @@ -96,6 +96,7 @@ export const moderationLogTypes = [ | ||||
| 	'createAbuseReportNotificationRecipient', | ||||
| 	'updateAbuseReportNotificationRecipient', | ||||
| 	'deleteAbuseReportNotificationRecipient', | ||||
| 	'deleteAccount', | ||||
| ] as const; | ||||
|  | ||||
| export type ModerationLogPayloads = { | ||||
| @@ -314,6 +315,11 @@ export type ModerationLogPayloads = { | ||||
| 		recipientId: string; | ||||
| 		recipient: any; | ||||
| 	}; | ||||
| 	deleteAccount: { | ||||
| 		userId: string; | ||||
| 		userUsername: string; | ||||
| 		userHost: string | null; | ||||
| 	}; | ||||
| }; | ||||
|  | ||||
| export type Serialized<T> = { | ||||
|   | ||||
| @@ -34,6 +34,7 @@ describe('Streaming', () => { | ||||
| 		let kyoko: misskey.entities.SignupResponse; | ||||
| 		let chitose: misskey.entities.SignupResponse; | ||||
| 		let kanako: misskey.entities.SignupResponse; | ||||
| 		let erin: misskey.entities.SignupResponse; | ||||
|  | ||||
| 		// Remote users | ||||
| 		let akari: misskey.entities.SignupResponse; | ||||
| @@ -53,6 +54,7 @@ describe('Streaming', () => { | ||||
| 			kyoko = await signup({ username: 'kyoko' }); | ||||
| 			chitose = await signup({ username: 'chitose' }); | ||||
| 			kanako = await signup({ username: 'kanako' }); | ||||
| 			erin = await signup({ username: 'erin' }); // erin:  A generic fifth participant | ||||
|  | ||||
| 			akari = await signup({ username: 'akari', host: 'example.com' }); | ||||
| 			chinatsu = await signup({ username: 'chinatsu', host: 'example.com' }); | ||||
| @@ -71,6 +73,12 @@ describe('Streaming', () => { | ||||
| 			// Follow: kyoko => chitose | ||||
| 			await api('following/create', { userId: chitose.id }, kyoko); | ||||
|  | ||||
| 			// Follow: erin <=> ayano each other. | ||||
| 			// erin => ayano: withReplies: true | ||||
| 			await api('following/create', { userId: ayano.id, withReplies: true }, erin); | ||||
| 			// ayano => erin: withReplies: false | ||||
| 			await api('following/create', { userId: erin.id, withReplies: false }, ayano); | ||||
|  | ||||
| 			// Mute: chitose => kanako | ||||
| 			await api('mute/create', { userId: kanako.id }, chitose); | ||||
|  | ||||
| @@ -297,6 +305,28 @@ describe('Streaming', () => { | ||||
|  | ||||
| 				assert.strictEqual(fired, true); | ||||
| 			}); | ||||
|  | ||||
| 			test('withReplies: true のとき自分のfollowers投稿に対するリプライが流れる', async () => { | ||||
| 				const erinNote = await post(erin, { text: 'hi', visibility: 'followers' }); | ||||
| 				const fired = await waitFire( | ||||
| 					erin, 'homeTimeline',	// erin:home | ||||
| 					() => api('notes/create', { text: 'hello', replyId: erinNote.id }, ayano),	// ayano reply to erin's followers post | ||||
| 					msg => msg.type === 'note' && msg.body.userId === ayano.id,	// wait ayano | ||||
| 				); | ||||
|  | ||||
| 				assert.strictEqual(fired, true); | ||||
| 			}); | ||||
|  | ||||
| 			test('withReplies: false でも自分の投稿に対するリプライが流れる', async () => { | ||||
| 				const ayanoNote = await post(ayano, { text: 'hi', visibility: 'followers' }); | ||||
| 				const fired = await waitFire( | ||||
| 					ayano, 'homeTimeline',	// ayano:home | ||||
| 					() => api('notes/create', { text: 'hello', replyId: ayanoNote.id }, erin),	// erin reply to ayano's followers post | ||||
| 					msg => msg.type === 'note' && msg.body.userId === erin.id,	// wait erin | ||||
| 				); | ||||
|  | ||||
| 				assert.strictEqual(fired, true); | ||||
| 			}); | ||||
| 		});	// Home | ||||
|  | ||||
| 		describe('Local Timeline', () => { | ||||
| @@ -475,6 +505,38 @@ describe('Streaming', () => { | ||||
|  | ||||
| 				assert.strictEqual(fired, false); | ||||
| 			}); | ||||
|  | ||||
| 			test('withReplies: true のとき自分のfollowers投稿に対するリプライが流れる', async () => { | ||||
| 				const erinNote = await post(erin, { text: 'hi', visibility: 'followers' }); | ||||
| 				const fired = await waitFire( | ||||
| 					erin, 'homeTimeline',	// erin:home | ||||
| 					() => api('notes/create', { text: 'hello', replyId: erinNote.id }, ayano),	// ayano reply to erin's followers post | ||||
| 					msg => msg.type === 'note' && msg.body.userId === ayano.id,	// wait ayano | ||||
| 				); | ||||
|  | ||||
| 				assert.strictEqual(fired, true); | ||||
| 			}); | ||||
|  | ||||
| 			test('withReplies: false でも自分の投稿に対するリプライが流れる', async () => { | ||||
| 				const ayanoNote = await post(ayano, { text: 'hi', visibility: 'followers' }); | ||||
| 				const fired = await waitFire( | ||||
| 					ayano, 'homeTimeline',	// ayano:home | ||||
| 					() => api('notes/create', { text: 'hello', replyId: ayanoNote.id }, erin),	// erin reply to ayano's followers post | ||||
| 					msg => msg.type === 'note' && msg.body.userId === erin.id,	// wait erin | ||||
| 				); | ||||
|  | ||||
| 				assert.strictEqual(fired, true); | ||||
| 			}); | ||||
|  | ||||
| 			test('withReplies: true のフォローしていない人のfollowersノートに対するリプライが流れない', async () => { | ||||
| 				const fired = await waitFire( | ||||
| 					erin, 'homeTimeline',	// erin:home | ||||
| 					() => api('notes/create', { text: 'hello', replyId: chitose.id }, ayano),	// ayano reply to chitose's post | ||||
| 					msg => msg.type === 'note' && msg.body.userId === ayano.id,	// wait ayano | ||||
| 				); | ||||
|  | ||||
| 				assert.strictEqual(fired, false); | ||||
| 			}); | ||||
| 		}); | ||||
|  | ||||
| 		describe('Global Timeline', () => { | ||||
|   | ||||
| @@ -127,6 +127,7 @@ describe('Timelines', () => { | ||||
| 		test.concurrent('withReplies: true でフォローしているユーザーの他人の visibility: followers な投稿への返信が含まれない', async () => { | ||||
| 			const [alice, bob, carol] = await Promise.all([signup(), signup(), signup()]); | ||||
|  | ||||
| 			await api('following/create', { userId: carol.id }, bob); | ||||
| 			await api('following/create', { userId: bob.id }, alice); | ||||
| 			await api('following/update', { userId: bob.id, withReplies: true }, alice); | ||||
| 			await setTimeout(1000); | ||||
| @@ -161,6 +162,24 @@ describe('Timelines', () => { | ||||
| 			assert.strictEqual(res.body.find(note => note.id === carolNote.id)?.text, 'hi'); | ||||
| 		}); | ||||
|  | ||||
| 		test.concurrent('withReplies: true でフォローしているユーザーの自分の visibility: followers な投稿への返信が含まれる', async () => { | ||||
| 			const [alice, bob] = await Promise.all([signup(), signup()]); | ||||
|  | ||||
| 			await api('following/create', { userId: bob.id }, alice); | ||||
| 			await api('following/create', { userId: alice.id }, bob); | ||||
| 			await api('following/update', { userId: bob.id, withReplies: true }, alice); | ||||
| 			await setTimeout(1000); | ||||
| 			const aliceNote = await post(alice, { text: 'hi', visibility: 'followers' }); | ||||
| 			const bobNote = await post(bob, { text: 'hi', replyId: aliceNote.id }); | ||||
|  | ||||
| 			await waitForPushToTl(); | ||||
|  | ||||
| 			const res = await api('notes/timeline', { limit: 100 }, alice); | ||||
|  | ||||
| 			assert.strictEqual(res.body.some((note: any) => note.id === bobNote.id), true); | ||||
| 			assert.strictEqual(res.body.some((note: any) => note.id === aliceNote.id), true); | ||||
| 		}); | ||||
|  | ||||
| 		test.concurrent('withReplies: true でフォローしているユーザーの行った別のフォローしているユーザーの投稿への visibility: specified な返信が含まれない', async () => { | ||||
| 			const [alice, bob, carol] = await Promise.all([signup(), signup(), signup()]); | ||||
|  | ||||
| @@ -684,6 +703,21 @@ describe('Timelines', () => { | ||||
| 			assert.strictEqual(res.body.some(note => note.id === bobNote.id), true); | ||||
| 		}); | ||||
|  | ||||
| 		test.concurrent('withReplies: false でフォローしていないユーザーからの自分への返信が含まれる', async () => { | ||||
| 			const [alice, bob] = await Promise.all([signup(), signup()]); | ||||
|  | ||||
| 			await setTimeout(1000); | ||||
| 			const aliceNote = await post(alice, { text: 'hi' }); | ||||
| 			const bobNote = await post(bob, { text: 'hi', replyId: aliceNote.id }); | ||||
|  | ||||
| 			await waitForPushToTl(); | ||||
|  | ||||
| 			const res = await api('notes/local-timeline', { limit: 100 }, alice); | ||||
|  | ||||
| 			assert.strictEqual(res.body.some(note => note.id === aliceNote.id), true); | ||||
| 			assert.strictEqual(res.body.some(note => note.id === bobNote.id), true); | ||||
| 		}); | ||||
|  | ||||
| 		test.concurrent('[withReplies: true] 他人の他人への返信が含まれる', async () => { | ||||
| 			const [alice, bob, carol] = await Promise.all([signup(), signup(), signup()]); | ||||
|  | ||||
| @@ -768,6 +802,62 @@ describe('Timelines', () => { | ||||
| 			assert.strictEqual(res.body.some(note => note.id === bobNote.id), true); | ||||
| 		}); | ||||
|  | ||||
| 		test.concurrent('withReplies: true でフォローしているユーザーの他人の visibility: followers な投稿への返信が含まれない', async () => { | ||||
| 			const [alice, bob, carol] = await Promise.all([signup(), signup(), signup()]); | ||||
|  | ||||
| 			await api('following/create', { userId: carol.id }, bob); | ||||
| 			await api('following/create', { userId: bob.id }, alice); | ||||
| 			await api('following/update', { userId: bob.id, withReplies: true }, alice); | ||||
| 			await setTimeout(1000); | ||||
| 			const carolNote = await post(carol, { text: 'hi', visibility: 'followers' }); | ||||
| 			const bobNote = await post(bob, { text: 'hi', replyId: carolNote.id }); | ||||
|  | ||||
| 			await waitForPushToTl(); | ||||
|  | ||||
| 			const res = await api('notes/hybrid-timeline', { limit: 100 }, alice); | ||||
|  | ||||
| 			assert.strictEqual(res.body.some((note: any) => note.id === bobNote.id), false); | ||||
| 			assert.strictEqual(res.body.some((note: any) => note.id === carolNote.id), false); | ||||
| 		}); | ||||
|  | ||||
| 		test.concurrent('withReplies: true でフォローしているユーザーの行った別のフォローしているユーザーの visibility: followers な投稿への返信が含まれる', async () => { | ||||
| 			const [alice, bob, carol] = await Promise.all([signup(), signup(), signup()]); | ||||
|  | ||||
| 			await api('following/create', { userId: bob.id }, alice); | ||||
| 			await api('following/create', { userId: carol.id }, alice); | ||||
| 			await api('following/create', { userId: carol.id }, bob); | ||||
| 			await api('following/update', { userId: bob.id, withReplies: true }, alice); | ||||
| 			await setTimeout(1000); | ||||
| 			const carolNote = await post(carol, { text: 'hi', visibility: 'followers' }); | ||||
| 			const bobNote = await post(bob, { text: 'hi', replyId: carolNote.id }); | ||||
|  | ||||
| 			await waitForPushToTl(); | ||||
|  | ||||
| 			const res = await api('notes/hybrid-timeline', { limit: 100 }, alice); | ||||
|  | ||||
| 			assert.strictEqual(res.body.some((note: any) => note.id === bobNote.id), true); | ||||
| 			assert.strictEqual(res.body.some((note: any) => note.id === carolNote.id), true); | ||||
| 			assert.strictEqual(res.body.find((note: any) => note.id === carolNote.id)?.text, 'hi'); | ||||
| 		}); | ||||
|  | ||||
| 		test.concurrent('withReplies: true でフォローしているユーザーの自分の visibility: followers な投稿への返信が含まれる', async () => { | ||||
| 			const [alice, bob] = await Promise.all([signup(), signup()]); | ||||
|  | ||||
| 			await api('following/create', { userId: bob.id }, alice); | ||||
| 			await api('following/create', { userId: alice.id }, bob); | ||||
| 			await api('following/update', { userId: bob.id, withReplies: true }, alice); | ||||
| 			await setTimeout(1000); | ||||
| 			const aliceNote = await post(alice, { text: 'hi', visibility: 'followers' }); | ||||
| 			const bobNote = await post(bob, { text: 'hi', replyId: aliceNote.id }); | ||||
|  | ||||
| 			await waitForPushToTl(); | ||||
|  | ||||
| 			const res = await api('notes/hybrid-timeline', { limit: 100 }, alice); | ||||
|  | ||||
| 			assert.strictEqual(res.body.some((note: any) => note.id === bobNote.id), true); | ||||
| 			assert.strictEqual(res.body.some((note: any) => note.id === aliceNote.id), true); | ||||
| 		}); | ||||
|  | ||||
| 		test.concurrent('他人の他人への返信が含まれない', async () => { | ||||
| 			const [alice, bob, carol] = await Promise.all([signup(), signup(), signup()]); | ||||
|  | ||||
| @@ -824,6 +914,21 @@ describe('Timelines', () => { | ||||
| 			assert.strictEqual(res.body.some(note => note.id === bobNote.id), true); | ||||
| 		}); | ||||
|  | ||||
| 		test.concurrent('withReplies: false でフォローしていないユーザーからの自分への返信が含まれる', async () => { | ||||
| 			const [alice, bob] = await Promise.all([signup(), signup()]); | ||||
|  | ||||
| 			await setTimeout(1000); | ||||
| 			const aliceNote = await post(alice, { text: 'hi' }); | ||||
| 			const bobNote = await post(bob, { text: 'hi', replyId: aliceNote.id }); | ||||
|  | ||||
| 			await waitForPushToTl(); | ||||
|  | ||||
| 			const res = await api('notes/local-timeline', { limit: 100 }, alice); | ||||
|  | ||||
| 			assert.strictEqual(res.body.some(note => note.id === aliceNote.id), true); | ||||
| 			assert.strictEqual(res.body.some(note => note.id === bobNote.id), true); | ||||
| 		}); | ||||
|  | ||||
| 		test.concurrent('[withReplies: true] 他人の他人への返信が含まれる', async () => { | ||||
| 			const [alice, bob, carol] = await Promise.all([signup(), signup(), signup()]); | ||||
|  | ||||
|   | ||||
| @@ -20,7 +20,8 @@ import { CoreModule } from '@/core/CoreModule.js'; | ||||
| import { FederatedInstanceService } from '@/core/FederatedInstanceService.js'; | ||||
| import { LoggerService } from '@/core/LoggerService.js'; | ||||
| import type { IActor, IApDocument, ICollection, IObject, IPost } from '@/core/activitypub/type.js'; | ||||
| import { MiMeta, MiNote } from '@/models/_.js'; | ||||
| import { MiMeta, MiNote, UserProfilesRepository } from '@/models/_.js'; | ||||
| import { DI } from '@/di-symbols.js'; | ||||
| import { secureRndstr } from '@/misc/secure-rndstr.js'; | ||||
| import { DownloadService } from '@/core/DownloadService.js'; | ||||
| import { MetaService } from '@/core/MetaService.js'; | ||||
| @@ -86,6 +87,7 @@ async function createRandomRemoteUser( | ||||
| } | ||||
|  | ||||
| describe('ActivityPub', () => { | ||||
| 	let userProfilesRepository: UserProfilesRepository; | ||||
| 	let imageService: ApImageService; | ||||
| 	let noteService: ApNoteService; | ||||
| 	let personService: ApPersonService; | ||||
| @@ -127,6 +129,8 @@ describe('ActivityPub', () => { | ||||
| 		await app.init(); | ||||
| 		app.enableShutdownHooks(); | ||||
|  | ||||
| 		userProfilesRepository = app.get(DI.userProfilesRepository); | ||||
|  | ||||
| 		noteService = app.get<ApNoteService>(ApNoteService); | ||||
| 		personService = app.get<ApPersonService>(ApPersonService); | ||||
| 		rendererService = app.get<ApRendererService>(ApRendererService); | ||||
| @@ -205,6 +209,53 @@ describe('ActivityPub', () => { | ||||
| 		}); | ||||
| 	}); | ||||
|  | ||||
| 	describe('Collection visibility', () => { | ||||
| 		test('Public following/followers', async () => { | ||||
| 			const actor = createRandomActor(); | ||||
| 			actor.following = { | ||||
| 				id: `${actor.id}/following`, | ||||
| 				type: 'OrderedCollection', | ||||
| 				totalItems: 0, | ||||
| 				first: `${actor.id}/following?page=1`, | ||||
| 			}; | ||||
| 			actor.followers = `${actor.id}/followers`; | ||||
|  | ||||
| 			resolver.register(actor.id, actor); | ||||
| 			resolver.register(actor.followers, { | ||||
| 				id: actor.followers, | ||||
| 				type: 'OrderedCollection', | ||||
| 				totalItems: 0, | ||||
| 				first: `${actor.followers}?page=1`, | ||||
| 			}); | ||||
|  | ||||
| 			const user = await personService.createPerson(actor.id, resolver); | ||||
| 			const userProfile = await userProfilesRepository.findOneByOrFail({ userId: user.id }); | ||||
|  | ||||
| 			assert.deepStrictEqual(userProfile.followingVisibility, 'public'); | ||||
| 			assert.deepStrictEqual(userProfile.followersVisibility, 'public'); | ||||
| 		}); | ||||
|  | ||||
| 		test('Private following/followers', async () => { | ||||
| 			const actor = createRandomActor(); | ||||
| 			actor.following = { | ||||
| 				id: `${actor.id}/following`, | ||||
| 				type: 'OrderedCollection', | ||||
| 				totalItems: 0, | ||||
| 				// first: … | ||||
| 			}; | ||||
| 			actor.followers = `${actor.id}/followers`; | ||||
|  | ||||
| 			resolver.register(actor.id, actor); | ||||
| 			//resolver.register(actor.followers, { … }); | ||||
|  | ||||
| 			const user = await personService.createPerson(actor.id, resolver); | ||||
| 			const userProfile = await userProfilesRepository.findOneByOrFail({ userId: user.id }); | ||||
|  | ||||
| 			assert.deepStrictEqual(userProfile.followingVisibility, 'private'); | ||||
| 			assert.deepStrictEqual(userProfile.followersVisibility, 'private'); | ||||
| 		}); | ||||
| 	}); | ||||
|  | ||||
| 	describe('Renderer', () => { | ||||
| 		test('Render an announce with visibility: followers', () => { | ||||
| 			rendererService.renderAnnounce('https://example.com/notes/00example', { | ||||
|   | ||||
| @@ -3,6 +3,7 @@ | ||||
|  * SPDX-License-Identifier: AGPL-3.0-only | ||||
|  */ | ||||
|  | ||||
| import { AISCRIPT_VERSION } from '@syuilo/aiscript'; | ||||
| import type { entities } from 'misskey-js' | ||||
|  | ||||
| export function abuseUserReport() { | ||||
| @@ -47,18 +48,7 @@ export function clip(id = 'someclipid', name = 'Some Clip'): entities.Clip { | ||||
| 		createdAt: '2016-12-28T22:49:51.000Z', | ||||
| 		lastClippedAt: null, | ||||
| 		userId: 'someuserid', | ||||
| 		user: { | ||||
| 			id: 'someuserid', | ||||
| 			name: 'Misskey User', | ||||
| 			username: 'miskist', | ||||
| 			host: 'misskey-hub.net', | ||||
| 			avatarUrl: 'https://github.com/misskey-dev/misskey/blob/master/packages/frontend/assets/about-icon.png?raw=true', | ||||
| 			avatarBlurhash: 'eQFRshof5NWBRi},juayfPju53WB?0ofs;s*a{ofjuay^SoMEJR%ay', | ||||
| 			avatarDecorations: [], | ||||
| 			emojis: {}, | ||||
| 			badgeRoles: [], | ||||
| 			onlineStatus: 'unknown', | ||||
| 		}, | ||||
| 		user: userLite(), | ||||
| 		notesCount: undefined, | ||||
| 		name, | ||||
| 		description: 'Some clip description', | ||||
| @@ -125,6 +115,49 @@ export function file(isSensitive = false) { | ||||
| 	}; | ||||
| } | ||||
|  | ||||
| const script = `/// @ ${AISCRIPT_VERSION} | ||||
|  | ||||
| var name = "" | ||||
|  | ||||
| Ui:render([ | ||||
| 	Ui:C:textInput({ | ||||
| 		label: "Your name" | ||||
| 		onInput: @(v) { name = v } | ||||
| 	}) | ||||
| 	Ui:C:button({ | ||||
| 		text: "Hello" | ||||
| 		onClick: @() { | ||||
| 			Mk:dialog(null, \`Hello, {name}!\`) | ||||
| 		} | ||||
| 	}) | ||||
| ]) | ||||
| `; | ||||
|  | ||||
| export function flash(): entities.Flash { | ||||
| 	return { | ||||
| 		id: 'someflashid', | ||||
| 		createdAt: '2016-12-28T22:49:51.000Z', | ||||
| 		updatedAt: '2016-12-28T22:49:51.000Z', | ||||
| 		userId: 'someuserid', | ||||
| 		user: userLite(), | ||||
| 		title: 'Some Play title', | ||||
| 		summary: 'Some Play summary', | ||||
| 		script, | ||||
| 		visibility: 'public', | ||||
| 		likedCount: 0, | ||||
| 		isLiked: false, | ||||
| 	}; | ||||
| } | ||||
|  | ||||
| export function folder(id = 'somefolderid', name = 'Some Folder', parentId: string | null = null): entities.DriveFolder { | ||||
| 	return { | ||||
| 		id, | ||||
| 		createdAt: '2016-12-28T22:49:51.000Z', | ||||
| 		name, | ||||
| 		parentId, | ||||
| 	}; | ||||
| } | ||||
|  | ||||
| export function federationInstance(): entities.FederationInstance { | ||||
| 	return { | ||||
| 		id: 'someinstanceid', | ||||
| @@ -154,7 +187,27 @@ export function federationInstance(): entities.FederationInstance { | ||||
| 	}; | ||||
| } | ||||
|  | ||||
| export function userDetailed(id = 'someuserid', username = 'miskist', host:entities.UserDetailed['host'] = 'misskey-hub.net', name:entities.UserDetailed['name'] = 'Misskey User'): entities.UserDetailed { | ||||
| export function note(id = 'somenoteid'): entities.Note { | ||||
| 	return { | ||||
| 		id, | ||||
| 		createdAt: '2016-12-28T22:49:51.000Z', | ||||
| 		deletedAt: null, | ||||
| 		text: 'some note', | ||||
| 		cw: null, | ||||
| 		userId: 'someuserid', | ||||
| 		user: userLite(), | ||||
| 		visibility: 'public', | ||||
| 		reactionAcceptance: 'nonSensitiveOnly', | ||||
| 		reactionEmojis: {}, | ||||
| 		reactions: {}, | ||||
| 		myReaction: null, | ||||
| 		reactionCount: 0, | ||||
| 		renoteCount: 0, | ||||
| 		repliesCount: 0, | ||||
| 	}; | ||||
| } | ||||
|  | ||||
| export function userLite(id = 'someuserid', username = 'miskist', host: entities.UserDetailed['host'] = 'misskey-hub.net', name: entities.UserDetailed['name'] = 'Misskey User'): entities.UserLite { | ||||
| 	return { | ||||
| 		id, | ||||
| 		username, | ||||
| @@ -165,6 +218,12 @@ export function userDetailed(id = 'someuserid', username = 'miskist', host:entit | ||||
| 		avatarBlurhash: 'eQFRshof5NWBRi},juayfPju53WB?0ofs;s*a{ofjuay^SoMEJR%ay', | ||||
| 		avatarDecorations: [], | ||||
| 		emojis: {}, | ||||
| 	}; | ||||
| } | ||||
|  | ||||
| export function userDetailed(id = 'someuserid', username = 'miskist', host: entities.UserDetailed['host'] = 'misskey-hub.net', name: entities.UserDetailed['name'] = 'Misskey User'): entities.UserDetailed { | ||||
| 	return { | ||||
| 		...userLite(id, username, host, name), | ||||
| 		bannerBlurhash: 'eQA^IW^-MH8w9tE8I=S^o{$*R4RikXtSxutRozjEnNR.RQadoyozog', | ||||
| 		bannerUrl: 'https://github.com/misskey-dev/misskey/blob/master/packages/frontend/assets/fedi.jpg?raw=true', | ||||
| 		birthday: '2014-06-20', | ||||
| @@ -215,7 +274,7 @@ export function userDetailed(id = 'someuserid', username = 'miskist', host:entit | ||||
| 		movedTo: null, | ||||
| 		alsoKnownAs: null, | ||||
| 		notify: 'none', | ||||
| 		memo: null | ||||
| 		memo: null, | ||||
| 	}; | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -397,8 +397,8 @@ function toStories(component: string): Promise<string> { | ||||
| 	const globs = await Promise.all([ | ||||
| 		glob('src/components/global/Mk*.vue'), | ||||
| 		glob('src/components/global/RouterView.vue'), | ||||
| 		glob('src/components/Mk[A-C]*.vue'), | ||||
| 		glob('src/components/MkDigitalClock.vue'), | ||||
| 		glob('src/components/Mk[A-E]*.vue'), | ||||
| 		glob('src/components/MkFlashPreview.vue'), | ||||
| 		glob('src/components/MkGalleryPostPreview.vue'), | ||||
| 		glob('src/components/MkSignupServerRules.vue'), | ||||
| 		glob('src/components/MkUserSetupDialog.vue'), | ||||
|   | ||||
| @@ -28,7 +28,7 @@ | ||||
| 		"@tabler/icons-webfont": "3.3.0", | ||||
| 		"@twemoji/parser": "15.1.1", | ||||
| 		"@vitejs/plugin-vue": "5.1.0", | ||||
| 		"@vue/compiler-sfc": "3.4.34", | ||||
| 		"@vue/compiler-sfc": "3.4.37", | ||||
| 		"aiscript-vscode": "github:aiscript-dev/aiscript-vscode#v0.1.11", | ||||
| 		"astring": "1.8.6", | ||||
| 		"broadcast-channel": "7.0.0", | ||||
| @@ -72,7 +72,7 @@ | ||||
| 		"uuid": "10.0.0", | ||||
| 		"v-code-diff": "1.12.0", | ||||
| 		"vite": "5.3.5", | ||||
| 		"vue": "3.4.34", | ||||
| 		"vue": "3.4.37", | ||||
| 		"vuedraggable": "next" | ||||
| 	}, | ||||
| 	"devDependencies": { | ||||
| @@ -111,7 +111,7 @@ | ||||
| 		"@typescript-eslint/eslint-plugin": "7.17.0", | ||||
| 		"@typescript-eslint/parser": "7.17.0", | ||||
| 		"@vitest/coverage-v8": "1.6.0", | ||||
| 		"@vue/runtime-core": "3.4.34", | ||||
| 		"@vue/runtime-core": "3.4.37", | ||||
| 		"acorn": "8.12.1", | ||||
| 		"cross-env": "7.0.3", | ||||
| 		"cypress": "13.13.1", | ||||
|   | ||||
| @@ -231,17 +231,18 @@ export async function mainBoot() { | ||||
| 			claimAchievement('client60min'); | ||||
| 		}, 1000 * 60 * 60); | ||||
|  | ||||
| 		const lastUsed = miLocalStorage.getItem('lastUsed'); | ||||
| 		if (lastUsed) { | ||||
| 			const lastUsedDate = parseInt(lastUsed, 10); | ||||
| 			// 二時間以上前なら | ||||
| 			if (Date.now() - lastUsedDate > 1000 * 60 * 60 * 2) { | ||||
| 				toast(i18n.tsx.welcomeBackWithName({ | ||||
| 					name: $i.name || $i.username, | ||||
| 				})); | ||||
| 			} | ||||
| 		} | ||||
| 		miLocalStorage.setItem('lastUsed', Date.now().toString()); | ||||
| 		// 邪魔 | ||||
| 		//const lastUsed = miLocalStorage.getItem('lastUsed'); | ||||
| 		//if (lastUsed) { | ||||
| 		//	const lastUsedDate = parseInt(lastUsed, 10); | ||||
| 		//	// 二時間以上前なら | ||||
| 		//	if (Date.now() - lastUsedDate > 1000 * 60 * 60 * 2) { | ||||
| 		//		toast(i18n.tsx.welcomeBackWithName({ | ||||
| 		//			name: $i.name || $i.username, | ||||
| 		//		})); | ||||
| 		//	} | ||||
| 		//} | ||||
| 		//miLocalStorage.setItem('lastUsed', Date.now().toString()); | ||||
|  | ||||
| 		const latestDonationInfoShownAt = miLocalStorage.getItem('latestDonationInfoShownAt'); | ||||
| 		const neverShowDonationInfo = miLocalStorage.getItem('neverShowDonationInfo'); | ||||
|   | ||||
| @@ -0,0 +1,7 @@ | ||||
| /* | ||||
|  * SPDX-FileCopyrightText: syuilo and misskey-project | ||||
|  * SPDX-License-Identifier: AGPL-3.0-only | ||||
|  */ | ||||
|  | ||||
| import MkDateSeparatedList from './MkDateSeparatedList.vue'; | ||||
| void MkDateSeparatedList; | ||||
							
								
								
									
										159
									
								
								packages/frontend/src/components/MkDialog.stories.impl.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										159
									
								
								packages/frontend/src/components/MkDialog.stories.impl.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,159 @@ | ||||
| /* | ||||
|  * SPDX-FileCopyrightText: syuilo and misskey-project | ||||
|  * SPDX-License-Identifier: AGPL-3.0-only | ||||
|  */ | ||||
|  | ||||
| import { action } from '@storybook/addon-actions'; | ||||
| import { expect, userEvent, waitFor, within } from '@storybook/test'; | ||||
| import { StoryObj } from '@storybook/vue3'; | ||||
| import { i18n } from '@/i18n.js'; | ||||
| import MkDialog from './MkDialog.vue'; | ||||
| const Base = { | ||||
| 	render(args) { | ||||
| 		return { | ||||
| 			components: { | ||||
| 				MkDialog, | ||||
| 			}, | ||||
| 			setup() { | ||||
| 				return { | ||||
| 					args, | ||||
| 				}; | ||||
| 			}, | ||||
| 			computed: { | ||||
| 				props() { | ||||
| 					return { | ||||
| 						...this.args, | ||||
| 					}; | ||||
| 				}, | ||||
| 				events() { | ||||
| 					return { | ||||
| 						done: action('done'), | ||||
| 						closed: action('closed'), | ||||
| 					}; | ||||
| 				}, | ||||
| 			}, | ||||
| 			template: '<MkDialog v-bind="props" v-on="events" />', | ||||
| 		}; | ||||
| 	}, | ||||
| 	args: { | ||||
| 		text: 'Hello, world!', | ||||
| 	}, | ||||
| 	parameters: { | ||||
| 		layout: 'centered', | ||||
| 	}, | ||||
| } satisfies StoryObj<typeof MkDialog>; | ||||
| export const Success = { | ||||
| 	...Base, | ||||
| 	args: { | ||||
| 		...Base.args, | ||||
| 		type: 'success', | ||||
| 	}, | ||||
| } satisfies StoryObj<typeof MkDialog>; | ||||
| export const Error = { | ||||
| 	...Base, | ||||
| 	args: { | ||||
| 		...Base.args, | ||||
| 		type: 'error', | ||||
| 	}, | ||||
| } satisfies StoryObj<typeof MkDialog>; | ||||
| export const Warning = { | ||||
| 	...Base, | ||||
| 	args: { | ||||
| 		...Base.args, | ||||
| 		type: 'warning', | ||||
| 	}, | ||||
| } satisfies StoryObj<typeof MkDialog>; | ||||
| export const Info = { | ||||
| 	...Base, | ||||
| 	args: { | ||||
| 		...Base.args, | ||||
| 		type: 'info', | ||||
| 	}, | ||||
| } satisfies StoryObj<typeof MkDialog>; | ||||
| export const Question = { | ||||
| 	...Base, | ||||
| 	args: { | ||||
| 		...Base.args, | ||||
| 		type: 'question', | ||||
| 	}, | ||||
| } satisfies StoryObj<typeof MkDialog>; | ||||
| export const Waiting = { | ||||
| 	...Base, | ||||
| 	args: { | ||||
| 		...Base.args, | ||||
| 		type: 'waiting', | ||||
| 	}, | ||||
| } satisfies StoryObj<typeof MkDialog>; | ||||
| export const DialogWithActions = { | ||||
| 	...Question, | ||||
| 	args: { | ||||
| 		...Question.args, | ||||
| 		text: i18n.ts.areYouSure, | ||||
| 		actions: [ | ||||
| 			{ | ||||
| 				text: i18n.ts.yes, | ||||
| 				primary: true, | ||||
| 				callback() { | ||||
| 					action('YES')(); | ||||
| 				}, | ||||
| 			}, | ||||
| 			{ | ||||
| 				text: i18n.ts.no, | ||||
| 				callback() { | ||||
| 					action('NO')(); | ||||
| 				}, | ||||
| 			}, | ||||
| 		], | ||||
| 	}, | ||||
| } satisfies StoryObj<typeof MkDialog>; | ||||
| export const DialogWithDangerActions = { | ||||
| 	...Warning, | ||||
| 	args: { | ||||
| 		...Warning.args, | ||||
| 		text: i18n.ts.resetAreYouSure, | ||||
| 		actions: [ | ||||
| 			{ | ||||
| 				text: i18n.ts.yes, | ||||
| 				danger: true, | ||||
| 				primary: true, | ||||
| 				callback() { | ||||
| 					action('YES')(); | ||||
| 				}, | ||||
| 			}, | ||||
| 			{ | ||||
| 				text: i18n.ts.no, | ||||
| 				callback() { | ||||
| 					action('NO')(); | ||||
| 				}, | ||||
| 			}, | ||||
| 		], | ||||
| 	}, | ||||
| } satisfies StoryObj<typeof MkDialog>; | ||||
| export const DialogWithInput = { | ||||
| 	...Question, | ||||
| 	args: { | ||||
| 		...Question.args, | ||||
| 		title: 'Hello, world!', | ||||
| 		text: undefined, | ||||
| 		input: { | ||||
| 			placeholder: i18n.ts.inputMessageHere, | ||||
| 			type: 'text', | ||||
| 			default: null, | ||||
| 			minLength: 2, | ||||
| 			maxLength: 3, | ||||
| 		}, | ||||
| 	}, | ||||
| 	async play({ canvasElement }) { | ||||
| 		const canvas = within(canvasElement); | ||||
| 		await expect(canvasElement).toHaveTextContent(i18n.tsx._dialog.charactersBelow({ current: 0, min: 2 })); | ||||
| 		const okButton = canvas.getByRole('button', { name: i18n.ts.ok }); | ||||
| 		await expect(okButton).toBeDisabled(); | ||||
| 		const input = canvas.getByRole<HTMLInputElement>('combobox'); | ||||
| 		await waitFor(() => userEvent.hover(input)); | ||||
| 		await waitFor(() => userEvent.click(input)); | ||||
| 		await waitFor(() => userEvent.type(input, 'M')); | ||||
| 		await expect(canvasElement).toHaveTextContent(i18n.tsx._dialog.charactersBelow({ current: 1, min: 2 })); | ||||
| 		await waitFor(() => userEvent.type(input, 'i')); | ||||
| 		await expect(okButton).toBeEnabled(); | ||||
| 	}, | ||||
| } satisfies StoryObj<typeof MkDialog>; | ||||
| @@ -0,0 +1,7 @@ | ||||
| /* | ||||
|  * SPDX-FileCopyrightText: syuilo and misskey-project | ||||
|  * SPDX-License-Identifier: AGPL-3.0-only | ||||
|  */ | ||||
|  | ||||
| import MkDivider from './MkDivider.vue'; | ||||
| void MkDivider; | ||||
							
								
								
									
										54
									
								
								packages/frontend/src/components/MkDonation.stories.impl.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										54
									
								
								packages/frontend/src/components/MkDonation.stories.impl.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,54 @@ | ||||
| /* | ||||
|  * SPDX-FileCopyrightText: syuilo and misskey-project | ||||
|  * SPDX-License-Identifier: AGPL-3.0-only | ||||
|  */ | ||||
|  | ||||
| import { action } from '@storybook/addon-actions'; | ||||
| import { StoryObj } from '@storybook/vue3'; | ||||
| import { onBeforeUnmount } from 'vue'; | ||||
| import MkDonation from './MkDonation.vue'; | ||||
| import { instance } from '@/instance.js'; | ||||
| export const Default = { | ||||
| 	render(args) { | ||||
| 		return { | ||||
| 			components: { | ||||
| 				MkDonation, | ||||
| 			}, | ||||
| 			setup() { | ||||
| 				return { | ||||
| 					args, | ||||
| 				}; | ||||
| 			}, | ||||
| 			computed: { | ||||
| 				props() { | ||||
| 					return { | ||||
| 						...this.args, | ||||
| 					}; | ||||
| 				}, | ||||
| 				events() { | ||||
| 					return { | ||||
| 						closed: action('closed'), | ||||
| 					}; | ||||
| 				}, | ||||
| 			}, | ||||
| 			template: '<MkDonation v-bind="props" v-on="events" />', | ||||
| 		}; | ||||
| 	}, | ||||
| 	args: { | ||||
| 		// @ts-expect-error name is used for mocking instance | ||||
| 		name: 'Misskey Hub', | ||||
| 	}, | ||||
| 	decorators: [ | ||||
| 		(_, { args }) => ({ | ||||
| 			setup() { | ||||
| 				// @ts-expect-error name is used for mocking instance | ||||
| 				instance.name = args.name; | ||||
| 				onBeforeUnmount(() => instance.name = null); | ||||
| 			}, | ||||
| 			template: '<story/>', | ||||
| 		}), | ||||
| 	], | ||||
| 	parameters: { | ||||
| 		layout: 'centered', | ||||
| 	}, | ||||
| } satisfies StoryObj<typeof MkDonation>; | ||||
| @@ -0,0 +1,48 @@ | ||||
| /* | ||||
|  * SPDX-FileCopyrightText: syuilo and misskey-project | ||||
|  * SPDX-License-Identifier: AGPL-3.0-only | ||||
|  */ | ||||
|  | ||||
| import { action } from '@storybook/addon-actions'; | ||||
| import { StoryObj } from '@storybook/vue3'; | ||||
| import MkDrive_file from './MkDrive.file.vue'; | ||||
| import { file } from '../../.storybook/fakes.js'; | ||||
| export const Default = { | ||||
| 	render(args) { | ||||
| 		return { | ||||
| 			components: { | ||||
| 				MkDrive_file, | ||||
| 			}, | ||||
| 			setup() { | ||||
| 				return { | ||||
| 					args, | ||||
| 				}; | ||||
| 			}, | ||||
| 			computed: { | ||||
| 				props() { | ||||
| 					return { | ||||
| 						...this.args, | ||||
| 					}; | ||||
| 				}, | ||||
| 				events() { | ||||
| 					return { | ||||
| 						chosen: action('chosen'), | ||||
| 						dragstart: action('dragstart'), | ||||
| 						dragend: action('dragend'), | ||||
| 					}; | ||||
| 				}, | ||||
| 			}, | ||||
| 			template: '<MkDrive_file v-bind="props" v-on="events" />', | ||||
| 		}; | ||||
| 	}, | ||||
| 	args: { | ||||
| 		file: file(), | ||||
| 	}, | ||||
| 	parameters: { | ||||
| 		chromatic: { | ||||
| 			// NOTE: ロードが終わるまで待つ | ||||
| 			delay: 3000, | ||||
| 		}, | ||||
| 		layout: 'centered', | ||||
| 	}, | ||||
| } satisfies StoryObj<typeof MkDrive_file>; | ||||
| @@ -0,0 +1,70 @@ | ||||
| /* | ||||
|  * SPDX-FileCopyrightText: syuilo and misskey-project | ||||
|  * SPDX-License-Identifier: AGPL-3.0-only | ||||
|  */ | ||||
|  | ||||
| import { action } from '@storybook/addon-actions'; | ||||
| import { StoryObj } from '@storybook/vue3'; | ||||
| import { http, HttpResponse } from 'msw'; | ||||
| import * as Misskey from 'misskey-js'; | ||||
| import MkDrive_folder from './MkDrive.folder.vue'; | ||||
| import { folder } from '../../.storybook/fakes.js'; | ||||
| import { commonHandlers } from '../../.storybook/mocks.js'; | ||||
| export const Default = { | ||||
| 	render(args) { | ||||
| 		return { | ||||
| 			components: { | ||||
| 				MkDrive_folder, | ||||
| 			}, | ||||
| 			setup() { | ||||
| 				return { | ||||
| 					args, | ||||
| 				}; | ||||
| 			}, | ||||
| 			computed: { | ||||
| 				props() { | ||||
| 					return { | ||||
| 						...this.args, | ||||
| 					}; | ||||
| 				}, | ||||
| 				events() { | ||||
| 					return { | ||||
| 						chosen: action('chosen'), | ||||
| 						move: action('move'), | ||||
| 						upload: action('upload'), | ||||
| 						removeFile: action('removeFile'), | ||||
| 						removeFolder: action('removeFolder'), | ||||
| 						dragstart: action('dragstart'), | ||||
| 						dragend: action('dragend'), | ||||
| 					}; | ||||
| 				}, | ||||
| 			}, | ||||
| 			template: '<MkDrive_folder v-bind="props" v-on="events" />', | ||||
| 		}; | ||||
| 	}, | ||||
| 	args: { | ||||
| 		folder: folder(), | ||||
| 	}, | ||||
| 	parameters: { | ||||
| 		layout: 'centered', | ||||
| 		msw: { | ||||
| 			handlers: [ | ||||
| 				...commonHandlers, | ||||
| 				http.post('/api/drive/folders/delete', async ({ request }) => { | ||||
| 					action('POST /api/drive/folders/delete')(await request.json()); | ||||
| 					return HttpResponse.json(undefined, { status: 204 }); | ||||
| 				}), | ||||
| 				http.post('/api/drive/folders/update', async ({ request }) => { | ||||
| 					const req = await request.json() as Misskey.entities.DriveFoldersUpdateRequest; | ||||
| 					action('POST /api/drive/folders/update')(req); | ||||
| 					return HttpResponse.json({ | ||||
| 						...folder(), | ||||
| 						id: req.folderId, | ||||
| 						name: req.name ?? folder().name, | ||||
| 						parentId: req.parentId ?? folder().parentId, | ||||
| 					}); | ||||
| 				}), | ||||
| 			], | ||||
| 		}, | ||||
| 	}, | ||||
| } satisfies StoryObj<typeof MkDrive_folder>; | ||||
| @@ -0,0 +1,7 @@ | ||||
| /* | ||||
|  * SPDX-FileCopyrightText: syuilo and misskey-project | ||||
|  * SPDX-License-Identifier: AGPL-3.0-only | ||||
|  */ | ||||
|  | ||||
| import MkDrive_navFolder from './MkDrive.navFolder.vue'; | ||||
| void MkDrive_navFolder; | ||||
							
								
								
									
										82
									
								
								packages/frontend/src/components/MkDrive.stories.impl.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										82
									
								
								packages/frontend/src/components/MkDrive.stories.impl.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,82 @@ | ||||
| /* | ||||
|  * SPDX-FileCopyrightText: syuilo and misskey-project | ||||
|  * SPDX-License-Identifier: AGPL-3.0-only | ||||
|  */ | ||||
|  | ||||
| import { action } from '@storybook/addon-actions'; | ||||
| import { StoryObj } from '@storybook/vue3'; | ||||
| import { http, HttpResponse } from 'msw'; | ||||
| import * as Misskey from 'misskey-js'; | ||||
| import MkDrive from './MkDrive.vue'; | ||||
| import { file, folder } from '../../.storybook/fakes.js'; | ||||
| import { commonHandlers } from '../../.storybook/mocks.js'; | ||||
| export const Default = { | ||||
| 	render(args) { | ||||
| 		return { | ||||
| 			components: { | ||||
| 				MkDrive, | ||||
| 			}, | ||||
| 			setup() { | ||||
| 				return { | ||||
| 					args, | ||||
| 				}; | ||||
| 			}, | ||||
| 			computed: { | ||||
| 				props() { | ||||
| 					return { | ||||
| 						...this.args, | ||||
| 					}; | ||||
| 				}, | ||||
| 				events() { | ||||
| 					return { | ||||
| 						selected: action('selected'), | ||||
| 						'change-selection': action('change-selection'), | ||||
| 						'move-root': action('move-root'), | ||||
| 						cd: action('cd'), | ||||
| 						'open-folder': action('open-folder'), | ||||
| 					}; | ||||
| 				}, | ||||
| 			}, | ||||
| 			template: '<MkDrive v-bind="props" v-on="events" />', | ||||
| 		}; | ||||
| 	}, | ||||
| 	parameters: { | ||||
| 		chromatic: { | ||||
| 			// NOTE: ロードが終わるまで待つ | ||||
| 			delay: 3000, | ||||
| 		}, | ||||
| 		layout: 'centered', | ||||
| 		msw: { | ||||
| 			handlers: [ | ||||
| 				...commonHandlers, | ||||
| 				http.post('/api/drive/files', async ({ request }) => { | ||||
| 					action('POST /api/drive/files')(await request.json()); | ||||
| 					return HttpResponse.json([file()]); | ||||
| 				}), | ||||
| 				http.post('/api/drive/folders', async ({ request }) => { | ||||
| 					action('POST /api/drive/folders')(await request.json()); | ||||
| 					return HttpResponse.json([folder(crypto.randomUUID())]); | ||||
| 				}), | ||||
| 				http.post('/api/drive/folders/create', async ({ request }) => { | ||||
| 					const req = await request.json() as Misskey.entities.DriveFoldersCreateRequest; | ||||
| 					action('POST /api/drive/folders/create')(req); | ||||
| 					return HttpResponse.json(folder(crypto.randomUUID(), req.name, req.parentId)); | ||||
| 				}), | ||||
| 				http.post('/api/drive/folders/delete', async ({ request }) => { | ||||
| 					action('POST /api/drive/folders/delete')(await request.json()); | ||||
| 					return HttpResponse.json(undefined, { status: 204 }); | ||||
| 				}), | ||||
| 				http.post('/api/drive/folders/update', async ({ request }) => { | ||||
| 					const req = await request.json() as Misskey.entities.DriveFoldersUpdateRequest; | ||||
| 					action('POST /api/drive/folders/update')(req); | ||||
| 					return HttpResponse.json({ | ||||
| 						...folder(), | ||||
| 						id: req.folderId, | ||||
| 						name: req.name ?? folder().name, | ||||
| 						parentId: req.parentId ?? folder().parentId, | ||||
| 					}); | ||||
| 				}), | ||||
| 			] | ||||
| 		}, | ||||
| 	}, | ||||
| } satisfies StoryObj<typeof MkDrive>; | ||||
| @@ -0,0 +1,41 @@ | ||||
| /* | ||||
|  * SPDX-FileCopyrightText: syuilo and misskey-project | ||||
|  * SPDX-License-Identifier: AGPL-3.0-only | ||||
|  */ | ||||
|  | ||||
| import { StoryObj } from '@storybook/vue3'; | ||||
| import MkDriveFileThumbnail from './MkDriveFileThumbnail.vue'; | ||||
| import { file } from '../../.storybook/fakes.js'; | ||||
| export const Default = { | ||||
| 	render(args) { | ||||
| 		return { | ||||
| 			components: { | ||||
| 				MkDriveFileThumbnail, | ||||
| 			}, | ||||
| 			setup() { | ||||
| 				return { | ||||
| 					args, | ||||
| 				}; | ||||
| 			}, | ||||
| 			computed: { | ||||
| 				props() { | ||||
| 					return { | ||||
| 						...this.args, | ||||
| 					}; | ||||
| 				}, | ||||
| 			}, | ||||
| 			template: '<MkDriveFileThumbnail v-bind="props" />', | ||||
| 		}; | ||||
| 	}, | ||||
| 	args: { | ||||
| 		file: file(), | ||||
| 		fit: 'contain', | ||||
| 	}, | ||||
| 	parameters: { | ||||
| 		chromatic: { | ||||
| 			// NOTE: ロードが終わるまで待つ | ||||
| 			delay: 3000, | ||||
| 		}, | ||||
| 		layout: 'centered', | ||||
| 	}, | ||||
| } satisfies StoryObj<typeof MkDriveFileThumbnail>; | ||||
| @@ -26,7 +26,7 @@ import ImgWithBlurhash from '@/components/MkImgWithBlurhash.vue'; | ||||
|  | ||||
| const props = defineProps<{ | ||||
| 	file: Misskey.entities.DriveFile; | ||||
| 	fit: string; | ||||
| 	fit: 'cover' | 'contain'; | ||||
| }>(); | ||||
|  | ||||
| const is = computed(() => { | ||||
|   | ||||
| @@ -0,0 +1,7 @@ | ||||
| /* | ||||
|  * SPDX-FileCopyrightText: syuilo and misskey-project | ||||
|  * SPDX-License-Identifier: AGPL-3.0-only | ||||
|  */ | ||||
|  | ||||
| import MkDriveSelectDialog from './MkDriveSelectDialog.vue'; | ||||
| void MkDriveSelectDialog; | ||||
| @@ -0,0 +1,7 @@ | ||||
| /* | ||||
|  * SPDX-FileCopyrightText: syuilo and misskey-project | ||||
|  * SPDX-License-Identifier: AGPL-3.0-only | ||||
|  */ | ||||
|  | ||||
| import MkDriveWindow from './MkDriveWindow.vue'; | ||||
| void MkDriveWindow; | ||||
| @@ -0,0 +1,7 @@ | ||||
| /* | ||||
|  * SPDX-FileCopyrightText: syuilo and misskey-project | ||||
|  * SPDX-License-Identifier: AGPL-3.0-only | ||||
|  */ | ||||
|  | ||||
| import MkEmojiPicker_section from './MkEmojiPicker.section.vue'; | ||||
| void MkEmojiPicker_section; | ||||
| @@ -0,0 +1,54 @@ | ||||
| /* | ||||
|  * SPDX-FileCopyrightText: syuilo and misskey-project | ||||
|  * SPDX-License-Identifier: AGPL-3.0-only | ||||
|  */ | ||||
|  | ||||
| import { action } from '@storybook/addon-actions'; | ||||
| import { expect, userEvent, waitFor, within } from '@storybook/test'; | ||||
| import { StoryObj } from '@storybook/vue3'; | ||||
| import { i18n } from '@/i18n.js'; | ||||
| import MkEmojiPicker from './MkEmojiPicker.vue'; | ||||
| export const Default = { | ||||
| 	render(args) { | ||||
| 		return { | ||||
| 			components: { | ||||
| 				MkEmojiPicker, | ||||
| 			}, | ||||
| 			setup() { | ||||
| 				return { | ||||
| 					args, | ||||
| 				}; | ||||
| 			}, | ||||
| 			computed: { | ||||
| 				props() { | ||||
| 					return { | ||||
| 						...this.args, | ||||
| 					}; | ||||
| 				}, | ||||
| 				events() { | ||||
| 					return { | ||||
| 						chosen: action('chosen'), | ||||
| 					}; | ||||
| 				}, | ||||
| 			}, | ||||
| 			template: '<MkEmojiPicker v-bind="props" v-on="events" />', | ||||
| 		}; | ||||
| 	}, | ||||
| 	async play({ canvasElement }) { | ||||
| 		const canvas = within(canvasElement); | ||||
| 		const faceSection = canvas.getByText(/face/i); | ||||
| 		await waitFor(() => userEvent.click(faceSection)); | ||||
| 		const grinning = canvasElement.querySelector('[data-emoji="😀"]'); | ||||
| 		await expect(grinning).toBeInTheDocument(); | ||||
| 		if (grinning == null) throw new Error(); // NOTE: not called | ||||
| 		await waitFor(() => userEvent.click(grinning)); | ||||
| 		const recentUsedSection = canvas.getByText(new RegExp(i18n.ts.recentUsed)).parentElement; | ||||
| 		await expect(recentUsedSection).toBeInTheDocument(); | ||||
| 		if (recentUsedSection == null) throw new Error(); // NOTE: not called | ||||
| 		await expect(within(recentUsedSection).getByAltText('😀')).toBeInTheDocument(); | ||||
| 		await expect(within(recentUsedSection).queryByAltText('😬')).toEqual(null); | ||||
| 	}, | ||||
| 	parameters: { | ||||
| 		layout: 'centered', | ||||
| 	}, | ||||
| } satisfies StoryObj<typeof MkEmojiPicker>; | ||||
| @@ -0,0 +1,7 @@ | ||||
| /* | ||||
|  * SPDX-FileCopyrightText: syuilo and misskey-project | ||||
|  * SPDX-License-Identifier: AGPL-3.0-only | ||||
|  */ | ||||
|  | ||||
| import MkEmojiPickerDialog from './MkEmojiPickerDialog.vue'; | ||||
| void MkEmojiPickerDialog; | ||||
| @@ -0,0 +1,53 @@ | ||||
| /* | ||||
|  * SPDX-FileCopyrightText: syuilo and misskey-project | ||||
|  * SPDX-License-Identifier: AGPL-3.0-only | ||||
|  */ | ||||
|  | ||||
| import { StoryObj } from '@storybook/vue3'; | ||||
| import MkFlashPreview from './MkFlashPreview.vue'; | ||||
| import { flash } from './../../.storybook/fakes.js'; | ||||
| export const Public = { | ||||
| 	render(args) { | ||||
| 		return { | ||||
| 			components: { | ||||
| 				MkFlashPreview, | ||||
| 			}, | ||||
| 			setup() { | ||||
| 				return { | ||||
| 					args, | ||||
| 				}; | ||||
| 			}, | ||||
| 			computed: { | ||||
| 				props() { | ||||
| 					return { | ||||
| 						...this.args, | ||||
| 					}; | ||||
| 				}, | ||||
| 			}, | ||||
| 			template: '<MkFlashPreview v-bind="props" />', | ||||
| 		}; | ||||
| 	}, | ||||
| 	args: { | ||||
| 		flash: { | ||||
| 			...flash(), | ||||
| 			visibility: 'public', | ||||
| 		}, | ||||
| 	}, | ||||
| 	parameters: { | ||||
| 		layout: 'fullscreen', | ||||
| 	}, | ||||
| 	decorators: [ | ||||
| 		() => ({ | ||||
| 			template: '<div style="display: flex; align-items: center; justify-content: center; height: 100vh"><div style="max-width: 700px; width: 100%; margin: 3rem"><story/></div></div>', | ||||
| 		}), | ||||
| 	], | ||||
| } satisfies StoryObj<typeof MkFlashPreview>; | ||||
| export const Private = { | ||||
| 	...Public, | ||||
| 	args: { | ||||
| 		flash: { | ||||
| 			...flash(), | ||||
| 			visibility: 'private', | ||||
| 		}, | ||||
| 	}, | ||||
| } satisfies StoryObj<typeof MkFlashPreview>; | ||||
| @@ -4,7 +4,7 @@ SPDX-License-Identifier: AGPL-3.0-only | ||||
| --> | ||||
|  | ||||
| <template> | ||||
| <MkA :to="`/play/${flash.id}`" class="vhpxefrk _panel"> | ||||
| <MkA :to="`/play/${flash.id}`" class="vhpxefrk _panel" :class="[{ gray: flash.visibility === 'private' }]"> | ||||
| 	<article> | ||||
| 		<header> | ||||
| 			<h1 :title="flash.title">{{ flash.title }}</h1> | ||||
| @@ -22,11 +22,11 @@ SPDX-License-Identifier: AGPL-3.0-only | ||||
|  | ||||
| <script lang="ts" setup> | ||||
| import { } from 'vue'; | ||||
| import * as Misskey from 'misskey-js'; | ||||
| import { userName } from '@/filters/user.js'; | ||||
|  | ||||
| const props = defineProps<{ | ||||
| 	//flash: Misskey.entities.Flash; | ||||
| 	flash: any; | ||||
| 	flash: Misskey.entities.Flash; | ||||
| }>(); | ||||
| </script> | ||||
|  | ||||
| @@ -91,6 +91,12 @@ const props = defineProps<{ | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	&:global(.gray) { | ||||
| 		--c: var(--bg); | ||||
| 		background-image: linear-gradient(45deg, var(--c) 16.67%, transparent 16.67%, transparent 50%, var(--c) 50%, var(--c) 66.67%, transparent 66.67%, transparent 100%); | ||||
| 		background-size: 16px 16px; | ||||
| 	} | ||||
|  | ||||
| 	@media (max-width: 700px) { | ||||
| 	} | ||||
|  | ||||
|   | ||||
| @@ -19,8 +19,8 @@ SPDX-License-Identifier: AGPL-3.0-only | ||||
| 		{{ mode === 'create' ? i18n.ts._webhookSettings.createWebhook : i18n.ts._webhookSettings.modifyWebhook }} | ||||
| 	</template> | ||||
|  | ||||
| 	<div> | ||||
| 		<MkSpacer :marginMin="20" :marginMax="28"> | ||||
| 	<div style="display: flex; flex-direction: column; min-height: 100%;"> | ||||
| 		<MkSpacer :marginMin="20" :marginMax="28" style="flex-grow: 1;"> | ||||
| 			<MkLoading v-if="loading !== 0"/> | ||||
| 			<div v-else :class="$style.root" class="_gaps_m"> | ||||
| 				<MkInput v-model="title"> | ||||
| @@ -33,7 +33,7 @@ SPDX-License-Identifier: AGPL-3.0-only | ||||
| 					<template #label>{{ i18n.ts._webhookSettings.secret }}</template> | ||||
| 				</MkInput> | ||||
| 				<MkFolder :defaultOpen="true"> | ||||
| 					<template #label>{{ i18n.ts._webhookSettings.events }}</template> | ||||
| 					<template #label>{{ i18n.ts._webhookSettings.trigger }}</template> | ||||
|  | ||||
| 					<div class="_gaps_s"> | ||||
| 						<MkSwitch v-model="events.abuseReport" :disabled="disabledEvents.abuseReport"> | ||||
| @@ -226,6 +226,7 @@ onMounted(async () => { | ||||
|  | ||||
| .footer { | ||||
| 	position: sticky; | ||||
| 	z-index: 10000; | ||||
| 	bottom: 0; | ||||
| 	left: 0; | ||||
| 	padding: 12px; | ||||
|   | ||||
| @@ -19,6 +19,7 @@ SPDX-License-Identifier: AGPL-3.0-only | ||||
| <script lang="ts" setup> | ||||
| import { computed, watch, onUnmounted, provide, ref, shallowRef } from 'vue'; | ||||
| import * as Misskey from 'misskey-js'; | ||||
| import type { BasicTimelineType } from '@/timelines.js'; | ||||
| import MkNotes from '@/components/MkNotes.vue'; | ||||
| import MkPullToRefresh from '@/components/MkPullToRefresh.vue'; | ||||
| import { useStream } from '@/stream.js'; | ||||
| @@ -29,7 +30,7 @@ import { defaultStore } from '@/store.js'; | ||||
| import { Paging } from '@/components/MkPagination.vue'; | ||||
|  | ||||
| const props = withDefaults(defineProps<{ | ||||
| 	src: 'home' | 'local' | 'social' | 'global' | 'mentions' | 'directs' | 'list' | 'antenna' | 'channel' | 'role'; | ||||
| 	src: BasicTimelineType | 'mentions' | 'directs' | 'list' | 'antenna' | 'channel' | 'role'; | ||||
| 	list?: string; | ||||
| 	antenna?: string; | ||||
| 	channel?: string; | ||||
|   | ||||
| @@ -7,10 +7,9 @@ SPDX-License-Identifier: AGPL-3.0-only | ||||
| <div class="_gaps"> | ||||
| 	<div style="text-align: center; padding: 0 16px;">{{ i18n.ts._initialTutorial._timeline.description1 }}</div> | ||||
| 	<div class="_gaps_s"> | ||||
| 		<div><i class="ti ti-home"></i> <b>{{ i18n.ts._timelines.home }}</b> … {{ i18n.ts._initialTutorial._timeline.home }}</div> | ||||
| 		<div><i class="ti ti-planet"></i> <b>{{ i18n.ts._timelines.local }}</b> … {{ i18n.ts._initialTutorial._timeline.local }}</div> | ||||
| 		<div><i class="ti ti-universe"></i> <b>{{ i18n.ts._timelines.social }}</b> … {{ i18n.ts._initialTutorial._timeline.social }}</div> | ||||
| 		<div><i class="ti ti-whirl"></i> <b>{{ i18n.ts._timelines.global }}</b> … {{ i18n.ts._initialTutorial._timeline.global }}</div> | ||||
| 		<div v-for="tl in basicTimelineTypes"> | ||||
| 			<i :class="basicTimelineIconClass(tl)"></i> <b>{{ i18n.ts._timelines[tl] }}</b> … {{ i18n.ts._initialTutorial._timeline[tl] }} | ||||
| 		</div> | ||||
| 	</div> | ||||
| 	<div class="_gaps_s"> | ||||
| 		<div>{{ i18n.ts._initialTutorial._timeline.description2 }}</div> | ||||
| @@ -22,12 +21,12 @@ SPDX-License-Identifier: AGPL-3.0-only | ||||
| 			<a href="https://misskey-hub.net/docs/for-users/features/timeline/" target="_blank" class="_link">{{ i18n.ts.help }}</a> | ||||
| 		</template> | ||||
| 	</I18n> | ||||
|  | ||||
| </div> | ||||
| </template> | ||||
|  | ||||
| <script setup lang="ts"> | ||||
| import { i18n } from '@/i18n.js'; | ||||
| import { basicTimelineIconClass, basicTimelineTypes } from '@/timelines.js'; | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" module> | ||||
|   | ||||
| @@ -258,6 +258,15 @@ const patronsWithIcon = [{ | ||||
| }, { | ||||
| 	name: 'えとゔぁす', | ||||
| 	icon: 'https://assets.misskey-hub.net/patrons/2578f441b82a44cfaa55ba83a318b26e.jpg', | ||||
| }, { | ||||
| 	name: 'Soli', | ||||
| 	icon: 'https://assets.misskey-hub.net/patrons/448070c81ebd41eda4ea2328291b2efe.jpg', | ||||
| }, { | ||||
| 	name: 'ささくれりょう', | ||||
| 	icon: 'https://assets.misskey-hub.net/patrons/cf55022cee6c41da8e70a43587aaad9a.jpg', | ||||
| }, { | ||||
| 	name: 'Macop', | ||||
| 	icon: 'https://assets.misskey-hub.net/patrons/ee052bf550014d36a643ce3dce595640.jpg', | ||||
| }]; | ||||
|  | ||||
| const patrons = [ | ||||
|   | ||||
| @@ -22,7 +22,7 @@ SPDX-License-Identifier: AGPL-3.0-only | ||||
| 					</div> | ||||
| 				</div> | ||||
|  | ||||
| 				<MkInfo v-if="user.username.includes('.')">{{ i18n.ts.isSystemAccount }}</MkInfo> | ||||
| 				<MkInfo v-if="['instance.actor', 'relay.actor'].includes(user.username)">{{ i18n.ts.isSystemAccount }}</MkInfo> | ||||
|  | ||||
| 				<FormLink v-if="user.host" :to="`/instance-info/${user.host}`">{{ i18n.ts.instanceInfo }}</FormLink> | ||||
|  | ||||
|   | ||||
| @@ -16,8 +16,8 @@ SPDX-License-Identifier: AGPL-3.0-only | ||||
| 	<template #header> | ||||
| 		{{ mode === 'create' ? i18n.ts._abuseReport._notificationRecipient.createRecipient : i18n.ts._abuseReport._notificationRecipient.modifyRecipient }} | ||||
| 	</template> | ||||
| 	<div v-if="loading === 0"> | ||||
| 		<MkSpacer :marginMin="20" :marginMax="28"> | ||||
| 	<div v-if="loading === 0" style="display: flex; flex-direction: column; min-height: 100%;"> | ||||
| 		<MkSpacer :marginMin="20" :marginMax="28" style="flex-grow: 1;"> | ||||
| 			<div :class="$style.root" class="_gaps_m"> | ||||
| 				<MkInput v-model="title"> | ||||
| 					<template #label>{{ i18n.ts.title }}</template> | ||||
| @@ -44,7 +44,7 @@ SPDX-License-Identifier: AGPL-3.0-only | ||||
| 								{{ webhook.name }} | ||||
| 							</option> | ||||
| 						</MkSelect> | ||||
| 						<MkButton rounded @click="onEditSystemWebhookClicked"> | ||||
| 						<MkButton rounded :class="$style.systemWebhookEditButton" @click="onEditSystemWebhookClicked"> | ||||
| 							<span v-if="systemWebhookId === null" class="ti ti-plus" style="line-height: normal"/> | ||||
| 							<span v-else class="ti ti-settings" style="line-height: normal"/> | ||||
| 						</MkButton> | ||||
| @@ -60,8 +60,8 @@ SPDX-License-Identifier: AGPL-3.0-only | ||||
| 		</MkSpacer> | ||||
|  | ||||
| 		<div :class="$style.footer" class="_buttonsCenter"> | ||||
| 			<MkButton primary :disabled="disableSubmitButton" @click="onSubmitClicked"><i class="ti ti-check"></i> {{ i18n.ts.ok }}</MkButton> | ||||
| 			<MkButton @click="onCancelClicked"><i class="ti ti-x"></i> {{ i18n.ts.cancel }}</MkButton> | ||||
| 			<MkButton primary rounded :disabled="disableSubmitButton" @click="onSubmitClicked"><i class="ti ti-check"></i> {{ i18n.ts.ok }}</MkButton> | ||||
| 			<MkButton rounded @click="onCancelClicked"><i class="ti ti-x"></i> {{ i18n.ts.cancel }}</MkButton> | ||||
| 		</div> | ||||
| 	</div> | ||||
| 	<div v-else> | ||||
| @@ -289,10 +289,15 @@ onMounted(async () => { | ||||
| } | ||||
|  | ||||
| .footer { | ||||
| 	display: flex; | ||||
| 	justify-content: center; | ||||
| 	align-items: flex-end; | ||||
| 	margin-top: 20px; | ||||
| 	position: sticky; | ||||
| 	z-index: 10000; | ||||
| 	bottom: 0; | ||||
| 	left: 0; | ||||
| 	padding: 12px; | ||||
| 	border-top: solid 0.5px var(--divider); | ||||
| 	background: var(--acrylicBg); | ||||
| 	-webkit-backdrop-filter: var(--blur, blur(15px)); | ||||
| 	backdrop-filter: var(--blur, blur(15px)); | ||||
| } | ||||
|  | ||||
| .systemWebhook { | ||||
| @@ -301,8 +306,9 @@ onMounted(async () => { | ||||
| 	justify-content: stretch; | ||||
| 	align-items: flex-end; | ||||
| 	gap: 8px; | ||||
| } | ||||
|  | ||||
| 	button { | ||||
| .systemWebhookEditButton { | ||||
| 	min-width: 0; | ||||
| 	min-height: 0; | ||||
| 	width: 34px; | ||||
| @@ -311,6 +317,5 @@ onMounted(async () => { | ||||
| 	box-sizing: border-box; | ||||
| 	margin: 1px 0; | ||||
| 	padding: 6px; | ||||
| 	} | ||||
| } | ||||
| </style> | ||||
|   | ||||
| @@ -7,7 +7,7 @@ SPDX-License-Identifier: AGPL-3.0-only | ||||
| <div ref="el" class="hiyeyicy" :class="{ wide: !narrow }"> | ||||
| 	<div v-if="!narrow || currentPage?.route.name == null" class="nav"> | ||||
| 		<MkSpacer :contentMax="700" :marginMin="16"> | ||||
| 			<div class="lxpfedzu"> | ||||
| 			<div class="lxpfedzu _gaps"> | ||||
| 				<div class="banner"> | ||||
| 					<img :src="instance.iconUrl || '/favicon.ico'" alt="" class="icon"/> | ||||
| 				</div> | ||||
| @@ -61,10 +61,10 @@ const narrow = ref(false); | ||||
| const view = ref(null); | ||||
| const el = ref<HTMLDivElement | null>(null); | ||||
| const pageProps = ref({}); | ||||
| let noMaintainerInformation = isEmpty(instance.maintainerName) || isEmpty(instance.maintainerEmail); | ||||
| let noBotProtection = !instance.disableRegistration && !instance.enableHcaptcha && !instance.enableRecaptcha && !instance.enableTurnstile; | ||||
| let noEmailServer = !instance.enableEmail; | ||||
| let noInquiryUrl = isEmpty(instance.inquiryUrl); | ||||
| const noMaintainerInformation = computed(() => isEmpty(instance.maintainerName) || isEmpty(instance.maintainerEmail)); | ||||
| const noBotProtection = computed(() => !instance.disableRegistration && !instance.enableHcaptcha && !instance.enableRecaptcha && !instance.enableTurnstile && !instance.enableMcaptcha); | ||||
| const noEmailServer = computed(() => !instance.enableEmail); | ||||
| const noInquiryUrl = computed(() => isEmpty(instance.inquiryUrl)); | ||||
| const thereIsUnresolvedAbuseReport = ref(false); | ||||
| const currentPage = computed(() => router.currentRef.value.child); | ||||
|  | ||||
| @@ -235,25 +235,22 @@ const menuDef = computed(() => [{ | ||||
| 	}], | ||||
| }]); | ||||
|  | ||||
| watch(narrow.value, () => { | ||||
| 	if (currentPage.value?.route.name == null && !narrow.value) { | ||||
| 		router.push('/admin/overview'); | ||||
| 	} | ||||
| }); | ||||
|  | ||||
| onMounted(() => { | ||||
| 	if (el.value != null) { | ||||
| 		ro.observe(el.value); | ||||
|  | ||||
| 		narrow.value = el.value.offsetWidth < NARROW_THRESHOLD; | ||||
| 	} | ||||
| 	if (currentPage.value?.route.name == null && !narrow.value) { | ||||
| 		router.push('/admin/overview'); | ||||
| 		router.replace('/admin/overview'); | ||||
| 	} | ||||
| }); | ||||
|  | ||||
| onActivated(() => { | ||||
| 	if (el.value != null) { | ||||
| 		narrow.value = el.value.offsetWidth < NARROW_THRESHOLD; | ||||
| 	} | ||||
| 	if (currentPage.value?.route.name == null && !narrow.value) { | ||||
| 		router.push('/admin/overview'); | ||||
| 		router.replace('/admin/overview'); | ||||
| 	} | ||||
| }); | ||||
|  | ||||
|   | ||||
| @@ -8,14 +8,22 @@ SPDX-License-Identifier: AGPL-3.0-only | ||||
| 	<template #header><XHeader v-model:tab="tab" :actions="headerActions" :tabs="headerTabs"/></template> | ||||
| 	<MkSpacer :contentMax="700" :marginMin="16" :marginMax="32"> | ||||
| 		<FormSuspense :p="init"> | ||||
| 			<MkTextarea v-if="tab === 'block'" v-model="blockedHosts"> | ||||
| 			<template v-if="tab === 'block'"> | ||||
| 				<MkTextarea v-model="blockedHosts"> | ||||
| 					<span>{{ i18n.ts.blockedInstances }}</span> | ||||
| 					<template #caption>{{ i18n.ts.blockedInstancesDescription }}</template> | ||||
| 				</MkTextarea> | ||||
| 			<MkTextarea v-else-if="tab === 'silence'" v-model="silencedHosts" class="_formBlock"> | ||||
| 			</template> | ||||
| 			<template v-else-if="tab === 'silence'"> | ||||
| 				<MkTextarea v-model="silencedHosts" class="_formBlock"> | ||||
| 					<span>{{ i18n.ts.silencedInstances }}</span> | ||||
| 					<template #caption>{{ i18n.ts.silencedInstancesDescription }}</template> | ||||
| 				</MkTextarea> | ||||
| 				<MkTextarea v-model="mediaSilencedHosts" class="_formBlock"> | ||||
| 					<span>{{ i18n.ts.mediaSilencedInstances }}</span> | ||||
| 					<template #caption>{{ i18n.ts.mediaSilencedInstancesDescription }}</template> | ||||
| 				</MkTextarea> | ||||
| 			</template> | ||||
| 			<MkButton primary @click="save"><i class="ti ti-device-floppy"></i> {{ i18n.ts.save }}</MkButton> | ||||
| 		</FormSuspense> | ||||
| 	</MkSpacer> | ||||
| @@ -36,18 +44,21 @@ import { definePageMetadata } from '@/scripts/page-metadata.js'; | ||||
|  | ||||
| const blockedHosts = ref<string>(''); | ||||
| const silencedHosts = ref<string>(''); | ||||
| const mediaSilencedHosts = ref<string>(''); | ||||
| const tab = ref('block'); | ||||
|  | ||||
| async function init() { | ||||
| 	const meta = await misskeyApi('admin/meta'); | ||||
| 	blockedHosts.value = meta.blockedHosts.join('\n'); | ||||
| 	silencedHosts.value = meta.silencedHosts.join('\n'); | ||||
| 	mediaSilencedHosts.value = meta.mediaSilencedHosts.join('\n'); | ||||
| } | ||||
|  | ||||
| function save() { | ||||
| 	os.apiWithDialog('admin/update-meta', { | ||||
| 		blockedHosts: blockedHosts.value.split('\n') || [], | ||||
| 		silencedHosts: silencedHosts.value.split('\n') || [], | ||||
| 		mediaSilencedHosts: mediaSilencedHosts.value.split('\n') || [], | ||||
|  | ||||
| 	}).then(() => { | ||||
| 		fetchInstance(true); | ||||
|   | ||||
| @@ -21,12 +21,12 @@ SPDX-License-Identifier: AGPL-3.0-only | ||||
| 				].includes(log.type), | ||||
| 				[$style.logYellow]: [ | ||||
| 					'markSensitiveDriveFile', | ||||
| 					'resetPassword' | ||||
| 					'resetPassword', | ||||
| 					'suspendRemoteInstance', | ||||
| 				].includes(log.type), | ||||
| 				[$style.logRed]: [ | ||||
| 					'suspend', | ||||
| 					'deleteRole', | ||||
| 					'suspendRemoteInstance', | ||||
| 					'deleteGlobalAnnouncement', | ||||
| 					'deleteUserAnnouncement', | ||||
| 					'deleteCustomEmoji', | ||||
| @@ -36,6 +36,7 @@ SPDX-License-Identifier: AGPL-3.0-only | ||||
| 					'deleteAvatarDecoration', | ||||
| 					'deleteSystemWebhook', | ||||
| 					'deleteAbuseReportNotificationRecipient', | ||||
| 					'deleteAccount', | ||||
| 				].includes(log.type) | ||||
| 			}" | ||||
| 		>{{ i18n.ts._moderationLogTypes[log.type] }}</b> | ||||
| @@ -72,6 +73,7 @@ SPDX-License-Identifier: AGPL-3.0-only | ||||
| 		<span v-else-if="log.type === 'createAbuseReportNotificationRecipient'">: {{ log.info.recipient.name }}</span> | ||||
| 		<span v-else-if="log.type === 'updateAbuseReportNotificationRecipient'">: {{ log.info.before.name }}</span> | ||||
| 		<span v-else-if="log.type === 'deleteAbuseReportNotificationRecipient'">: {{ log.info.recipient.name }}</span> | ||||
| 		<span v-else-if="log.type === 'deleteAccount'">: @{{ log.info.userUsername }}{{ log.info.userHost ? '@' + log.info.userHost : '' }}</span> | ||||
| 	</template> | ||||
| 	<template #icon> | ||||
| 		<MkAvatar :user="log.user" :class="$style.avatar"/> | ||||
| @@ -143,7 +145,6 @@ SPDX-License-Identifier: AGPL-3.0-only | ||||
| 			</div> | ||||
| 		</template> | ||||
| 		<template v-else-if="log.type === 'updateRemoteInstanceNote'"> | ||||
| 			<div>{{ i18n.ts.user }}: {{ log.info.userId }}</div> | ||||
| 			<div :class="$style.diff"> | ||||
| 				<CodeDiff :context="5" :hideHeader="true" :oldString="log.info.before ?? ''" :newString="log.info.after ?? ''" maxHeight="300px"/> | ||||
| 			</div> | ||||
|   | ||||
| @@ -33,11 +33,11 @@ SPDX-License-Identifier: AGPL-3.0-only | ||||
| 					</MkSelect> | ||||
| 				</div> | ||||
| 				<div :class="$style.inputs"> | ||||
| 					<MkInput v-model="searchUsername" style="flex: 1;" type="text" :spellcheck="false" @update:modelValue="$refs.users.reload()"> | ||||
| 					<MkInput v-model="searchUsername" style="flex: 1;" type="text" :spellcheck="false"> | ||||
| 						<template #prefix>@</template> | ||||
| 						<template #label>{{ i18n.ts.username }}</template> | ||||
| 					</MkInput> | ||||
| 					<MkInput v-model="searchHost" style="flex: 1;" type="text" :spellcheck="false" :disabled="pagination.params.origin === 'local'" @update:modelValue="$refs.users.reload()"> | ||||
| 					<MkInput v-model="searchHost" style="flex: 1;" type="text" :spellcheck="false" :disabled="pagination.params.origin === 'local'"> | ||||
| 						<template #prefix>@</template> | ||||
| 						<template #label>{{ i18n.ts.host }}</template> | ||||
| 					</MkInput> | ||||
|   | ||||
| @@ -15,8 +15,8 @@ SPDX-License-Identifier: AGPL-3.0-only | ||||
| 	<template v-if="emoji" #header>:{{ emoji.name }}:</template> | ||||
| 	<template v-else #header>New emoji</template> | ||||
|  | ||||
| 	<div> | ||||
| 		<MkSpacer :marginMin="20" :marginMax="28"> | ||||
| 	<div style="display: flex; flex-direction: column; min-height: 100%;"> | ||||
| 		<MkSpacer :marginMin="20" :marginMax="28" style="flex-grow: 1;"> | ||||
| 			<div class="_gaps_m"> | ||||
| 				<div v-if="imgUrl != null" :class="$style.imgs"> | ||||
| 					<div style="background: #000;" :class="$style.imgContainer"> | ||||
| @@ -239,6 +239,7 @@ async function del() { | ||||
|  | ||||
| .footer { | ||||
| 	position: sticky; | ||||
| 	z-index: 10000; | ||||
| 	bottom: 0; | ||||
| 	left: 0; | ||||
| 	padding: 12px; | ||||
|   | ||||
| @@ -369,7 +369,6 @@ const props = defineProps<{ | ||||
| }>(); | ||||
|  | ||||
| const flash = ref<Misskey.entities.Flash | null>(null); | ||||
| const visibility = ref<'private' | 'public'>('public'); | ||||
|  | ||||
| if (props.id) { | ||||
| 	flash.value = await misskeyApi('flash/show', { | ||||
| @@ -380,6 +379,7 @@ if (props.id) { | ||||
| const title = ref(flash.value?.title ?? 'New Play'); | ||||
| const summary = ref(flash.value?.summary ?? ''); | ||||
| const permissions = ref(flash.value?.permissions ?? []); | ||||
| const visibility = ref<'private' | 'public'>(flash.value?.visibility ?? 'public'); | ||||
| const script = ref(flash.value?.script ?? PRESET_DEFAULT); | ||||
|  | ||||
| function selectPreset(ev: MouseEvent) { | ||||
|   | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user