Compare commits
	
		
			246 Commits
		
	
	
		
			2023.9.0-b
			...
			tl-push
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | 77498f84d8 | ||
|   | 0575207463 | ||
|   | 5ee93dc4a2 | ||
|   | 10ae0b329a | ||
|   | 7a3dd400d8 | ||
|   | e840544dd2 | ||
|   | 0e58f515fd | ||
|   | c3714c02ba | ||
|   | 9de11da170 | ||
|   | e12943c15b | ||
|   | 7022b16bce | ||
|   | ee3d40bc1b | ||
|   | 878e73cd37 | ||
|   | 96da6e28ea | ||
|   | 45c3ab2142 | ||
|   | 000abcd2f0 | ||
|   | e00fdc2d59 | ||
|   | 58eec94250 | ||
|   | b66df850e5 | ||
|   | 72d5b1f4ae | ||
|   | 15caa375a5 | ||
|   | 0e302c69bd | ||
|   | 236eed94bb | ||
|   | 6d68cfd1e3 | ||
|   | ea0d050b71 | ||
|   | d6ff810560 | ||
|   | aad48b4b24 | ||
|   | 2f00e4b2b1 | ||
|   | 152047ca14 | ||
|   | 58d2512d0e | ||
|   | bbcda73af8 | ||
|   | 880448d068 | ||
|   | 55e5056216 | ||
|   | d40f35b3ad | ||
|   | 8843669684 | ||
|   | c7c4c7807a | ||
|   | 79e5075564 | ||
|   | caca0da912 | ||
|   | 35e743c955 | ||
|   | 6f17993cba | ||
|   | e4de402ca1 | ||
|   | 0db117b0ab | ||
|   | cb821d42a6 | ||
|   | b4c1de11f5 | ||
|   | 3924a9e494 | ||
|   | d9aac112d3 | ||
|   | 7f4c00541c | ||
|   | 85430fd889 | ||
|   | f0a2c3ce76 | ||
|   | c019e9cad5 | ||
|   | 167aaabf20 | ||
|   | 72f7413f40 | ||
|   | 783a97fe06 | ||
|   | 3dd3c69303 | ||
|   | 06cfe618bb | ||
|   | 6840434661 | ||
|   | 09dfb9bde3 | ||
|   | b0714cbd7b | ||
|   | d0917aac1a | ||
|   | ff6600da2e | ||
|   | 7e74cff126 | ||
|   | e53749773e | ||
|   | 392de4df36 | ||
|   | cc6a96e1c9 | ||
|   | 0e681f3cc4 | ||
|   | a512915a84 | ||
|   | 5edc885c22 | ||
|   | e5c339b86a | ||
|   | d92e2b6ae0 | ||
|   | eb38f08e13 | ||
|   | f269841a83 | ||
|   | b55ffa2cbe | ||
|   | 0b0e58d405 | ||
|   | b349d0baf8 | ||
|   | 961f5a0caa | ||
|   | ac19b055c7 | ||
|   | eb23fd4e60 | ||
|   | fbab67df35 | ||
|   | 2529830bca | ||
|   | c01731f091 | ||
|   | 9771f1c435 | ||
|   | 424bb78387 | ||
|   | 9c448055a3 | ||
|   | b9da1415a5 | ||
|   | 4216a67462 | ||
|   | 7ce86a6196 | ||
|   | 2438c047a7 | ||
|   | c106db89e1 | ||
|   | a388e25f3e | ||
|   | 63c6a9bb80 | ||
|   | 772d2432b6 | ||
|   | eb740e2c72 | ||
|   | d854942a1f | ||
|   | ce1218a2b2 | ||
|   | d860e53b67 | ||
|   | 055464a624 | ||
|   | 9d0c077311 | ||
|   | 440f3144ae | ||
|   | 5ad0906c89 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 2039e244c5 | ||
|   | bd19d75c9c | ||
|   | ee44f35fea | ||
|   | 89edf8f81e | ||
|   | ece5469277 | ||
|   | 576158e883 | ||
|   | dcaea66dbf | ||
|   | 5318532a8d | ||
|   | 646a8d1a54 | ||
|   | dc8ab01168 | ||
|   | 281369d8c5 | ||
|   | 65aef45050 | ||
|   | 48314a39e0 | ||
|   | fe570fe16b | ||
|   | cf573add27 | ||
|   | 4a7f6e6de4 | ||
|   | 00659220a5 | ||
|   | 51546ad1ce | ||
|   | 80d52f65eb | ||
|   | 841e6ff901 | ||
|   | 82a51d49a0 | ||
|   | 30b231225c | ||
|   | d05563c448 | ||
|   | 03c868b727 | ||
|   | 8d2fb99662 | ||
|   | 20689638db | ||
|   | 2b561d2648 | ||
|   | 509cea511c | ||
|   | 72075314a8 | ||
|   | 7a3ddc869e | ||
|   | eb7c65ccb3 | ||
|   | 8e5a90589d | ||
|   | ed983a5baf | ||
|   | 2ad3b1fd74 | ||
|   | ed53b5f9bc | ||
|   | 19bc9c20a6 | ||
|   | fdf149cf52 | ||
|   | 76c4fedb7f | ||
|   | c3ccec723f | ||
|   | 7893da4d99 | ||
|   | 531c61ed2b | ||
|   | b60b214c0c | ||
|   | 10924fd229 | ||
|   | 9e4d3ebe5f | ||
|   | ba6e85482e | ||
|   | 504541a158 | ||
|   | 5a4cf059ee | ||
|   | d8a023063d | ||
|   | 8f77350089 | ||
|   | 063d24ad4f | ||
|   | cac1c2f1e9 | ||
|   | ad8ddbf12f | ||
|   | e8a098af62 | ||
|   | c25d66316c | ||
|   | 98209be01a | ||
|   | 1924bd20bb | ||
|   | 3085739e0e | ||
|   | d2ae80dd21 | ||
|   | f748c9144c | ||
|   | 526b3ae0e4 | ||
|   | 90a5511a54 | ||
|   | 032b6c6afb | ||
|   | ee83b9542e | ||
|   | 3bbc2e55b1 | ||
|   | 7dc9fe4e24 | ||
|   | c836157edb | ||
|   | eca8c7a52f | ||
|   | 03b5acf17f | ||
|   | e3f151e230 | ||
|   | f195fa4ab9 | ||
|   | 51c3ef5561 | ||
|   | b654446f93 | ||
|   | e41619775f | ||
|   | 1250309a69 | ||
|   | 6459eadcf1 | ||
|   | 1dddc68709 | ||
|   | f9916d216a | ||
|   | 9abda93811 | ||
|   | 8e2d47b2e8 | ||
|   | b9c6992aac | ||
|   | cb026a7512 | ||
|   | fa13b815ef | ||
|   | 5d65e34078 | ||
|   | 0183d24786 | ||
|   | 053da10e94 | ||
|   | c3db55b5b6 | ||
|   | bb460a1785 | ||
|   | bb0b2df37e | ||
|   | 934e4be658 | ||
|   | 09c00d0a1b | ||
|   | e98fbfeec1 | ||
|   | 578b0ebe0c | ||
|   | b0f6c44f36 | ||
|   | 299c9c4118 | ||
|   | bec338aa00 | ||
|   | 5c48878dc5 | ||
|   | 44985ae858 | ||
|   | aa80cfdb81 | ||
|   | 25ae4bca9c | ||
|   | 350ebbadba | ||
|   | 1b463d9c31 | ||
|   | a2d58d9f48 | ||
|   | 3d61ca818b | ||
|   | 60f3cc6f07 | ||
|   | 52ec1b3fde | ||
|   | 0dca6afa1f | ||
|   | 0260a6af85 | ||
|   | fba08c6310 | ||
|   | f7c6932a83 | ||
|   | 907d519da3 | ||
|   | 5c6b7991ef | ||
|   | 9eb2bc1987 | ||
|   | 5488a8fe88 | ||
|   | ba59355716 | ||
|   | 814e28459e | ||
|   | d2831c612f | ||
|   | 1eebf3c921 | ||
|   | a692acec1e | ||
|   | 6cf466e5d1 | ||
|   | cf7c6558ae | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | efa66ae89a | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 6124772a5f | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d28fe24d0b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ba28f90fd1 | ||
|   | 0d505f8131 | ||
|   | d869481db9 | ||
|   | 3456680e1d | ||
|   | 42c7aad251 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | f9fc743c05 | ||
|   | 25e030a707 | ||
|   | cd6428715e | ||
|   | 98e40e666c | ||
|   | 74faa01db8 | ||
|   | 7bd0a5b7cb | ||
|   | c0838c473f | ||
|   | 85078601c2 | ||
|   | b434beb5e2 | ||
|   | 295665a177 | ||
|   | fd7d7318a7 | ||
|   | 054ba3fea5 | ||
|   | 8749716700 | ||
|   | 55d392818c | ||
|   | 19b10ca803 | ||
|   | cd7ab326cd | ||
|   | ff9a65e8fa | ||
|   | bc52d7a4fb | ||
|   | af7e129b1e | 
| @@ -95,6 +95,14 @@ redis: | |||||||
| #  #prefix: example-prefix | #  #prefix: example-prefix | ||||||
| #  #db: 1 | #  #db: 1 | ||||||
|  |  | ||||||
|  | #redisForTimelines: | ||||||
|  | #  host: redis | ||||||
|  | #  port: 6379 | ||||||
|  | #  #family: 0  # 0=Both, 4=IPv4, 6=IPv6 | ||||||
|  | #  #pass: example-pass | ||||||
|  | #  #prefix: example-prefix | ||||||
|  | #  #db: 1 | ||||||
|  |  | ||||||
| #   ┌───────────────────────────┐ | #   ┌───────────────────────────┐ | ||||||
| #───┘ MeiliSearch configuration └───────────────────────────── | #───┘ MeiliSearch configuration └───────────────────────────── | ||||||
|  |  | ||||||
|   | |||||||
| @@ -105,6 +105,16 @@ redis: | |||||||
| #  # You can specify more ioredis options... | #  # You can specify more ioredis options... | ||||||
| #  #username: example-username | #  #username: example-username | ||||||
|  |  | ||||||
|  | #redisForTimelines: | ||||||
|  | #  host: localhost | ||||||
|  | #  port: 6379 | ||||||
|  | #  #family: 0  # 0=Both, 4=IPv4, 6=IPv6 | ||||||
|  | #  #pass: example-pass | ||||||
|  | #  #prefix: example-prefix | ||||||
|  | #  #db: 1 | ||||||
|  | #  # You can specify more ioredis options... | ||||||
|  | #  #username: example-username | ||||||
|  |  | ||||||
| #   ┌───────────────────────────┐ | #   ┌───────────────────────────┐ | ||||||
| #───┘ MeiliSearch configuration └───────────────────────────── | #───┘ MeiliSearch configuration └───────────────────────────── | ||||||
|  |  | ||||||
| @@ -206,3 +216,6 @@ signToActivityPubGet: true | |||||||
|  |  | ||||||
| # Upload or download file size limits (bytes) | # Upload or download file size limits (bytes) | ||||||
| #maxFileSize: 262144000 | #maxFileSize: 262144000 | ||||||
|  |  | ||||||
|  | # PID File of master process | ||||||
|  | #pidFile: /tmp/misskey.pid | ||||||
|   | |||||||
| @@ -95,6 +95,14 @@ redis: | |||||||
| #  #prefix: example-prefix | #  #prefix: example-prefix | ||||||
| #  #db: 1 | #  #db: 1 | ||||||
|  |  | ||||||
|  | #redisForTimelines: | ||||||
|  | #  host: redis | ||||||
|  | #  port: 6379 | ||||||
|  | #  #family: 0  # 0=Both, 4=IPv4, 6=IPv6 | ||||||
|  | #  #pass: example-pass | ||||||
|  | #  #prefix: example-prefix | ||||||
|  | #  #db: 1 | ||||||
|  |  | ||||||
| #   ┌───────────────────────────┐ | #   ┌───────────────────────────┐ | ||||||
| #───┘ MeiliSearch configuration └───────────────────────────── | #───┘ MeiliSearch configuration └───────────────────────────── | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										9
									
								
								.github/reviewer-lottery.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										9
									
								
								.github/reviewer-lottery.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,9 +0,0 @@ | |||||||
| groups: |  | ||||||
|   - name: devs |  | ||||||
|     reviewers: 2 |  | ||||||
|     internal_reviewers: 1 |  | ||||||
|     usernames: |  | ||||||
|       - syuilo |  | ||||||
|       - acid-chicken |  | ||||||
|       - EbiseLutica |  | ||||||
|       - tamaina |  | ||||||
							
								
								
									
										2
									
								
								.github/workflows/api-misskey-js.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/api-misskey-js.yml
									
									
									
									
										vendored
									
									
								
							| @@ -9,7 +9,7 @@ jobs: | |||||||
|  |  | ||||||
|     steps: |     steps: | ||||||
|       - name: Checkout |       - name: Checkout | ||||||
|         uses: actions/checkout@v4.0.0 |         uses: actions/checkout@v4.1.0 | ||||||
|  |  | ||||||
|       - run: corepack enable |       - run: corepack enable | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								.github/workflows/check_copyright_year.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/check_copyright_year.yml
									
									
									
									
										vendored
									
									
								
							| @@ -10,7 +10,7 @@ jobs: | |||||||
|   check_copyright_year: |   check_copyright_year: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|     - uses: actions/checkout@v4.0.0 |     - uses: actions/checkout@v4.1.0 | ||||||
|     - run: | |     - run: | | ||||||
|         if [ "$(grep Copyright COPYING | sed -e 's/.*2014-\([0-9]*\) .*/\1/g')" -ne "$(date +%Y)" ]; then |         if [ "$(grep Copyright COPYING | sed -e 's/.*2014-\([0-9]*\) .*/\1/g')" -ne "$(date +%Y)" ]; then | ||||||
|           echo "Please change copyright year!" |           echo "Please change copyright year!" | ||||||
|   | |||||||
							
								
								
									
										10
									
								
								.github/workflows/docker-develop.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										10
									
								
								.github/workflows/docker-develop.yml
									
									
									
									
										vendored
									
									
								
							| @@ -13,24 +13,24 @@ jobs: | |||||||
|     if: github.repository == 'misskey-dev/misskey' |     if: github.repository == 'misskey-dev/misskey' | ||||||
|     steps: |     steps: | ||||||
|       - name: Check out the repo |       - name: Check out the repo | ||||||
|         uses: actions/checkout@v4.0.0 |         uses: actions/checkout@v4.1.0 | ||||||
|       - name: Set up Docker Buildx |       - name: Set up Docker Buildx | ||||||
|         id: buildx |         id: buildx | ||||||
|         uses: docker/setup-buildx-action@v2.10.0 |         uses: docker/setup-buildx-action@v3.0.0 | ||||||
|         with: |         with: | ||||||
|           platforms: linux/amd64,linux/arm64 |           platforms: linux/amd64,linux/arm64 | ||||||
|       - name: Docker meta |       - name: Docker meta | ||||||
|         id: meta |         id: meta | ||||||
|         uses: docker/metadata-action@v4 |         uses: docker/metadata-action@v5 | ||||||
|         with: |         with: | ||||||
|           images: misskey/misskey |           images: misskey/misskey | ||||||
|       - name: Log in to Docker Hub |       - name: Log in to Docker Hub | ||||||
|         uses: docker/login-action@v2 |         uses: docker/login-action@v3 | ||||||
|         with: |         with: | ||||||
|           username: ${{ secrets.DOCKER_USERNAME }} |           username: ${{ secrets.DOCKER_USERNAME }} | ||||||
|           password: ${{ secrets.DOCKER_PASSWORD }} |           password: ${{ secrets.DOCKER_PASSWORD }} | ||||||
|       - name: Build and Push to Docker Hub |       - name: Build and Push to Docker Hub | ||||||
|         uses: docker/build-push-action@v4 |         uses: docker/build-push-action@v5 | ||||||
|         with: |         with: | ||||||
|           builder: ${{ steps.buildx.outputs.name }} |           builder: ${{ steps.buildx.outputs.name }} | ||||||
|           context: . |           context: . | ||||||
|   | |||||||
							
								
								
									
										10
									
								
								.github/workflows/docker.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										10
									
								
								.github/workflows/docker.yml
									
									
									
									
										vendored
									
									
								
							| @@ -12,15 +12,15 @@ jobs: | |||||||
|  |  | ||||||
|     steps: |     steps: | ||||||
|       - name: Check out the repo |       - name: Check out the repo | ||||||
|         uses: actions/checkout@v4.0.0 |         uses: actions/checkout@v4.1.0 | ||||||
|       - name: Set up Docker Buildx |       - name: Set up Docker Buildx | ||||||
|         id: buildx |         id: buildx | ||||||
|         uses: docker/setup-buildx-action@v2.10.0 |         uses: docker/setup-buildx-action@v3.0.0 | ||||||
|         with: |         with: | ||||||
|           platforms: linux/amd64,linux/arm64 |           platforms: linux/amd64,linux/arm64 | ||||||
|       - name: Docker meta |       - name: Docker meta | ||||||
|         id: meta |         id: meta | ||||||
|         uses: docker/metadata-action@v4 |         uses: docker/metadata-action@v5 | ||||||
|         with: |         with: | ||||||
|           images: misskey/misskey |           images: misskey/misskey | ||||||
|           tags: | |           tags: | | ||||||
| @@ -31,12 +31,12 @@ jobs: | |||||||
|             type=semver,pattern={{major}}.{{minor}} |             type=semver,pattern={{major}}.{{minor}} | ||||||
|             type=semver,pattern={{major}} |             type=semver,pattern={{major}} | ||||||
|       - name: Log in to Docker Hub |       - name: Log in to Docker Hub | ||||||
|         uses: docker/login-action@v2 |         uses: docker/login-action@v3 | ||||||
|         with: |         with: | ||||||
|           username: ${{ secrets.DOCKER_USERNAME }} |           username: ${{ secrets.DOCKER_USERNAME }} | ||||||
|           password: ${{ secrets.DOCKER_PASSWORD }} |           password: ${{ secrets.DOCKER_PASSWORD }} | ||||||
|       - name: Build and Push to Docker Hub |       - name: Build and Push to Docker Hub | ||||||
|         uses: docker/build-push-action@v4 |         uses: docker/build-push-action@v5 | ||||||
|         with: |         with: | ||||||
|           builder: ${{ steps.buildx.outputs.name }} |           builder: ${{ steps.buildx.outputs.name }} | ||||||
|           context: . |           context: . | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								.github/workflows/dockle.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/dockle.yml
									
									
									
									
										vendored
									
									
								
							| @@ -14,7 +14,7 @@ jobs: | |||||||
|     env: |     env: | ||||||
|       DOCKER_CONTENT_TRUST: 1 |       DOCKER_CONTENT_TRUST: 1 | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v4.0.0 |       - uses: actions/checkout@v4.1.0 | ||||||
|       - run: | |       - run: | | ||||||
|           curl -L -o dockle.deb "https://github.com/goodwithtech/dockle/releases/download/v0.4.10/dockle_0.4.10_Linux-64bit.deb" |           curl -L -o dockle.deb "https://github.com/goodwithtech/dockle/releases/download/v0.4.10/dockle_0.4.10_Linux-64bit.deb" | ||||||
|           sudo dpkg -i dockle.deb |           sudo dpkg -i dockle.deb | ||||||
|   | |||||||
							
								
								
									
										6
									
								
								.github/workflows/lint.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										6
									
								
								.github/workflows/lint.yml
									
									
									
									
										vendored
									
									
								
							| @@ -11,7 +11,7 @@ jobs: | |||||||
|   pnpm_install: |   pnpm_install: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|     - uses: actions/checkout@v4.0.0 |     - uses: actions/checkout@v4.1.0 | ||||||
|       with: |       with: | ||||||
|         fetch-depth: 0 |         fetch-depth: 0 | ||||||
|         submodules: true |         submodules: true | ||||||
| @@ -38,7 +38,7 @@ jobs: | |||||||
|         - sw |         - sw | ||||||
|         - misskey-js |         - misskey-js | ||||||
|     steps: |     steps: | ||||||
|     - uses: actions/checkout@v4.0.0 |     - uses: actions/checkout@v4.1.0 | ||||||
|       with: |       with: | ||||||
|         fetch-depth: 0 |         fetch-depth: 0 | ||||||
|         submodules: true |         submodules: true | ||||||
| @@ -64,7 +64,7 @@ jobs: | |||||||
|         - backend |         - backend | ||||||
|         - misskey-js |         - misskey-js | ||||||
|     steps: |     steps: | ||||||
|     - uses: actions/checkout@v4.0.0 |     - uses: actions/checkout@v4.1.0 | ||||||
|       with: |       with: | ||||||
|         fetch-depth: 0 |         fetch-depth: 0 | ||||||
|         submodules: true |         submodules: true | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								.github/workflows/ok-to-test.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/ok-to-test.yml
									
									
									
									
										vendored
									
									
								
							| @@ -17,7 +17,7 @@ jobs: | |||||||
|     # See app.yml for an example app manifest |     # See app.yml for an example app manifest | ||||||
|     - name: Generate token |     - name: Generate token | ||||||
|       id: generate_token |       id: generate_token | ||||||
|       uses: tibdex/github-app-token@v1 |       uses: tibdex/github-app-token@v2 | ||||||
|       with: |       with: | ||||||
|         app_id: ${{ secrets.DEPLOYBOT_APP_ID }} |         app_id: ${{ secrets.DEPLOYBOT_APP_ID }} | ||||||
|         private_key: ${{ secrets.DEPLOYBOT_PRIVATE_KEY }} |         private_key: ${{ secrets.DEPLOYBOT_PRIVATE_KEY }} | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								.github/workflows/pr-preview-deploy.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/pr-preview-deploy.yml
									
									
									
									
										vendored
									
									
								
							| @@ -53,7 +53,7 @@ jobs: | |||||||
|  |  | ||||||
|     # Check out merge commit |     # Check out merge commit | ||||||
|     - name: Fork based /deploy checkout |     - name: Fork based /deploy checkout | ||||||
|       uses: actions/checkout@v4.0.0 |       uses: actions/checkout@v4.1.0 | ||||||
|       with: |       with: | ||||||
|         ref: 'refs/pull/${{ github.event.client_payload.pull_request.number }}/merge' |         ref: 'refs/pull/${{ github.event.client_payload.pull_request.number }}/merge' | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								.github/workflows/test-backend.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/test-backend.yml
									
									
									
									
										vendored
									
									
								
							| @@ -29,7 +29,7 @@ jobs: | |||||||
|           - 56312:6379 |           - 56312:6379 | ||||||
|  |  | ||||||
|     steps: |     steps: | ||||||
|     - uses: actions/checkout@v4.0.0 |     - uses: actions/checkout@v4.1.0 | ||||||
|       with: |       with: | ||||||
|         submodules: true |         submodules: true | ||||||
|     - name: Install pnpm |     - name: Install pnpm | ||||||
|   | |||||||
							
								
								
									
										4
									
								
								.github/workflows/test-frontend.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.github/workflows/test-frontend.yml
									
									
									
									
										vendored
									
									
								
							| @@ -16,7 +16,7 @@ jobs: | |||||||
|         node-version: [20.5.1] |         node-version: [20.5.1] | ||||||
|  |  | ||||||
|     steps: |     steps: | ||||||
|     - uses: actions/checkout@v4.0.0 |     - uses: actions/checkout@v4.1.0 | ||||||
|       with: |       with: | ||||||
|         submodules: true |         submodules: true | ||||||
|     - name: Install pnpm |     - name: Install pnpm | ||||||
| @@ -68,7 +68,7 @@ jobs: | |||||||
|           - 56312:6379 |           - 56312:6379 | ||||||
|  |  | ||||||
|     steps: |     steps: | ||||||
|     - uses: actions/checkout@v4.0.0 |     - uses: actions/checkout@v4.1.0 | ||||||
|       with: |       with: | ||||||
|         submodules: true |         submodules: true | ||||||
|     # https://github.com/cypress-io/cypress-docker-images/issues/150 |     # https://github.com/cypress-io/cypress-docker-images/issues/150 | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								.github/workflows/test-misskey-js.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/test-misskey-js.yml
									
									
									
									
										vendored
									
									
								
							| @@ -21,7 +21,7 @@ jobs: | |||||||
|  |  | ||||||
|     steps: |     steps: | ||||||
|       - name: Checkout |       - name: Checkout | ||||||
|         uses: actions/checkout@v4.0.0 |         uses: actions/checkout@v4.1.0 | ||||||
|  |  | ||||||
|       - run: corepack enable |       - run: corepack enable | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								.github/workflows/test-production.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/test-production.yml
									
									
									
									
										vendored
									
									
								
							| @@ -19,7 +19,7 @@ jobs: | |||||||
|         node-version: [20.5.1] |         node-version: [20.5.1] | ||||||
|  |  | ||||||
|     steps: |     steps: | ||||||
|     - uses: actions/checkout@v4.0.0 |     - uses: actions/checkout@v4.1.0 | ||||||
|       with: |       with: | ||||||
|         submodules: true |         submodules: true | ||||||
|     - name: Install pnpm |     - name: Install pnpm | ||||||
|   | |||||||
							
								
								
									
										153
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						
									
										153
									
								
								CHANGELOG.md
									
									
									
									
									
								
							| @@ -12,34 +12,134 @@ | |||||||
|  |  | ||||||
| --> | --> | ||||||
|  |  | ||||||
| ## 2023.9.0 (unreleased) | ## 2023.10.0 | ||||||
|  | ### NOTE | ||||||
|  | - muted_noteテーブルは使われなくなったため手動で削除を行ってください。 | ||||||
|  |  | ||||||
|  | ### Changes | ||||||
|  | - API: users/notes, notes/local-timeline で fileType 指定はできなくなりました | ||||||
|  | - API: notes/global-timeline は現在常に `[]` を返します | ||||||
|  |  | ||||||
| ### General | ### General | ||||||
| - OAuth 2.0のサポート | - ユーザーごとに他ユーザーへの返信をタイムラインに含めるか設定可能になりました | ||||||
| - お知らせ機能の強化 | - ユーザーリスト内のメンバーごとに他ユーザーへの返信をユーザーリストタイムラインに含めるか設定可能になりました | ||||||
|  | - ソフトワードミュートとハードワードミュートは統合されました | ||||||
|  |  | ||||||
|  | ### Client | ||||||
|  | - Fix: リアクションしたユーザ一覧のUIが稀に左上に残ってしまう不具合を修正 | ||||||
|  |  | ||||||
|  | ### Server | ||||||
|  | - タイムライン取得時のパフォーマンスを改善 | ||||||
|  |  | ||||||
|  | ## 2023.9.3 | ||||||
|  | ### General | ||||||
|  | - Enhance: ノートの翻訳機能の利用可否をロールで設定可能に | ||||||
|  |  | ||||||
|  | ### Client | ||||||
|  | - Enhance: AiScriptでホストのアドレスを参照する定数`SERVER_URL`を追加 | ||||||
|  | - Enhance: モデレーションログ機能の強化 | ||||||
|  | - Enhance: ローカリゼーションの更新 | ||||||
|  |  | ||||||
|  | ### Server | ||||||
|  | - Fix: Redisに古いバージョンのキャッシュが残っている場合、キャッシュが消えるまでの間通知が届かなくなる問題を修正 | ||||||
|  | - Fix: 後方互換性の修正 | ||||||
|  |  | ||||||
|  | ## 2023.9.2 | ||||||
|  |  | ||||||
|  | ### General | ||||||
|  | - Feat: ノートの編集をできるように | ||||||
|  | 	- ロールで編集可否を設定可能 | ||||||
|  | - Feat: 通知を種類ごとに 全員から受け取る/フォロー中のユーザーのみ受け取る/フォロワーのみ受け取る/相互のみ受け取る/指定したリストのメンバーのみ受け取る/受け取らない から選べるように | ||||||
|  | - Enhance: タイムラインからRenoteを除外するオプションを追加 | ||||||
|  | - Enhance: ユーザーページのノート一覧でRenoteを除外できるように | ||||||
|  | - Enhance: タイムラインでファイルが添付されたノートのみ表示するオプションを追加 | ||||||
|  | - Enhance: モデレーションログ機能の強化 | ||||||
|  | - Enhance: 依存関係の更新 | ||||||
|  | - Enhance: ローカリゼーションの更新 | ||||||
|  |  | ||||||
|  | ### Client | ||||||
|  | - Enhance: Plugin:register_post_form_actionを用いてCWを取得・変更できるように | ||||||
|  | - Enhance: admin/ad/listにて掲載中の広告が絞り込めるように | ||||||
|  | - Enhance: AiScriptにリモートサーバーのAPIを叩く用の関数を追加(`Mk:apiExternal`) | ||||||
|  |  | ||||||
|  | ### Server | ||||||
|  | - Enhance: MasterプロセスのPIDを書き出せるように | ||||||
|  | - Enhance: admin/ad/createにてレスポンス200、設定した広告情報を返すように | ||||||
|  |  | ||||||
|  | ## 2023.9.1 | ||||||
|  |  | ||||||
|  | ### General | ||||||
|  | - Enhance: モデレーションログ機能の強化 | ||||||
|  |  | ||||||
|  | ### Client | ||||||
|  | - Fix: ノートのメニューにある「詳細」ボタンの表示がログイン/ログアウト状態で統一されていない問題を修正 | ||||||
|  |  | ||||||
|  | ### Server | ||||||
|  | - Fix: お知らせのページネーションが機能しない | ||||||
|  | - Fix: 「ユーザーの新規投稿」の通知設定を切り替えるとサーバー内部エラーが出る | ||||||
|  |  | ||||||
|  | ## 2023.9.0 | ||||||
|  |  | ||||||
|  | ### Note | ||||||
|  | - meilisearchを使用する場合、v1.2以上が必要です | ||||||
|  |  | ||||||
|  | ### General | ||||||
|  | - Feat: OAuth 2.0のサポート | ||||||
|  | - Feat: お知らせ機能の強化 | ||||||
| 	- ユーザー個別のお知らせを作成可能に | 	- ユーザー個別のお知らせを作成可能に | ||||||
| 	- お知らせのバナー表示やダイアログ表示が可能に | 	- お知らせのバナー表示やダイアログ表示が可能に | ||||||
| 	- お知らせのアイコンを設定可能に | 	- お知らせのアイコンを設定可能に | ||||||
| - チャンネルをセンシティブ指定できるようになりました | - Feat: チャンネルをセンシティブ指定できるようになりました | ||||||
| 	- センシティブチャンネルのNoteのReNoteはデフォルトでHome TLに流れるようになりました | 	- センシティブチャンネルのNoteのReNoteはデフォルトでHome TLに流れるようになりました | ||||||
| - 二要素認証のバックアップコードが生成されるようになりました ref. https://github.com/MisskeyIO/misskey/pull/121 | 	- センシティブチャンネルのノートはユーザープロフィールに表示されません | ||||||
|  | - Feat: 二要素認証のバックアップコードが生成されるようになりました | ||||||
|  | 	- ref. https://github.com/MisskeyIO/misskey/pull/121 | ||||||
|  | - Feat: 二要素認証でパスキーをサポートするようになりました | ||||||
|  | - Feat: 指定したユーザーが投稿したときに通知できるようになりました | ||||||
|  | - Feat: プロフィールでのリンク検証 | ||||||
|  | - Feat: モデレーションログ機能 | ||||||
|  | - Feat: 通知をテストできるようになりました | ||||||
|  | - Feat: PWAのアイコンが設定できるようになりました | ||||||
|  | - Enhance: サーバー名の略称が設定できるようになりました | ||||||
|  | - Enhance: アンテナの受信ソースに指定したユーザを除外するものを追加 | ||||||
|  | - Enhance: 二要素認証設定時のセキュリティを強化 | ||||||
|  | 	- パスワード入力が必要な操作を行う際、二要素認証が有効であれば確認コードの入力も必要になりました | ||||||
|  | - Enhance: manifest.jsonをオーバーライド可能に | ||||||
|  | - Enhance: 依存関係の更新 | ||||||
|  | - Enhance: ローカリゼーションの更新 | ||||||
|  |  | ||||||
| ### Client | ### Client | ||||||
| - プロフィールにその人が作ったPlayの一覧出せるように | - Feat: 任意のユーザーリストをタイムラインページにピン留めできるように | ||||||
| - メニューのスイッチの動作を改善 | 	- 設定->クライアント設定->全般 から設定可能です | ||||||
| - 絵文字ピッカーの検索の表示件数を100件に増加 | - Feat: Playで直接投稿フォームを埋め込めるように(`Ui:C:postForm`) | ||||||
| - 投稿フォームのプレビューの表示状態を記憶するように | - Feat: クライアントを起動している間、デバイスの画面が自動でオフになるのを防ぐオプションを追加 | ||||||
| - ノート詳細ページ読み込み時のパフォーマンスを改善 | - Feat: 新しい実績を追加 | ||||||
| - AiScriptからMisskeyサーバーAPIを呼び出す際の制限を撤廃 | - Enhance: ノート詳細ページでリノート一覧、リアクション一覧タブを追加 | ||||||
|  | 	- ノートのメニューからは当該項目は消えました | ||||||
|  | - Enhance: センシティブなメディアを目立たせる設定を追加 | ||||||
|  | - Enhance: プロフィールにその人が作ったPlayの一覧出せるように | ||||||
|  | - Enhance: メニューのスイッチの動作を改善 | ||||||
|  | - Enhance: 絵文字ピッカーの検索の表示件数を100件に増加 | ||||||
|  | - Enhance: 投稿フォームのプレビューの表示状態を記憶するように | ||||||
| - Enhance: ユーザーメニューでスイッチでユーザーリストに追加・削除できるように | - Enhance: ユーザーメニューでスイッチでユーザーリストに追加・削除できるように | ||||||
| - Enhance: 自分が押したリアクションのデザインを改善 | - Enhance: 自分が押したリアクションのデザインを改善 | ||||||
| - Enhance: ノート検索にローカルのみ検索可能なオプションの追加 | - Enhance: ノート検索にローカルのみ検索可能なオプションの追加 | ||||||
| - Enhance: AiScriptで`LOCALE`として現在の設定言語を取得できるように |  | ||||||
| - Enhance: Renote自体を通報できるように | - Enhance: Renote自体を通報できるように | ||||||
| - Enhance: データセーバーモードの強化 | - Enhance: データセーバーモードの強化 | ||||||
| - Enhance: Renoteを管理者権限で削除可能に | - Enhance: Renoteを管理者権限で削除可能に | ||||||
| - `$[rainbow ]`記法が、動きのあるMFMが無効になっていても使用できるようになりました | - Enhance: `$[rainbow ]`記法が、動きのあるMFMが無効になっていても使用できるようになりました | ||||||
| - Playの操作を行うAPI TokenをAPIコンソールから発行できるように | - Enhance: Playの操作を行うAPI TokenをAPIコンソールから発行できるように | ||||||
|  | - Enhance: リアクションの表示サイズをより大きくできるように | ||||||
|  | - Enhance: AiScriptを0.16.0に更新 | ||||||
|  | - Enhance: AiScriptからMisskeyサーバーAPIを呼び出す際の制限を撤廃 | ||||||
|  | - Enhance: AiScriptで`LOCALE`として現在の設定言語を取得できるように | ||||||
|  | - Enhance: Mk:apiが失敗した時にエラー型の値(AiScript 0.16.0で追加)を返すように | ||||||
|  | - Enhance: ScratchpadでAsync:系関数やボタンのコールバックなどのエラーにもダイアログを出すように(試験的なためPlayなどには未実装) | ||||||
|  | - Enhance: ノート詳細ページ読み込み時のパフォーマンスが向上しました | ||||||
|  | - Enhance: タイムラインでリスト/アンテナ選択時のパフォーマンスを改善 | ||||||
|  | - Enhance: 「Moderation note」、「Add moderation note」をローカライズできるように | ||||||
|  | - Enhance: プラグインのソースコードを確認・コピーできるように | ||||||
|  | - Enhance: 細かなデザインの調整 | ||||||
| - Fix: サーバー情報画面(`/instance-info/{domain}`)でブロックができないのを修正 | - Fix: サーバー情報画面(`/instance-info/{domain}`)でブロックができないのを修正 | ||||||
| - Fix: 未読のお知らせの「わかった」をクリック・タップしてもその場で「わかった」が消えない問題を修正 | - Fix: 未読のお知らせの「わかった」をクリック・タップしてもその場で「わかった」が消えない問題を修正 | ||||||
| - Fix: iOSで画面を回転させるとテキストサイズが変わる問題を修正 | - Fix: iOSで画面を回転させるとテキストサイズが変わる問題を修正 | ||||||
| @@ -47,18 +147,32 @@ | |||||||
| - Fix: タイムラインを下にスクロールしてノート画面に移動して再び戻ったら以前のスクロール位置を失う問題を修正 | - Fix: タイムラインを下にスクロールしてノート画面に移動して再び戻ったら以前のスクロール位置を失う問題を修正 | ||||||
| - Fix: Misskeyプラグインをインストールする際のAiScriptバージョンのチェックが0.14.0以降に対応していない問題を修正 | - Fix: Misskeyプラグインをインストールする際のAiScriptバージョンのチェックが0.14.0以降に対応していない問題を修正 | ||||||
| - Fix: 他のサーバーのユーザーへ「メッセージを送信」した時の初期テキストのメンションが間違っている問題を修正 | - Fix: 他のサーバーのユーザーへ「メッセージを送信」した時の初期テキストのメンションが間違っている問題を修正 | ||||||
|  | - Fix: 環境によってはMisskey Webが開けない問題を修正 | ||||||
|  | - Fix: プラグインの権限リストが見れない問題を修正 | ||||||
|  | - Fix: 複数の階層があるメニューで、短くタップすると正常に動かない場合がある問題を修正 | ||||||
|  | - Fix: アニメーションがオフのとき、スマホで子メニューの選択ができない問題を修正 | ||||||
|  | - Fix: ドロワーメニューで、親メニュー項目をマウスでホバーすると子メニューが表示されてしまう問題を修正 | ||||||
|  | - Fix: AiScriptでMk:apiが外部と通信できる問題を修正 | ||||||
|  |  | ||||||
| ### Server | ### Server | ||||||
|  | - Change: cacheRemoteFilesの初期値はfalseになりました | ||||||
|  | - Enhance: ファイルアップロード時等にファイル名の拡張子を修正する関数(correctFilename)の挙動を改善 | ||||||
|  | - Enhance: Webhookのペイロードにサーバーのurlが含まれるようになりました | ||||||
|  | - Enhance: Webhook設定でsecretを空に出来るように | ||||||
|  | - Enhance: 使われていないアンテナの自動停止を設定可能に | ||||||
|  | - Enhance: nodeinfo 2.1対応 | ||||||
|  | - Enhance: 自分へのメンション一覧を取得する際のパフォーマンスを向上 | ||||||
|  | - Enhance: Docker環境でjemallocを使用することでメモリ使用量を削減 | ||||||
|  | - Enhance: ID生成方式としてaidxを追加、かつデフォルトに | ||||||
|  | - Enhance: Add address bind config option (outgoingAddress) | ||||||
|  | - Fix: MK_ONLY_SERVERオプションを指定した際にクラッシュする問題を修正 | ||||||
|  | - Fix: notes/reactionsのページネーションが機能しない問題を修正 | ||||||
| - Fix: ノート検索 `notes/search` にてhostを指定した際に検索結果に反映されるように | - Fix: ノート検索 `notes/search` にてhostを指定した際に検索結果に反映されるように | ||||||
| - cacheRemoteFilesの初期値はfalseになりました |  | ||||||
| - ファイルアップロード時等にファイル名の拡張子を修正する関数(correctFilename)の挙動を改善 |  | ||||||
| - Webhookのペイロードにサーバーのurlが含まれるようになりました |  | ||||||
| - Webhook設定でsecretを空に出来るように |  | ||||||
| - 使われていないアンテナの自動停止を設定可能に |  | ||||||
| - Fix: 一部のfeatured noteを照会できない問題を修正 | - Fix: 一部のfeatured noteを照会できない問題を修正 | ||||||
| - Fix: muteがapiからのuser list timeline取得で機能しない問題を修正 | - Fix: muteがapiからのuser list timeline取得で機能しない問題を修正 | ||||||
| - Fix: ジョブキュー管理画面の認証を回避できる問題を修正 | - Fix: ジョブキュー管理画面の認証を回避できる問題を修正 | ||||||
| - Fix: 一部のサーバー内部エラーがスタックトレースを返さないように修正 | - Fix: 一部のサーバー内部エラーがスタックトレースを返さないように修正 | ||||||
|  | - Fix: 一部のリモートユーザーをフォローすることができない問題を修正 | ||||||
|  |  | ||||||
| ## 13.14.2 | ## 13.14.2 | ||||||
|  |  | ||||||
| @@ -73,7 +187,6 @@ | |||||||
| ### Server | ### Server | ||||||
| - Fix: APIのオフセットが壊れていたせいで「もっと見る」でもっと見れない問題を修正 | - Fix: APIのオフセットが壊れていたせいで「もっと見る」でもっと見れない問題を修正 | ||||||
| - Fix: 外部サーバーの投稿がタイムラインに表示されないことがある問題を修正 | - Fix: 外部サーバーの投稿がタイムラインに表示されないことがある問題を修正 | ||||||
| - Enhance: Add address bind config option (outgoingAddress) |  | ||||||
|  |  | ||||||
| ## 13.14.1 | ## 13.14.1 | ||||||
|  |  | ||||||
|   | |||||||
| @@ -436,3 +436,6 @@ marginはそのコンポーネントを使う側が設定する | |||||||
| ## その他 | ## その他 | ||||||
| ### HTMLのクラス名で follow という単語は使わない | ### HTMLのクラス名で follow という単語は使わない | ||||||
| 広告ブロッカーで誤ってブロックされる | 広告ブロッカーで誤ってブロックされる | ||||||
|  |  | ||||||
|  | ### indexというファイル名を使うな | ||||||
|  | ESMではディレクトリインポートは廃止されているのと、ディレクトリインポートせずともファイル名が index だと何故か一部のライブラリ?でディレクトリインポートだと見做されてエラーになる | ||||||
|   | |||||||
| @@ -62,7 +62,8 @@ ARG GID="991" | |||||||
|  |  | ||||||
| RUN apt-get update \ | RUN apt-get update \ | ||||||
| 	&& apt-get install -y --no-install-recommends \ | 	&& apt-get install -y --no-install-recommends \ | ||||||
| 	ffmpeg tini curl \ | 	ffmpeg tini curl libjemalloc-dev libjemalloc2 \ | ||||||
|  | 	&& ln -s /usr/lib/$(uname -m)-linux-gnu/libjemalloc.so.2 /usr/local/lib/libjemalloc.so \ | ||||||
| 	&& corepack enable \ | 	&& corepack enable \ | ||||||
| 	&& groupadd -g "${GID}" misskey \ | 	&& groupadd -g "${GID}" misskey \ | ||||||
| 	&& useradd -l -u "${UID}" -g "${GID}" -m -d /misskey misskey \ | 	&& useradd -l -u "${UID}" -g "${GID}" -m -d /misskey misskey \ | ||||||
| @@ -81,6 +82,7 @@ COPY --chown=misskey:misskey --from=native-builder /misskey/packages/backend/bui | |||||||
| COPY --chown=misskey:misskey --from=native-builder /misskey/fluent-emojis /misskey/fluent-emojis | COPY --chown=misskey:misskey --from=native-builder /misskey/fluent-emojis /misskey/fluent-emojis | ||||||
| COPY --chown=misskey:misskey . ./ | COPY --chown=misskey:misskey . ./ | ||||||
|  |  | ||||||
|  | ENV LD_PRELOAD=/usr/local/lib/libjemalloc.so | ||||||
| ENV NODE_ENV=production | ENV NODE_ENV=production | ||||||
| HEALTHCHECK --interval=5s --retries=20 CMD ["/bin/bash", "/misskey/healthcheck.sh"] | HEALTHCHECK --interval=5s --retries=20 CMD ["/bin/bash", "/misskey/healthcheck.sh"] | ||||||
| ENTRYPOINT ["/usr/bin/tini", "--"] | ENTRYPOINT ["/usr/bin/tini", "--"] | ||||||
|   | |||||||
| @@ -116,6 +116,14 @@ redis: | |||||||
| #  #prefix: example-prefix | #  #prefix: example-prefix | ||||||
| #  #db: 1 | #  #db: 1 | ||||||
|  |  | ||||||
|  | #redisForTimelines: | ||||||
|  | #  host: redis | ||||||
|  | #  port: 6379 | ||||||
|  | #  #family: 0  # 0=Both, 4=IPv4, 6=IPv6 | ||||||
|  | #  #pass: example-pass | ||||||
|  | #  #prefix: example-prefix | ||||||
|  | #  #db: 1 | ||||||
|  |  | ||||||
| #   ┌───────────────────────────┐ | #   ┌───────────────────────────┐ | ||||||
| #───┘ MeiliSearch configuration └───────────────────────────── | #───┘ MeiliSearch configuration └───────────────────────────── | ||||||
|  |  | ||||||
|   | |||||||
| @@ -50,7 +50,7 @@ services: | |||||||
|  |  | ||||||
| #  meilisearch: | #  meilisearch: | ||||||
| #    restart: always | #    restart: always | ||||||
| #    image: getmeili/meilisearch:v1.1.1 | #    image: getmeili/meilisearch:v1.3.4 | ||||||
| #    environment: | #    environment: | ||||||
| #      - MEILI_NO_ANALYTICS=true | #      - MEILI_NO_ANALYTICS=true | ||||||
| #      - MEILI_ENV=production | #      - MEILI_ENV=production | ||||||
|   | |||||||
| @@ -348,7 +348,6 @@ invite: "دعوة" | |||||||
| driveCapacityPerLocalAccount: "حصة التخزين لكل مستخدم محلي" | driveCapacityPerLocalAccount: "حصة التخزين لكل مستخدم محلي" | ||||||
| driveCapacityPerRemoteAccount: "حصة التخزين لكل مستخدم بعيد" | driveCapacityPerRemoteAccount: "حصة التخزين لكل مستخدم بعيد" | ||||||
| inMb: "بالميغابايت" | inMb: "بالميغابايت" | ||||||
| iconUrl: "رابط الأيقونة" |  | ||||||
| bannerUrl: "رابط صورة اللافتة" | bannerUrl: "رابط صورة اللافتة" | ||||||
| backgroundImageUrl: "رابط صورة الخلفية" | backgroundImageUrl: "رابط صورة الخلفية" | ||||||
| basicInfo: "المعلومات الأساسية " | basicInfo: "المعلومات الأساسية " | ||||||
| @@ -998,6 +997,8 @@ expirationDate: "تاريخ انتهاء الصلاحية" | |||||||
| unused: "غير مستعمَل" | unused: "غير مستعمَل" | ||||||
| expired: "منتهية صلاحيته" | expired: "منتهية صلاحيته" | ||||||
| icon: "الصورة الرمزية" | icon: "الصورة الرمزية" | ||||||
|  | replies: "رد" | ||||||
|  | renotes: "أعد النشر" | ||||||
| _initialAccountSetting: | _initialAccountSetting: | ||||||
|   accountCreated: "نجح إنشاء حسابك!" |   accountCreated: "نجح إنشاء حسابك!" | ||||||
|   letsStartAccountSetup: "إذا كنت جديدًا لنعدّ حسابك الشخصي." |   letsStartAccountSetup: "إذا كنت جديدًا لنعدّ حسابك الشخصي." | ||||||
| @@ -1139,6 +1140,7 @@ _plugin: | |||||||
|   install: "ثبّت إضافات" |   install: "ثبّت إضافات" | ||||||
|   installWarn: "رجاءً لا تثبت إضافات غير موثوقة." |   installWarn: "رجاءً لا تثبت إضافات غير موثوقة." | ||||||
|   manage: "إدارة الإضافات" |   manage: "إدارة الإضافات" | ||||||
|  |   viewSource: "اظهر المصدر" | ||||||
| _preferencesBackups: | _preferencesBackups: | ||||||
|   createdAt: "تم إنشاؤه: {date} {time}" |   createdAt: "تم إنشاؤه: {date} {time}" | ||||||
|   updatedAt: "آخر تحديث: {date} {time}" |   updatedAt: "آخر تحديث: {date} {time}" | ||||||
| @@ -1554,3 +1556,6 @@ _webhookSettings: | |||||||
|   active: "مُفعّل" |   active: "مُفعّل" | ||||||
|   _events: |   _events: | ||||||
|     reaction: "عند التفاعل" |     reaction: "عند التفاعل" | ||||||
|  | _moderationLogTypes: | ||||||
|  |   suspend: "علِق" | ||||||
|  |   resetPassword: "أعد تعيين كلمتك السرية" | ||||||
|   | |||||||
| @@ -328,7 +328,6 @@ invite: "আমন্ত্রণ" | |||||||
| driveCapacityPerLocalAccount: "প্রত্যেক স্থানীয় ব্যাবহারকারীর জন্য ড্রাইভের জায়গা" | driveCapacityPerLocalAccount: "প্রত্যেক স্থানীয় ব্যাবহারকারীর জন্য ড্রাইভের জায়গা" | ||||||
| driveCapacityPerRemoteAccount: "প্রত্যেক রিমোট ব্যাবহারকারীর জন্য ড্রাইভের জায়গা" | driveCapacityPerRemoteAccount: "প্রত্যেক রিমোট ব্যাবহারকারীর জন্য ড্রাইভের জায়গা" | ||||||
| inMb: "মেগাবাইটে লিখুন" | inMb: "মেগাবাইটে লিখুন" | ||||||
| iconUrl: "আইকনের URL (ফ্যাভিকন, ইত্যাদি)" |  | ||||||
| bannerUrl: "ব্যানার ছবির URL" | bannerUrl: "ব্যানার ছবির URL" | ||||||
| backgroundImageUrl: "পটভূমির চিত্রের URL" | backgroundImageUrl: "পটভূমির চিত্রের URL" | ||||||
| basicInfo: "আপনার ব্যক্তিগত তথ্য" | basicInfo: "আপনার ব্যক্তিগত তথ্য" | ||||||
| @@ -839,6 +838,8 @@ color: "রং" | |||||||
| horizontal: "পাশে" | horizontal: "পাশে" | ||||||
| youFollowing: "অনুসরণ করা হচ্ছে" | youFollowing: "অনুসরণ করা হচ্ছে" | ||||||
| icon: "প্রোফাইল ছবি" | icon: "প্রোফাইল ছবি" | ||||||
|  | replies: "জবাব" | ||||||
|  | renotes: "রিনোট" | ||||||
| _role: | _role: | ||||||
|   priority: "অগ্রাধিকার" |   priority: "অগ্রাধিকার" | ||||||
|   _priority: |   _priority: | ||||||
| @@ -888,6 +889,7 @@ _plugin: | |||||||
|   install: "প্লাগইন ইন্সটল করুন" |   install: "প্লাগইন ইন্সটল করুন" | ||||||
|   installWarn: "অবিশ্বস্ত প্লাগইন ইনস্টল করবেন না।" |   installWarn: "অবিশ্বস্ত প্লাগইন ইনস্টল করবেন না।" | ||||||
|   manage: "প্লাগইন ম্যানেজ করুন" |   manage: "প্লাগইন ম্যানেজ করুন" | ||||||
|  |   viewSource: "উৎস দেখুন" | ||||||
| _registry: | _registry: | ||||||
|   scope: "স্কোপ" |   scope: "স্কোপ" | ||||||
|   key: "কী" |   key: "কী" | ||||||
| @@ -1331,3 +1333,6 @@ _deck: | |||||||
| _webhookSettings: | _webhookSettings: | ||||||
|   name: "নাম" |   name: "নাম" | ||||||
|   active: "চালু" |   active: "চালু" | ||||||
|  | _moderationLogTypes: | ||||||
|  |   suspend: "স্থগিত করা" | ||||||
|  |   resetPassword: "পাসওয়ার্ড রিসেট করুন" | ||||||
|   | |||||||
| @@ -381,6 +381,8 @@ user: "Usuaris" | |||||||
| global: "Global" | global: "Global" | ||||||
| searchByGoogle: "Cercar" | searchByGoogle: "Cercar" | ||||||
| file: "Fitxers" | file: "Fitxers" | ||||||
|  | replies: "Respondre" | ||||||
|  | renotes: "Impulsa" | ||||||
| _role: | _role: | ||||||
|   _options: |   _options: | ||||||
|     antennaMax: "Nombre màxim d'antenes" |     antennaMax: "Nombre màxim d'antenes" | ||||||
| @@ -477,3 +479,6 @@ _deck: | |||||||
|     list: "Llistes" |     list: "Llistes" | ||||||
|     mentions: "Mencions" |     mentions: "Mencions" | ||||||
|     direct: "Publicacions directes" |     direct: "Publicacions directes" | ||||||
|  | _moderationLogTypes: | ||||||
|  |   suspend: "Suspèn" | ||||||
|  |   resetPassword: "Restableix la contrasenya" | ||||||
|   | |||||||
| @@ -354,7 +354,6 @@ invite: "Pozvat" | |||||||
| driveCapacityPerLocalAccount: "Kapacita disku na lokálního uživatele" | driveCapacityPerLocalAccount: "Kapacita disku na lokálního uživatele" | ||||||
| driveCapacityPerRemoteAccount: "Kapacita disku na vzdáleného uživatele" | driveCapacityPerRemoteAccount: "Kapacita disku na vzdáleného uživatele" | ||||||
| inMb: "V megabajtech" | inMb: "V megabajtech" | ||||||
| iconUrl: "Favicon URL" |  | ||||||
| bannerUrl: "Baner URL" | bannerUrl: "Baner URL" | ||||||
| backgroundImageUrl: "Adresa URL obrázku pozadí" | backgroundImageUrl: "Adresa URL obrázku pozadí" | ||||||
| basicInfo: "Základní informace" | basicInfo: "Základní informace" | ||||||
| @@ -1019,7 +1018,6 @@ retryAllQueuesConfirmText: "Tohle dočasně zvýší zatěž na server." | |||||||
| enableChartsForRemoteUser: "Vygenerovat grafy dat vzdálených uživatelů" | enableChartsForRemoteUser: "Vygenerovat grafy dat vzdálených uživatelů" | ||||||
| enableChartsForFederatedInstances: "Vygenerovat grafy dat vzdálených instancí" | enableChartsForFederatedInstances: "Vygenerovat grafy dat vzdálených instancí" | ||||||
| showClipButtonInNoteFooter: "Přidat \"Připnout\" do akčního menu poznámky" | showClipButtonInNoteFooter: "Přidat \"Připnout\" do akčního menu poznámky" | ||||||
| largeNoteReactions: "Zvětšit zobrazované reakce" |  | ||||||
| noteIdOrUrl: "ID nebo URL poznámky" | noteIdOrUrl: "ID nebo URL poznámky" | ||||||
| video: "Video" | video: "Video" | ||||||
| videos: "Videa" | videos: "Videa" | ||||||
| @@ -1096,6 +1094,8 @@ doYouAgree: "Souhlasíte?" | |||||||
| beSureToReadThisAsItIsImportant: "Přečtěte si prosím tyto důležité informace." | beSureToReadThisAsItIsImportant: "Přečtěte si prosím tyto důležité informace." | ||||||
| iHaveReadXCarefullyAndAgree: "Přečetl jsem si text \"{x}\" a souhlasím s ním." | iHaveReadXCarefullyAndAgree: "Přečetl jsem si text \"{x}\" a souhlasím s ním." | ||||||
| icon: "Avatar" | icon: "Avatar" | ||||||
|  | replies: "Odpovědět" | ||||||
|  | renotes: "Přeposlat" | ||||||
| _initialAccountSetting: | _initialAccountSetting: | ||||||
|   accountCreated: "Váš účet byl úspěšně vytvořen!" |   accountCreated: "Váš účet byl úspěšně vytvořen!" | ||||||
|   letsStartAccountSetup: "Pro začátek si nastavte svůj profil." |   letsStartAccountSetup: "Pro začátek si nastavte svůj profil." | ||||||
| @@ -1113,6 +1113,8 @@ _initialAccountSetting: | |||||||
|   laterAreYouSure: "Opravdu chcete provést nastavení profilu později?" |   laterAreYouSure: "Opravdu chcete provést nastavení profilu později?" | ||||||
| _serverRules: | _serverRules: | ||||||
|   description: "Soubor pravidel, která se zobrazí před registrací. Doporučuje se nastavit shrnutí podmínek služby." |   description: "Soubor pravidel, která se zobrazí před registrací. Doporučuje se nastavit shrnutí podmínek služby." | ||||||
|  | _serverSettings: | ||||||
|  |   iconUrl: "URL ikony" | ||||||
| _accountMigration: | _accountMigration: | ||||||
|   moveFrom: "Migrace jiného účtu na tento účet" |   moveFrom: "Migrace jiného účtu na tento účet" | ||||||
|   moveFromSub: "Vytvořit alias na jiný účet" |   moveFromSub: "Vytvořit alias na jiný účet" | ||||||
| @@ -1490,6 +1492,7 @@ _plugin: | |||||||
|   install: "Instalovat plugin" |   install: "Instalovat plugin" | ||||||
|   installWarn: "Neinstalujte nedůvěryhodné pluginy." |   installWarn: "Neinstalujte nedůvěryhodné pluginy." | ||||||
|   manage: "Správce pluginů" |   manage: "Správce pluginů" | ||||||
|  |   viewSource: "Zobrazit zdroj" | ||||||
| _preferencesBackups: | _preferencesBackups: | ||||||
|   list: "Vytvořit backup" |   list: "Vytvořit backup" | ||||||
|   saveNew: "Uložit novou zálohu" |   saveNew: "Uložit novou zálohu" | ||||||
| @@ -1677,7 +1680,6 @@ _timelineTutorial: | |||||||
| _2fa: | _2fa: | ||||||
|   alreadyRegistered: "Již jste zaregistrovali dvoufaktorové ověřovací zařízení." |   alreadyRegistered: "Již jste zaregistrovali dvoufaktorové ověřovací zařízení." | ||||||
|   registerTOTP: "Registrovat aplikaci autentizátoru" |   registerTOTP: "Registrovat aplikaci autentizátoru" | ||||||
|   passwordToTOTP: "Zadejte své heslo" |  | ||||||
|   step1: "Nejprve si do zařízení nainstalujte aplikaci pro ověřování (například {a} nebo {b})." |   step1: "Nejprve si do zařízení nainstalujte aplikaci pro ověřování (například {a} nebo {b})." | ||||||
|   step2: "Poté naskenujte QR kód zobrazený na této obrazovce." |   step2: "Poté naskenujte QR kód zobrazený na této obrazovce." | ||||||
|   step2Click: "Kliknutím na tento QR kód můžete zaregistrovat 2FA do bezpečnostního klíče nebo aplikace autentizace telefonu." |   step2Click: "Kliknutím na tento QR kód můžete zaregistrovat 2FA do bezpečnostního klíče nebo aplikace autentizace telefonu." | ||||||
| @@ -1687,7 +1689,6 @@ _2fa: | |||||||
|   securityKeyNotSupported: "Váš prohlížeč nepodporuje bezpečnostní klíče." |   securityKeyNotSupported: "Váš prohlížeč nepodporuje bezpečnostní klíče." | ||||||
|   registerTOTPBeforeKey: "Nastavte aplikaci autentizátoru pro registraci bezpečnostního nebo přístupového klíče." |   registerTOTPBeforeKey: "Nastavte aplikaci autentizátoru pro registraci bezpečnostního nebo přístupového klíče." | ||||||
|   securityKeyInfo: "Kromě ověřování otiskem prstu nebo PIN můžete nastavit také ověřování pomocí hardwarových bezpečnostních klíčů, které podporují FIDO2, a svůj účet tak dále zabezpečit." |   securityKeyInfo: "Kromě ověřování otiskem prstu nebo PIN můžete nastavit také ověřování pomocí hardwarových bezpečnostních klíčů, které podporují FIDO2, a svůj účet tak dále zabezpečit." | ||||||
|   chromePasskeyNotSupported: "Chrome passkeys nejsou v současné době podporovány." |  | ||||||
|   registerSecurityKey: "Registrace bezpečnostního nebo přístupového klíče" |   registerSecurityKey: "Registrace bezpečnostního nebo přístupového klíče" | ||||||
|   securityKeyName: "Zadejte název klíče" |   securityKeyName: "Zadejte název klíče" | ||||||
|   tapSecurityKey: "Při registraci bezpečnostního nebo přístupového klíče postupujte podle svého prohlížeče." |   tapSecurityKey: "Při registraci bezpečnostního nebo přístupového klíče postupujte podle svého prohlížeče." | ||||||
| @@ -2035,3 +2036,7 @@ _webhookSettings: | |||||||
|     renote: "Při renotaci poznámky" |     renote: "Při renotaci poznámky" | ||||||
|     reaction: "Při obdržení reakce" |     reaction: "Při obdržení reakce" | ||||||
|     mention: "Při zmínce" |     mention: "Při zmínce" | ||||||
|  | _moderationLogTypes: | ||||||
|  |   suspend: "Zmrazit" | ||||||
|  |   resetPassword: "Resetovat heslo" | ||||||
|  |   createInvitation: "Vygenerovat pozvánku" | ||||||
|   | |||||||
| @@ -2,7 +2,7 @@ | |||||||
| _lang_: "Deutsch" | _lang_: "Deutsch" | ||||||
| headlineMisskey: "Ein durch Notizen verbundenes Netzwerk" | headlineMisskey: "Ein durch Notizen verbundenes Netzwerk" | ||||||
| introMisskey: "Willkommen! Misskey ist eine dezentralisierte Open-Source Microblogging-Platform.\nVerfasse „Notizen“ um mitzuteilen, was gerade passiert oder um Ereignisse mit anderen zu teilen. 📡\nMit „Reaktionen“ kannst du außerdem schnell deine Gefühle über Notizen anderer Benutzer zum Ausdruck bringen. 👍\nEine neue Welt wartet auf dich! 🚀" | introMisskey: "Willkommen! Misskey ist eine dezentralisierte Open-Source Microblogging-Platform.\nVerfasse „Notizen“ um mitzuteilen, was gerade passiert oder um Ereignisse mit anderen zu teilen. 📡\nMit „Reaktionen“ kannst du außerdem schnell deine Gefühle über Notizen anderer Benutzer zum Ausdruck bringen. 👍\nEine neue Welt wartet auf dich! 🚀" | ||||||
| poweredByMisskeyDescription: "{name} ist einer der durch die Open-Source-Plattform <b>Misskey</b> betriebenen Dienste (meist als \"Misskey-Instanz\" bezeichnet)." | poweredByMisskeyDescription: "{name} ist einer der durch die Open-Source-Plattform <b>Misskey</b> betriebenen Dienste." | ||||||
| monthAndDay: "{day}.{month}." | monthAndDay: "{day}.{month}." | ||||||
| search: "Suchen" | search: "Suchen" | ||||||
| notifications: "Benachrichtigungen" | notifications: "Benachrichtigungen" | ||||||
| @@ -75,7 +75,7 @@ import: "Import" | |||||||
| export: "Export" | export: "Export" | ||||||
| files: "Dateien" | files: "Dateien" | ||||||
| download: "Herunterladen" | download: "Herunterladen" | ||||||
| driveFileDeleteConfirm: "Möchtest du die Datei „{name}“ wirklich löschen? Sie wird in allen Inhalten, die sie verwenden, auch verschwinden." | driveFileDeleteConfirm: "Möchtest du die Datei „{name}“ wirklich löschen? Einige Inhalte, die diese Datei verwenden, werden auch verschwinden." | ||||||
| unfollowConfirm: "Möchtest du {name} wirklich nicht mehr folgen?" | unfollowConfirm: "Möchtest du {name} wirklich nicht mehr folgen?" | ||||||
| exportRequested: "Du hast einen Export angefragt. Dies kann etwas Zeit in Anspruch nehmen. Sobald der Export abgeschlossen ist, wird er deiner Drive hinzugefügt." | exportRequested: "Du hast einen Export angefragt. Dies kann etwas Zeit in Anspruch nehmen. Sobald der Export abgeschlossen ist, wird er deiner Drive hinzugefügt." | ||||||
| importRequested: "Du hast einen Import angefragt. Dies kann etwas Zeit in Anspruch nehmen." | importRequested: "Du hast einen Import angefragt. Dies kann etwas Zeit in Anspruch nehmen." | ||||||
| @@ -321,7 +321,7 @@ copyUrl: "URL kopieren" | |||||||
| rename: "Umbenennen" | rename: "Umbenennen" | ||||||
| avatar: "Profilbild" | avatar: "Profilbild" | ||||||
| banner: "Banner" | banner: "Banner" | ||||||
| displayOfSensitiveMedia: "Anzeige von sensiblen Medien" | displayOfSensitiveMedia: "Darstellung sensibler Medien" | ||||||
| whenServerDisconnected: "Bei Verbindungsverlust zum Server" | whenServerDisconnected: "Bei Verbindungsverlust zum Server" | ||||||
| disconnectedFromServer: "Die Verbindung zum Server wurde getrennt" | disconnectedFromServer: "Die Verbindung zum Server wurde getrennt" | ||||||
| reload: "Aktualisieren" | reload: "Aktualisieren" | ||||||
| @@ -356,7 +356,6 @@ invite: "Einladen" | |||||||
| driveCapacityPerLocalAccount: "Drive-Kapazität pro lokalem Benutzerkonto" | driveCapacityPerLocalAccount: "Drive-Kapazität pro lokalem Benutzerkonto" | ||||||
| driveCapacityPerRemoteAccount: "Drive-Kapazität pro Benutzer fremder Instanzen" | driveCapacityPerRemoteAccount: "Drive-Kapazität pro Benutzer fremder Instanzen" | ||||||
| inMb: "In Megabytes" | inMb: "In Megabytes" | ||||||
| iconUrl: "Icon-URL (favicon etc)" |  | ||||||
| bannerUrl: "Banner-URL" | bannerUrl: "Banner-URL" | ||||||
| backgroundImageUrl: "Hintergrundbild-URL" | backgroundImageUrl: "Hintergrundbild-URL" | ||||||
| basicInfo: "Grundlegende Informationen" | basicInfo: "Grundlegende Informationen" | ||||||
| @@ -417,9 +416,12 @@ totp: "Authentifizierungs-App" | |||||||
| totpDescription: "Logge dich via Authentifizierungs-App mit Einmalpasswort ein" | totpDescription: "Logge dich via Authentifizierungs-App mit Einmalpasswort ein" | ||||||
| moderator: "Moderator" | moderator: "Moderator" | ||||||
| moderation: "Moderation" | moderation: "Moderation" | ||||||
|  | moderationNote: "Moderationsnotiz" | ||||||
|  | addModerationNote: "Moderationsnotiz hinzufügen" | ||||||
|  | moderationLogs: "Moderationsprotokolle" | ||||||
| nUsersMentioned: "Von {n} Benutzern erwähnt" | nUsersMentioned: "Von {n} Benutzern erwähnt" | ||||||
| securityKeyAndPasskey: "Security-Tokens und Passkeys" | securityKeyAndPasskey: "Hardware-Sicherheitsschlüssel und Passkeys" | ||||||
| securityKey: "Sicherheitsschlüssel" | securityKey: "Hardware-Sicherheitsschlüssel" | ||||||
| lastUsed: "Zuletzt benutzt" | lastUsed: "Zuletzt benutzt" | ||||||
| lastUsedAt: "Zuletzt verwendet: {t}" | lastUsedAt: "Zuletzt verwendet: {t}" | ||||||
| unregister: "Deaktivieren" | unregister: "Deaktivieren" | ||||||
| @@ -634,11 +636,11 @@ regexpErrorDescription: "Im regulären Ausdruck deiner {tab}en Wortstummschaltun | |||||||
| instanceMute: "Instanzstummschaltungen" | instanceMute: "Instanzstummschaltungen" | ||||||
| userSaysSomething: "{name} hat etwas gesagt" | userSaysSomething: "{name} hat etwas gesagt" | ||||||
| makeActive: "Aktivieren" | makeActive: "Aktivieren" | ||||||
| display: "Anzeigen" | display: "Anzeigeart" | ||||||
| copy: "Kopieren" | copy: "Kopieren" | ||||||
| metrics: "Metriken" | metrics: "Metriken" | ||||||
| overview: "Übersicht" | overview: "Übersicht" | ||||||
| logs: "Logs" | logs: "Protokolle" | ||||||
| delayed: "Verzögert" | delayed: "Verzögert" | ||||||
| database: "Datenbank" | database: "Datenbank" | ||||||
| channel: "Kanäle" | channel: "Kanäle" | ||||||
| @@ -709,6 +711,7 @@ lockedAccountInfo: "Auch wenn du Follow-Anfragen auf manuelle Bestätigung setzt | |||||||
| alwaysMarkSensitive: "Medien standardmäßig als sensibel markieren" | alwaysMarkSensitive: "Medien standardmäßig als sensibel markieren" | ||||||
| loadRawImages: "Anstatt Vorschaubilder immer Originalbilder anzeigen" | loadRawImages: "Anstatt Vorschaubilder immer Originalbilder anzeigen" | ||||||
| disableShowingAnimatedImages: "Animierte Bilder nicht abspielen" | disableShowingAnimatedImages: "Animierte Bilder nicht abspielen" | ||||||
|  | highlightSensitiveMedia: "Sensitive Medien markieren" | ||||||
| verificationEmailSent: "Eine Bestätigungsmail wurde an deine Email-Adresse versendet. Besuche den dort enthaltenen Link, um die Verifizierung abzuschließen." | verificationEmailSent: "Eine Bestätigungsmail wurde an deine Email-Adresse versendet. Besuche den dort enthaltenen Link, um die Verifizierung abzuschließen." | ||||||
| notSet: "Nicht konfiguriert" | notSet: "Nicht konfiguriert" | ||||||
| emailVerified: "Email-Adresse bestätigt" | emailVerified: "Email-Adresse bestätigt" | ||||||
| @@ -912,7 +915,7 @@ typeToConfirm: "Bitte gib zur Bestätigung {x} ein" | |||||||
| deleteAccount: "Benutzerkonto löschen" | deleteAccount: "Benutzerkonto löschen" | ||||||
| document: "Dokumentation" | document: "Dokumentation" | ||||||
| numberOfPageCache: "Seitencachegröße" | numberOfPageCache: "Seitencachegröße" | ||||||
| numberOfPageCacheDescription: "Das Erhöhen dieses Caches führt zu einer angenehmerern Benutzererfahrung, erhöht aber Serverlast und Arbeitsspeicherauslastung." | numberOfPageCacheDescription: "Das Erhöhen dieses Caches führt zu einer angenehmerern Benutzererfahrung, aber erhöht Last und Arbeitsspeicherauslastung auf dem Nutzergerät." | ||||||
| logoutConfirm: "Wirklich abmelden?" | logoutConfirm: "Wirklich abmelden?" | ||||||
| lastActiveDate: "Zuletzt verwendet am" | lastActiveDate: "Zuletzt verwendet am" | ||||||
| statusbar: "Statusleiste" | statusbar: "Statusleiste" | ||||||
| @@ -1023,7 +1026,7 @@ retryAllQueuesConfirmText: "Dies wird zu einer temporären Erhöhung der Serverl | |||||||
| enableChartsForRemoteUser: "Diagramme für Nutzer fremder Instanzen erstellen" | enableChartsForRemoteUser: "Diagramme für Nutzer fremder Instanzen erstellen" | ||||||
| enableChartsForFederatedInstances: "Diagramme für fremde Instanzen erstellen" | enableChartsForFederatedInstances: "Diagramme für fremde Instanzen erstellen" | ||||||
| showClipButtonInNoteFooter: "\"Clip\" zum Notizmenu hinzufügen" | showClipButtonInNoteFooter: "\"Clip\" zum Notizmenu hinzufügen" | ||||||
| largeNoteReactions: "Reaktionen vergrößert anzeigen" | reactionsDisplaySize: "Reaktionsanzeigegröße" | ||||||
| noteIdOrUrl: "Notiz-ID oder URL" | noteIdOrUrl: "Notiz-ID oder URL" | ||||||
| video: "Video" | video: "Video" | ||||||
| videos: "Videos" | videos: "Videos" | ||||||
| @@ -1047,7 +1050,7 @@ vertical: "Vertikal" | |||||||
| horizontal: "Horizontal" | horizontal: "Horizontal" | ||||||
| position: "Position" | position: "Position" | ||||||
| serverRules: "Serverregeln" | serverRules: "Serverregeln" | ||||||
| pleaseConfirmBelowBeforeSignup: "Lies bitte Untenstehendes vor der Registration." | pleaseConfirmBelowBeforeSignup: "Lies bitte diese Informationen und stimme ihnen vor der Registration zu." | ||||||
| pleaseAgreeAllToContinue: "Zum Fortfahren muss allen obigen Feldern zugestimmt werden." | pleaseAgreeAllToContinue: "Zum Fortfahren muss allen obigen Feldern zugestimmt werden." | ||||||
| continue: "Fortfahren" | continue: "Fortfahren" | ||||||
| preservedUsernames: "Reservierte Benutzernamen" | preservedUsernames: "Reservierte Benutzernamen" | ||||||
| @@ -1057,7 +1060,7 @@ archive: "Archivieren" | |||||||
| channelArchiveConfirmTitle: "{name} wirklich archivieren?" | channelArchiveConfirmTitle: "{name} wirklich archivieren?" | ||||||
| channelArchiveConfirmDescription: "Ein archivierter Kanal taucht nicht mehr in der Kanalliste oder in Suchergebnissen auf. Zudem können ihm keine Beiträge mehr hinzugefügt werden." | channelArchiveConfirmDescription: "Ein archivierter Kanal taucht nicht mehr in der Kanalliste oder in Suchergebnissen auf. Zudem können ihm keine Beiträge mehr hinzugefügt werden." | ||||||
| thisChannelArchived: "Dieser Kanal wurde archiviert." | thisChannelArchived: "Dieser Kanal wurde archiviert." | ||||||
| displayOfNote: "Anzeige von Notizen" | displayOfNote: "Darstellung von Notizen" | ||||||
| initialAccountSetting: "Kontoeinrichtung" | initialAccountSetting: "Kontoeinrichtung" | ||||||
| youFollowing: "Gefolgt" | youFollowing: "Gefolgt" | ||||||
| preventAiLearning: "Verwendung in machinellem Lernen (Generative bzw. Prediktive AI/KI) ablehnen" | preventAiLearning: "Verwendung in machinellem Lernen (Generative bzw. Prediktive AI/KI) ablehnen" | ||||||
| @@ -1105,6 +1108,24 @@ forYou: "Für dich" | |||||||
| currentAnnouncements: "Aktuelle Ankündigungen" | currentAnnouncements: "Aktuelle Ankündigungen" | ||||||
| pastAnnouncements: "Alte Ankündigungen" | pastAnnouncements: "Alte Ankündigungen" | ||||||
| youHaveUnreadAnnouncements: "Es gibt neue Ankündigungen." | youHaveUnreadAnnouncements: "Es gibt neue Ankündigungen." | ||||||
|  | useSecurityKey: "Folge bitten den Anweisungen deines Browsers bzw. Gerätes und verwende deinen Hardware-Sicherheitsschlüssel oder Passkey." | ||||||
|  | replies: "Antworten" | ||||||
|  | renotes: "Renotes" | ||||||
|  | loadReplies: "Antworten anzeigen" | ||||||
|  | loadConversation: "Unterhaltung anzeigen" | ||||||
|  | pinnedList: "Angeheftete Liste" | ||||||
|  | keepScreenOn: "Bildschirm angeschaltet lassen" | ||||||
|  | verifiedLink: "Link-Besitz wurde verifiziert" | ||||||
|  | notifyNotes: "Über neue Notizen benachrichtigen" | ||||||
|  | unnotifyNotes: "Nicht über neue Notizen benachrichtigen" | ||||||
|  | authentication: "Authentifikation" | ||||||
|  | authenticationRequiredToContinue: "Bitte authentifiziere dich, um fortzufahren" | ||||||
|  | dateAndTime: "Zeit" | ||||||
|  | showRenotes: "Renotes anzeigen" | ||||||
|  | edited: "Bearbeitet" | ||||||
|  | notificationRecieveConfig: "Benachrichtigungseinstellungen" | ||||||
|  | mutualFollow: "Gegenseitig gefolgt" | ||||||
|  | fileAttachedOnly: "Nur Notizen mit Dateien" | ||||||
| _announcement: | _announcement: | ||||||
|   forExistingUsers: "Nur für existierende Nutzer" |   forExistingUsers: "Nur für existierende Nutzer" | ||||||
|   forExistingUsersDescription: "Ist diese Option aktiviert, wird diese Ankündigung nur Nutzern angezeigt, die zum Zeitpunkt der Ankündigung bereits registriert sind. Ist sie deaktiviert, wird sie auch Nutzern, die sich nach dessen Veröffentlichung registrieren, angezeigt." |   forExistingUsersDescription: "Ist diese Option aktiviert, wird diese Ankündigung nur Nutzern angezeigt, die zum Zeitpunkt der Ankündigung bereits registriert sind. Ist sie deaktiviert, wird sie auch Nutzern, die sich nach dessen Veröffentlichung registrieren, angezeigt." | ||||||
| @@ -1131,6 +1152,15 @@ _initialAccountSetting: | |||||||
|   laterAreYouSure: "Die Kontoeinrichtung wirklich später erledigen?" |   laterAreYouSure: "Die Kontoeinrichtung wirklich später erledigen?" | ||||||
| _serverRules: | _serverRules: | ||||||
|   description: "Eine Reihe von Regeln, die vor der Registrierung angezeigt werden. Eine Zusammenfassung der Nutzungsbedingungen anzuzeigen ist empfohlen." |   description: "Eine Reihe von Regeln, die vor der Registrierung angezeigt werden. Eine Zusammenfassung der Nutzungsbedingungen anzuzeigen ist empfohlen." | ||||||
|  | _serverSettings: | ||||||
|  |   iconUrl: "Icon-URL" | ||||||
|  |   appIconDescription: "Gibt das zu verwendende Icon bei der Anzeige von {host} als App an." | ||||||
|  |   appIconUsageExample: "Beispielsweise als PWA, oder bei Lesezeichen auf dem Startbildschirm von Smartphones" | ||||||
|  |   appIconStyleRecommendation: "Da das Icon zu einem Kreis oder Quadrat zugeschnitten wird, wird ein Icon mit gefülltem Margin um den Inhalt herum empfohlen." | ||||||
|  |   appIconResolutionMustBe: "Die Mindestauflösung ist {resolution}." | ||||||
|  |   manifestJsonOverride: "Überschreiben von manifest.json" | ||||||
|  |   shortName: "Abkürzung" | ||||||
|  |   shortNameDescription: "Ein Kürzel für den Namen der Instanz, der angezeigt werden kann, falls der volle Instanzname lang ist." | ||||||
| _accountMigration: | _accountMigration: | ||||||
|   moveFrom: "Von einem anderen Konto zu diesem migrieren" |   moveFrom: "Von einem anderen Konto zu diesem migrieren" | ||||||
|   moveFromSub: "Alias für ein anderes Konto erstellen" |   moveFromSub: "Alias für ein anderes Konto erstellen" | ||||||
| @@ -1385,6 +1415,9 @@ _achievements: | |||||||
|       title: "Brain Diver" |       title: "Brain Diver" | ||||||
|       description: "Sende den Link zu Brain Diver" |       description: "Sende den Link zu Brain Diver" | ||||||
|       flavor: "Misskey-Misskey La-Tu-Ma" |       flavor: "Misskey-Misskey La-Tu-Ma" | ||||||
|  |     _smashTestNotificationButton: | ||||||
|  |       title: "Testüberfluss" | ||||||
|  |       description: "Betätige den Benachrichtigungstest mehrfach innerhalb einer extrem kurzen Zeitspanne" | ||||||
| _role: | _role: | ||||||
|   new: "Rolle erstellen" |   new: "Rolle erstellen" | ||||||
|   edit: "Rolle bearbeiten" |   edit: "Rolle bearbeiten" | ||||||
| @@ -1423,6 +1456,7 @@ _role: | |||||||
|     gtlAvailable: "Kann auf die globale Chronik zugreifen" |     gtlAvailable: "Kann auf die globale Chronik zugreifen" | ||||||
|     ltlAvailable: "Kann auf die lokale Chronik zugreifen" |     ltlAvailable: "Kann auf die lokale Chronik zugreifen" | ||||||
|     canPublicNote: "Kann öffentliche Notizen erstellen" |     canPublicNote: "Kann öffentliche Notizen erstellen" | ||||||
|  |     canEditNote: "Notizbearbeitung" | ||||||
|     canInvite: "Erstellung von Einladungscodes für diese Instanz" |     canInvite: "Erstellung von Einladungscodes für diese Instanz" | ||||||
|     inviteLimit: "Maximalanzahl an Einladungen" |     inviteLimit: "Maximalanzahl an Einladungen" | ||||||
|     inviteLimitCycle: "Zyklus des Einladungslimits" |     inviteLimitCycle: "Zyklus des Einladungslimits" | ||||||
| @@ -1508,6 +1542,7 @@ _plugin: | |||||||
|   install: "Plugins installieren" |   install: "Plugins installieren" | ||||||
|   installWarn: "Installiere bitte nur vertrauenswürdige Plugins." |   installWarn: "Installiere bitte nur vertrauenswürdige Plugins." | ||||||
|   manage: "Plugins verwalten" |   manage: "Plugins verwalten" | ||||||
|  |   viewSource: "Quelltext anzeigen" | ||||||
| _preferencesBackups: | _preferencesBackups: | ||||||
|   list: "Erstellte Backups" |   list: "Erstellte Backups" | ||||||
|   saveNew: "Neu erstellen" |   saveNew: "Neu erstellen" | ||||||
| @@ -1695,20 +1730,18 @@ _timelineTutorial: | |||||||
| _2fa: | _2fa: | ||||||
|   alreadyRegistered: "Du hast bereits ein Gerät für Zwei-Faktor-Authentifizierung registriert." |   alreadyRegistered: "Du hast bereits ein Gerät für Zwei-Faktor-Authentifizierung registriert." | ||||||
|   registerTOTP: "Authentifizierungs-App registrieren" |   registerTOTP: "Authentifizierungs-App registrieren" | ||||||
|   passwordToTOTP: "Bitte Passwort eingeben" |  | ||||||
|   step1: "Installiere zuerst eine Authentifizierungsapp (z.B. {a} oder {b}) auf deinem Gerät." |   step1: "Installiere zuerst eine Authentifizierungsapp (z.B. {a} oder {b}) auf deinem Gerät." | ||||||
|   step2: "Dann, scanne den angezeigten QR-Code mit deinem Gerät." |   step2: "Dann, scanne den angezeigten QR-Code mit deinem Gerät." | ||||||
|   step2Click: "Durch Klicken dieses QR-Codes kannst du Verifikation mit deinem Security-Token oder einer App registrieren." |   step2Click: "Durch Klicken dieses QR-Codes kannst du Verifikation mit deinem Security-Token oder einer App registrieren." | ||||||
|   step2Uri: "Nutzt du ein Desktopprogramm, gib folgende URI eingeben" |   step2Uri: "Nutzt du ein Desktopprogramm, gib folgende URI eingeben" | ||||||
|   step3Title: "Authentifizierungsscode eingeben" |   step3Title: "Authentifizierungsscode eingeben" | ||||||
|   step3: "Gib zum Abschluss den Token ein, der von deiner App angezeigt wird." |   step3: "Gib zum Abschluss den Code (Token) ein, der von deiner App angezeigt wird." | ||||||
|   setupCompleted: "Einrichtung abgeschlossen" |   setupCompleted: "Einrichtung abgeschlossen" | ||||||
|   step4: "Alle folgenden Anmeldeversuche werden ab sofort die Eingabe eines solchen Tokens benötigen." |   step4: "Alle folgenden Anmeldeversuche werden ab sofort die Eingabe eines solchen Tokens benötigen." | ||||||
|   securityKeyNotSupported: "Dein Browser unterstützt keine Security-Tokens." |   securityKeyNotSupported: "Dein Browser unterstützt keine Hardware-Sicherheitsschlüssel." | ||||||
|   registerTOTPBeforeKey: "Um einen Security-Token oder einen Passkey zu registrieren, musst du zuerst eine Authentifizierungs-App registrieren." |   registerTOTPBeforeKey: "Um einen Security-Token oder einen Passkey zu registrieren, musst du zuerst eine Authentifizierungs-App registrieren." | ||||||
|   securityKeyInfo: "Du kannst neben Fingerabdruck- oder PIN-Authentifizierung auf deinem Gerät auch Anmeldung mit Hilfe eines FIDO2-kompatiblen Hardware-Sicherheitsschlüssels einrichten." |   securityKeyInfo: "Du kannst neben Fingerabdruck- oder PIN-Authentifizierung auf deinem Gerät auch Anmeldung mit Hilfe eines FIDO2-kompatiblen Hardware-Sicherheitsschlüssels einrichten." | ||||||
|   chromePasskeyNotSupported: "Chrome-Passkeys werden zur Zeit nicht unterstützt." |   registerSecurityKey: "Hardware-Sicherheitsschlüssel oder Passkey registrieren" | ||||||
|   registerSecurityKey: "Security-Token oder Passkey registrieren" |  | ||||||
|   securityKeyName: "Schlüsselname eingeben" |   securityKeyName: "Schlüsselname eingeben" | ||||||
|   tapSecurityKey: "Bitten folge den Anweisungen deines Browsers zur Registrierung" |   tapSecurityKey: "Bitten folge den Anweisungen deines Browsers zur Registrierung" | ||||||
|   removeKey: "Sicherheitsschlüssel entfernen" |   removeKey: "Sicherheitsschlüssel entfernen" | ||||||
| @@ -1775,6 +1808,7 @@ _antennaSources: | |||||||
|   homeTimeline: "Notizen von Benutzern, denen gefolgt wird" |   homeTimeline: "Notizen von Benutzern, denen gefolgt wird" | ||||||
|   users: "Notizen von einem oder mehreren angegebenen Benutzern" |   users: "Notizen von einem oder mehreren angegebenen Benutzern" | ||||||
|   userList: "Notizen von allen Benutzern einer Liste" |   userList: "Notizen von allen Benutzern einer Liste" | ||||||
|  |   userBlacklist: "Alle Notizen abgesehen derer angegebener Benutzer" | ||||||
| _weekday: | _weekday: | ||||||
|   sunday: "Sonntag" |   sunday: "Sonntag" | ||||||
|   monday: "Montag" |   monday: "Montag" | ||||||
| @@ -1874,6 +1908,7 @@ _profile: | |||||||
|   metadataContent: "Inhalt" |   metadataContent: "Inhalt" | ||||||
|   changeAvatar: "Profilbild ändern" |   changeAvatar: "Profilbild ändern" | ||||||
|   changeBanner: "Banner ändern" |   changeBanner: "Banner ändern" | ||||||
|  |   verifiedLinkDescription: "Gibst du hier eine URL ein, die einen Link zu deinem Profile enthält, wird neben diesem Feld ein Icon zur Besitzbestätigung angezeigt." | ||||||
| _exportOrImport: | _exportOrImport: | ||||||
|   allNotes: "Alle Notizen" |   allNotes: "Alle Notizen" | ||||||
|   favoritedNotes: "Als Favorit markierte Notizen" |   favoritedNotes: "Als Favorit markierte Notizen" | ||||||
| @@ -1992,11 +2027,17 @@ _notification: | |||||||
|   youReceivedFollowRequest: "Du hast eine Follow-Anfrage erhalten" |   youReceivedFollowRequest: "Du hast eine Follow-Anfrage erhalten" | ||||||
|   yourFollowRequestAccepted: "Deine Follow-Anfrage wurde akzeptiert" |   yourFollowRequestAccepted: "Deine Follow-Anfrage wurde akzeptiert" | ||||||
|   pollEnded: "Umfrageergebnisse sind verfügbar" |   pollEnded: "Umfrageergebnisse sind verfügbar" | ||||||
|  |   newNote: "Neue Notiz" | ||||||
|   unreadAntennaNote: "Antenne {name}" |   unreadAntennaNote: "Antenne {name}" | ||||||
|   emptyPushNotificationMessage: "Push-Benachrichtigungen wurden aktualisiert" |   emptyPushNotificationMessage: "Push-Benachrichtigungen wurden aktualisiert" | ||||||
|   achievementEarned: "Errungenschaft freigeschaltet" |   achievementEarned: "Errungenschaft freigeschaltet" | ||||||
|  |   testNotification: "Testbenachrichtigung" | ||||||
|  |   checkNotificationBehavior: "Aussehen von Benachrichtigungen überprüfen" | ||||||
|  |   sendTestNotification: "Testbenachrichtigung senden" | ||||||
|  |   notificationWillBeDisplayedLikeThis: "Benachrichtigungen sehen so aus" | ||||||
|   _types: |   _types: | ||||||
|     all: "Alle" |     all: "Alle" | ||||||
|  |     note: "Neue Notizen" | ||||||
|     follow: "Neue Follower" |     follow: "Neue Follower" | ||||||
|     mention: "Erwähnungen" |     mention: "Erwähnungen" | ||||||
|     reply: "Antworten" |     reply: "Antworten" | ||||||
| @@ -2066,3 +2107,34 @@ _webhookSettings: | |||||||
|     renote: "Wenn du ein Renote erhältst" |     renote: "Wenn du ein Renote erhältst" | ||||||
|     reaction: "Wenn du eine Reaktion erhältst" |     reaction: "Wenn du eine Reaktion erhältst" | ||||||
|     mention: "Wenn du erwähnt wirst" |     mention: "Wenn du erwähnt wirst" | ||||||
|  | _moderationLogTypes: | ||||||
|  |   createRole: "Rolle erstellt" | ||||||
|  |   deleteRole: "Rolle gelöscht" | ||||||
|  |   updateRole: "Rolle aktualisiert" | ||||||
|  |   assignRole: "Zu Rolle zugewiesen" | ||||||
|  |   unassignRole: "Aus Rolle entfernt" | ||||||
|  |   suspend: "Gesperrt" | ||||||
|  |   unsuspend: "Entsperrt" | ||||||
|  |   addCustomEmoji: "Benutzerdefiniertes Emoji hinzugefügt" | ||||||
|  |   updateCustomEmoji: "Benutzerdefiniertes Emoji aktualisiert" | ||||||
|  |   deleteCustomEmoji: "Benutzerdefiniertes Emoji gelöscht" | ||||||
|  |   updateServerSettings: "Servereinstellungen aktualisiert" | ||||||
|  |   updateUserNote: "Moderationsnotiz aktualisiert" | ||||||
|  |   deleteDriveFile: "Datei gelöscht" | ||||||
|  |   deleteNote: "Notiz gelöscht" | ||||||
|  |   createGlobalAnnouncement: "Globale Ankündigung erstellt" | ||||||
|  |   createUserAnnouncement: "Benutzerspezifische Ankündigung erstellt" | ||||||
|  |   updateGlobalAnnouncement: "Globale Ankündigung aktualisiert" | ||||||
|  |   updateUserAnnouncement: "Benutzerspezifische Ankündigung aktualisiert" | ||||||
|  |   deleteGlobalAnnouncement: "Globale Ankündigung gelöscht" | ||||||
|  |   deleteUserAnnouncement: "Benutzerspezifische Ankündigung gelöscht" | ||||||
|  |   resetPassword: "Passwort zurückgesetzt" | ||||||
|  |   suspendRemoteInstance: "Fremde Instanz gesperrt" | ||||||
|  |   unsuspendRemoteInstance: "Fremde Instanz entsperrt" | ||||||
|  |   markSensitiveDriveFile: "Datei als sensitiv markiert" | ||||||
|  |   unmarkSensitiveDriveFile: "Datei als nicht sensitiv markiert" | ||||||
|  |   resolveAbuseReport: "Meldung bearbeitet" | ||||||
|  |   createInvitation: "Einladung erstellt" | ||||||
|  |   createAd: "Werbung erstellt" | ||||||
|  |   deleteAd: "Werbung gelöscht" | ||||||
|  |   updateAd: "Werbung aktualisiert" | ||||||
|   | |||||||
| @@ -288,6 +288,8 @@ file: "Αρχεία" | |||||||
| recommended: "Προτεινόμενα" | recommended: "Προτεινόμενα" | ||||||
| cannotUploadBecauseNoFreeSpace: "Το ανέβασμα απέτυχε λόγω ανεπαρκούς Αποθηκευτικού Χώρου" | cannotUploadBecauseNoFreeSpace: "Το ανέβασμα απέτυχε λόγω ανεπαρκούς Αποθηκευτικού Χώρου" | ||||||
| icon: "Εικονίδιο" | icon: "Εικονίδιο" | ||||||
|  | replies: "Απάντηση" | ||||||
|  | renotes: "Κοινοποίηση σημειώματος" | ||||||
| _email: | _email: | ||||||
|   _follow: |   _follow: | ||||||
|     title: "Έχετε ένα νέο ακόλουθο" |     title: "Έχετε ένα νέο ακόλουθο" | ||||||
| @@ -395,3 +397,5 @@ _deck: | |||||||
|     mentions: "Επισημάνσεις" |     mentions: "Επισημάνσεις" | ||||||
| _webhookSettings: | _webhookSettings: | ||||||
|   name: "Όνομα" |   name: "Όνομα" | ||||||
|  | _moderationLogTypes: | ||||||
|  |   suspend: "Αποβολή" | ||||||
|   | |||||||
| @@ -48,7 +48,7 @@ copyLink: "Copy link" | |||||||
| copyLinkRenote: "Copy renote link" | copyLinkRenote: "Copy renote link" | ||||||
| delete: "Delete" | delete: "Delete" | ||||||
| deleteAndEdit: "Delete and edit" | deleteAndEdit: "Delete and edit" | ||||||
| deleteAndEditConfirm: "Are you sure you want to delete this note and edit it? You will lose all reactions, renotes and replies to it." | deleteAndEditConfirm: "Are you sure you want to redraft this note? This means you will lose all reactions, renotes, and replies to it." | ||||||
| addToList: "Add to list" | addToList: "Add to list" | ||||||
| addToAntenna: "Add to antenna" | addToAntenna: "Add to antenna" | ||||||
| sendMessage: "Send a message" | sendMessage: "Send a message" | ||||||
| @@ -106,7 +106,7 @@ unfollow: "Unfollow" | |||||||
| followRequestPending: "Follow request pending" | followRequestPending: "Follow request pending" | ||||||
| enterEmoji: "Enter an emoji" | enterEmoji: "Enter an emoji" | ||||||
| renote: "Renote" | renote: "Renote" | ||||||
| unrenote: "Take back renote" | unrenote: "Remove renote" | ||||||
| renoted: "Renoted." | renoted: "Renoted." | ||||||
| cantRenote: "This post can't be renoted." | cantRenote: "This post can't be renoted." | ||||||
| cantReRenote: "A renote can't be renoted." | cantReRenote: "A renote can't be renoted." | ||||||
| @@ -356,7 +356,6 @@ invite: "Invite" | |||||||
| driveCapacityPerLocalAccount: "Drive capacity per local user" | driveCapacityPerLocalAccount: "Drive capacity per local user" | ||||||
| driveCapacityPerRemoteAccount: "Drive capacity per remote user" | driveCapacityPerRemoteAccount: "Drive capacity per remote user" | ||||||
| inMb: "In megabytes" | inMb: "In megabytes" | ||||||
| iconUrl: "Icon URL" |  | ||||||
| bannerUrl: "Banner image URL" | bannerUrl: "Banner image URL" | ||||||
| backgroundImageUrl: "Background image URL" | backgroundImageUrl: "Background image URL" | ||||||
| basicInfo: "Basic info" | basicInfo: "Basic info" | ||||||
| @@ -417,6 +416,9 @@ totp: "Authenticator App" | |||||||
| totpDescription: "Use an authenticator app to enter one-time passwords" | totpDescription: "Use an authenticator app to enter one-time passwords" | ||||||
| moderator: "Moderator" | moderator: "Moderator" | ||||||
| moderation: "Moderation" | moderation: "Moderation" | ||||||
|  | moderationNote: "Moderation note" | ||||||
|  | addModerationNote: "Add moderation note" | ||||||
|  | moderationLogs: "Moderation logs" | ||||||
| nUsersMentioned: "Mentioned by {n} users" | nUsersMentioned: "Mentioned by {n} users" | ||||||
| securityKeyAndPasskey: "Security- and passkeys" | securityKeyAndPasskey: "Security- and passkeys" | ||||||
| securityKey: "Security key" | securityKey: "Security key" | ||||||
| @@ -582,7 +584,7 @@ serviceworkerInfo: "Must be enabled for push notifications." | |||||||
| deletedNote: "Deleted note" | deletedNote: "Deleted note" | ||||||
| invisibleNote: "Invisible note" | invisibleNote: "Invisible note" | ||||||
| enableInfiniteScroll: "Automatically load more" | enableInfiniteScroll: "Automatically load more" | ||||||
| visibility: "Visiblility" | visibility: "Visibility" | ||||||
| poll: "Poll" | poll: "Poll" | ||||||
| useCw: "Hide content" | useCw: "Hide content" | ||||||
| enablePlayer: "Open video player" | enablePlayer: "Open video player" | ||||||
| @@ -709,6 +711,7 @@ lockedAccountInfo: "Unless you set your note visiblity to \"Followers only\", yo | |||||||
| alwaysMarkSensitive: "Mark as sensitive by default" | alwaysMarkSensitive: "Mark as sensitive by default" | ||||||
| loadRawImages: "Load original images instead of showing thumbnails" | loadRawImages: "Load original images instead of showing thumbnails" | ||||||
| disableShowingAnimatedImages: "Don't play animated images" | disableShowingAnimatedImages: "Don't play animated images" | ||||||
|  | highlightSensitiveMedia: "Highlight sensitive media" | ||||||
| verificationEmailSent: "A verification email has been sent. Please follow the included link to complete verification." | verificationEmailSent: "A verification email has been sent. Please follow the included link to complete verification." | ||||||
| notSet: "Not set" | notSet: "Not set" | ||||||
| emailVerified: "Email has been verified" | emailVerified: "Email has been verified" | ||||||
| @@ -912,7 +915,7 @@ typeToConfirm: "Please enter {x} to confirm" | |||||||
| deleteAccount: "Delete account" | deleteAccount: "Delete account" | ||||||
| document: "Documentation" | document: "Documentation" | ||||||
| numberOfPageCache: "Number of cached pages" | numberOfPageCache: "Number of cached pages" | ||||||
| numberOfPageCacheDescription: "Increasing this number will improve convenience for users but cause more server load as well as more memory to be used." | numberOfPageCacheDescription: "Increasing this number will improve convenience for but cause more load as more memory usage on the user's device." | ||||||
| logoutConfirm: "Really log out?" | logoutConfirm: "Really log out?" | ||||||
| lastActiveDate: "Last used at" | lastActiveDate: "Last used at" | ||||||
| statusbar: "Status bar" | statusbar: "Status bar" | ||||||
| @@ -1023,7 +1026,7 @@ retryAllQueuesConfirmText: "This will temporarily increase the server load." | |||||||
| enableChartsForRemoteUser: "Generate remote user data charts" | enableChartsForRemoteUser: "Generate remote user data charts" | ||||||
| enableChartsForFederatedInstances: "Generate remote instance data charts" | enableChartsForFederatedInstances: "Generate remote instance data charts" | ||||||
| showClipButtonInNoteFooter: "Add \"Clip\" to note action menu" | showClipButtonInNoteFooter: "Add \"Clip\" to note action menu" | ||||||
| largeNoteReactions: "Enlargen displayed reactions" | reactionsDisplaySize: "Reaction display size" | ||||||
| noteIdOrUrl: "Note ID or URL" | noteIdOrUrl: "Note ID or URL" | ||||||
| video: "Video" | video: "Video" | ||||||
| videos: "Videos" | videos: "Videos" | ||||||
| @@ -1105,6 +1108,24 @@ forYou: "For you" | |||||||
| currentAnnouncements: "Current announcements" | currentAnnouncements: "Current announcements" | ||||||
| pastAnnouncements: "Past announcements" | pastAnnouncements: "Past announcements" | ||||||
| youHaveUnreadAnnouncements: "There are unread announcements." | youHaveUnreadAnnouncements: "There are unread announcements." | ||||||
|  | useSecurityKey: "Please follow your browser's or device's instructions to use your security- or passkey." | ||||||
|  | replies: "Reply" | ||||||
|  | renotes: "Renotes" | ||||||
|  | loadReplies: "Show replies" | ||||||
|  | loadConversation: "Show conversation" | ||||||
|  | pinnedList: "Pinned list" | ||||||
|  | keepScreenOn: "Keep screen on" | ||||||
|  | verifiedLink: "Link ownership has been verified" | ||||||
|  | notifyNotes: "Notify about new notes" | ||||||
|  | unnotifyNotes: "Stop notifying about new notes" | ||||||
|  | authentication: "Authentication" | ||||||
|  | authenticationRequiredToContinue: "Please authenticate to continue" | ||||||
|  | dateAndTime: "Timestamp" | ||||||
|  | showRenotes: "Show renotes" | ||||||
|  | edited: "Edited" | ||||||
|  | notificationRecieveConfig: "Notification Settings" | ||||||
|  | mutualFollow: "Mutual follow" | ||||||
|  | fileAttachedOnly: "Only notes with files" | ||||||
| _announcement: | _announcement: | ||||||
|   forExistingUsers: "Existing users only" |   forExistingUsers: "Existing users only" | ||||||
|   forExistingUsersDescription: "This announcement will only be shown to users existing at the point of publishment if enabled. If disabled, those newly signing up after it has been posted will also see it." |   forExistingUsersDescription: "This announcement will only be shown to users existing at the point of publishment if enabled. If disabled, those newly signing up after it has been posted will also see it." | ||||||
| @@ -1131,6 +1152,15 @@ _initialAccountSetting: | |||||||
|   laterAreYouSure: "Really do profile setup later?" |   laterAreYouSure: "Really do profile setup later?" | ||||||
| _serverRules: | _serverRules: | ||||||
|   description: "A set of rules to be displayed before registration. Setting a summary of the Terms of Service is recommended." |   description: "A set of rules to be displayed before registration. Setting a summary of the Terms of Service is recommended." | ||||||
|  | _serverSettings: | ||||||
|  |   iconUrl: "Icon URL" | ||||||
|  |   appIconDescription: "Specifies the icon to use when {host} is displayed as an app." | ||||||
|  |   appIconUsageExample: "E.g. As PWA, or when displayed as a home screen bookmark on a phone" | ||||||
|  |   appIconStyleRecommendation: "As the icon may be cropped to a square or circle, an icon with colored margin around the content is recommended." | ||||||
|  |   appIconResolutionMustBe: "The minimum resolution is {resolution}." | ||||||
|  |   manifestJsonOverride: "manifest.json Override" | ||||||
|  |   shortName: "Short name" | ||||||
|  |   shortNameDescription: "A shorthand for the instance's name that can be displayed if the full official name is long." | ||||||
| _accountMigration: | _accountMigration: | ||||||
|   moveFrom: "Migrate another account to this one" |   moveFrom: "Migrate another account to this one" | ||||||
|   moveFromSub: "Create alias to another account" |   moveFromSub: "Create alias to another account" | ||||||
| @@ -1385,6 +1415,9 @@ _achievements: | |||||||
|       title: "Brain Diver" |       title: "Brain Diver" | ||||||
|       description: "Post the link to Brain Diver" |       description: "Post the link to Brain Diver" | ||||||
|       flavor: "Misskey-Misskey La-Tu-Ma" |       flavor: "Misskey-Misskey La-Tu-Ma" | ||||||
|  |     _smashTestNotificationButton: | ||||||
|  |       title: "Test overflow" | ||||||
|  |       description: "Trigger the notification test repeatedly within an extremely short time" | ||||||
| _role: | _role: | ||||||
|   new: "New role" |   new: "New role" | ||||||
|   edit: "Edit role" |   edit: "Edit role" | ||||||
| @@ -1423,6 +1456,7 @@ _role: | |||||||
|     gtlAvailable: "Can view the global timeline" |     gtlAvailable: "Can view the global timeline" | ||||||
|     ltlAvailable: "Can view the local timeline" |     ltlAvailable: "Can view the local timeline" | ||||||
|     canPublicNote: "Can send public notes" |     canPublicNote: "Can send public notes" | ||||||
|  |     canEditNote: "Note editing" | ||||||
|     canInvite: "Can create instance invite codes" |     canInvite: "Can create instance invite codes" | ||||||
|     inviteLimit: "Invite limit" |     inviteLimit: "Invite limit" | ||||||
|     inviteLimitCycle: "Invite limit cooldown" |     inviteLimitCycle: "Invite limit cooldown" | ||||||
| @@ -1508,6 +1542,7 @@ _plugin: | |||||||
|   install: "Install plugins" |   install: "Install plugins" | ||||||
|   installWarn: "Please do not install untrustworthy plugins." |   installWarn: "Please do not install untrustworthy plugins." | ||||||
|   manage: "Manage plugins" |   manage: "Manage plugins" | ||||||
|  |   viewSource: "View source" | ||||||
| _preferencesBackups: | _preferencesBackups: | ||||||
|   list: "Created backups" |   list: "Created backups" | ||||||
|   saveNew: "Save new backup" |   saveNew: "Save new backup" | ||||||
| @@ -1695,7 +1730,6 @@ _timelineTutorial: | |||||||
| _2fa: | _2fa: | ||||||
|   alreadyRegistered: "You have already registered a 2-factor authentication device." |   alreadyRegistered: "You have already registered a 2-factor authentication device." | ||||||
|   registerTOTP: "Register authenticator app" |   registerTOTP: "Register authenticator app" | ||||||
|   passwordToTOTP: "Enter your password" |  | ||||||
|   step1: "First, install an authentication app (such as {a} or {b}) on your device." |   step1: "First, install an authentication app (such as {a} or {b}) on your device." | ||||||
|   step2: "Then, scan the QR code displayed on this screen." |   step2: "Then, scan the QR code displayed on this screen." | ||||||
|   step2Click: "Clicking on this QR code will allow you to register 2FA to your security key or phone authenticator app." |   step2Click: "Clicking on this QR code will allow you to register 2FA to your security key or phone authenticator app." | ||||||
| @@ -1707,7 +1741,6 @@ _2fa: | |||||||
|   securityKeyNotSupported: "Your browser does not support security keys." |   securityKeyNotSupported: "Your browser does not support security keys." | ||||||
|   registerTOTPBeforeKey: "Please set up an authenticator app to register a security or pass key." |   registerTOTPBeforeKey: "Please set up an authenticator app to register a security or pass key." | ||||||
|   securityKeyInfo: "Besides fingerprint or PIN authentication, you can also setup authentication via hardware security keys that support FIDO2 to further secure your account." |   securityKeyInfo: "Besides fingerprint or PIN authentication, you can also setup authentication via hardware security keys that support FIDO2 to further secure your account." | ||||||
|   chromePasskeyNotSupported: "Chrome passkeys are currently not supported." |  | ||||||
|   registerSecurityKey: "Register a security or pass key" |   registerSecurityKey: "Register a security or pass key" | ||||||
|   securityKeyName: "Enter a key name" |   securityKeyName: "Enter a key name" | ||||||
|   tapSecurityKey: "Please follow your browser to register the security or pass key" |   tapSecurityKey: "Please follow your browser to register the security or pass key" | ||||||
| @@ -1775,6 +1808,7 @@ _antennaSources: | |||||||
|   homeTimeline: "Notes from followed users" |   homeTimeline: "Notes from followed users" | ||||||
|   users: "Notes from specific users" |   users: "Notes from specific users" | ||||||
|   userList: "Notes from a specified list of users" |   userList: "Notes from a specified list of users" | ||||||
|  |   userBlacklist: "All notes except for those of one or more specified users" | ||||||
| _weekday: | _weekday: | ||||||
|   sunday: "Sunday" |   sunday: "Sunday" | ||||||
|   monday: "Monday" |   monday: "Monday" | ||||||
| @@ -1874,6 +1908,7 @@ _profile: | |||||||
|   metadataContent: "Content" |   metadataContent: "Content" | ||||||
|   changeAvatar: "Change avatar" |   changeAvatar: "Change avatar" | ||||||
|   changeBanner: "Change banner" |   changeBanner: "Change banner" | ||||||
|  |   verifiedLinkDescription: "By entering an URL that contains a link to your profile here, an ownership verification icon can be displayed next to the field." | ||||||
| _exportOrImport: | _exportOrImport: | ||||||
|   allNotes: "All notes" |   allNotes: "All notes" | ||||||
|   favoritedNotes: "Favorite notes" |   favoritedNotes: "Favorite notes" | ||||||
| @@ -1992,11 +2027,17 @@ _notification: | |||||||
|   youReceivedFollowRequest: "You've received a follow request" |   youReceivedFollowRequest: "You've received a follow request" | ||||||
|   yourFollowRequestAccepted: "Your follow request was accepted" |   yourFollowRequestAccepted: "Your follow request was accepted" | ||||||
|   pollEnded: "Poll results have become available" |   pollEnded: "Poll results have become available" | ||||||
|  |   newNote: "New note" | ||||||
|   unreadAntennaNote: "Antenna {name}" |   unreadAntennaNote: "Antenna {name}" | ||||||
|   emptyPushNotificationMessage: "Push notifications have been updated" |   emptyPushNotificationMessage: "Push notifications have been updated" | ||||||
|   achievementEarned: "Achievement unlocked" |   achievementEarned: "Achievement unlocked" | ||||||
|  |   testNotification: "Test notification" | ||||||
|  |   checkNotificationBehavior: "Check notification appearance" | ||||||
|  |   sendTestNotification: "Send test notification" | ||||||
|  |   notificationWillBeDisplayedLikeThis: "Notifications look like this" | ||||||
|   _types: |   _types: | ||||||
|     all: "All" |     all: "All" | ||||||
|  |     note: "New notes" | ||||||
|     follow: "New followers" |     follow: "New followers" | ||||||
|     mention: "Mentions" |     mention: "Mentions" | ||||||
|     reply: "Replies" |     reply: "Replies" | ||||||
| @@ -2066,3 +2107,34 @@ _webhookSettings: | |||||||
|     renote: "When renoted" |     renote: "When renoted" | ||||||
|     reaction: "When receiving a reaction" |     reaction: "When receiving a reaction" | ||||||
|     mention: "When being mentioned" |     mention: "When being mentioned" | ||||||
|  | _moderationLogTypes: | ||||||
|  |   createRole: "Role created" | ||||||
|  |   deleteRole: "Role deleted" | ||||||
|  |   updateRole: "Role updated" | ||||||
|  |   assignRole: "Assigned to role" | ||||||
|  |   unassignRole: "Removed from role" | ||||||
|  |   suspend: "Suspended" | ||||||
|  |   unsuspend: "Unsuspended" | ||||||
|  |   addCustomEmoji: "Custom emoji added" | ||||||
|  |   updateCustomEmoji: "Custom emoji updated" | ||||||
|  |   deleteCustomEmoji: "Custom emoji deleted" | ||||||
|  |   updateServerSettings: "Server settings updated" | ||||||
|  |   updateUserNote: "Moderation note updated" | ||||||
|  |   deleteDriveFile: "File deleted" | ||||||
|  |   deleteNote: "Note deleted" | ||||||
|  |   createGlobalAnnouncement: "Global announcement created" | ||||||
|  |   createUserAnnouncement: "User announcement created" | ||||||
|  |   updateGlobalAnnouncement: "Global announcement updated" | ||||||
|  |   updateUserAnnouncement: "User announcement updated" | ||||||
|  |   deleteGlobalAnnouncement: "Global announcement deleted" | ||||||
|  |   deleteUserAnnouncement: "User announcement deleted" | ||||||
|  |   resetPassword: "Password reset" | ||||||
|  |   suspendRemoteInstance: "Remote instance suspended" | ||||||
|  |   unsuspendRemoteInstance: "Remote instance unsuspended" | ||||||
|  |   markSensitiveDriveFile: "File marked as sensitive" | ||||||
|  |   unmarkSensitiveDriveFile: "File unmarked as sensitive" | ||||||
|  |   resolveAbuseReport: "Report resolved" | ||||||
|  |   createInvitation: "Invite generated" | ||||||
|  |   createAd: "Ad created" | ||||||
|  |   deleteAd: "Ad deleted" | ||||||
|  |   updateAd: "Ad updated" | ||||||
|   | |||||||
| @@ -20,7 +20,7 @@ noNotes: "No hay notas" | |||||||
| noNotifications: "No hay notificaciones" | noNotifications: "No hay notificaciones" | ||||||
| instance: "Instancia" | instance: "Instancia" | ||||||
| settings: "Configuración" | settings: "Configuración" | ||||||
| notificationSettings: "Configurar las notificaciones" | notificationSettings: "Ajustes de notificaciones" | ||||||
| basicSettings: "Configuración básica" | basicSettings: "Configuración básica" | ||||||
| otherSettings: "Configuración avanzada" | otherSettings: "Configuración avanzada" | ||||||
| openInWindow: "Abrir en una ventana" | openInWindow: "Abrir en una ventana" | ||||||
| @@ -56,8 +56,8 @@ copyRSS: "Copiar RSS" | |||||||
| copyUsername: "Copiar nombre de usuario" | copyUsername: "Copiar nombre de usuario" | ||||||
| copyUserId: "Copiar ID del usuario" | copyUserId: "Copiar ID del usuario" | ||||||
| copyNoteId: "Copiar ID de la nota" | copyNoteId: "Copiar ID de la nota" | ||||||
| copyFileId: "Copiar un archivo ID" | copyFileId: "Copiar ID del archivo" | ||||||
| copyFolderId: "Copiar carpeta ID" | copyFolderId: "Copiar ID de carpeta" | ||||||
| copyProfileUrl: "Copiar la URL del perfil" | copyProfileUrl: "Copiar la URL del perfil" | ||||||
| searchUser: "Buscar un usuario" | searchUser: "Buscar un usuario" | ||||||
| reply: "Responder" | reply: "Responder" | ||||||
| @@ -356,7 +356,6 @@ invite: "Invitar" | |||||||
| driveCapacityPerLocalAccount: "Capacidad del drive por usuario local" | driveCapacityPerLocalAccount: "Capacidad del drive por usuario local" | ||||||
| driveCapacityPerRemoteAccount: "Capacidad del drive por usuario remoto" | driveCapacityPerRemoteAccount: "Capacidad del drive por usuario remoto" | ||||||
| inMb: "En megabytes" | inMb: "En megabytes" | ||||||
| iconUrl: "URL de la imagen del avatar" |  | ||||||
| bannerUrl: "URL de la imagen del banner" | bannerUrl: "URL de la imagen del banner" | ||||||
| backgroundImageUrl: "URL de la imagen de fondo" | backgroundImageUrl: "URL de la imagen de fondo" | ||||||
| basicInfo: "Información básica" | basicInfo: "Información básica" | ||||||
| @@ -417,6 +416,9 @@ totp: "Aplicación autentícadora" | |||||||
| totpDescription: "Ingresa una contaseña de un sólo uso usando la aplicación autenticadora" | totpDescription: "Ingresa una contaseña de un sólo uso usando la aplicación autenticadora" | ||||||
| moderator: "Moderador" | moderator: "Moderador" | ||||||
| moderation: "Moderación" | moderation: "Moderación" | ||||||
|  | moderationNote: "Nota de moderación" | ||||||
|  | addModerationNote: "Añadir nota de moderación" | ||||||
|  | moderationLogs: "Log de moderación" | ||||||
| nUsersMentioned: "{n} usuarios mencionados" | nUsersMentioned: "{n} usuarios mencionados" | ||||||
| securityKeyAndPasskey: "Clave de seguridad / clave de paso" | securityKeyAndPasskey: "Clave de seguridad / clave de paso" | ||||||
| securityKey: "Clave de seguridad" | securityKey: "Clave de seguridad" | ||||||
| @@ -709,6 +711,7 @@ lockedAccountInfo: "A menos que configures la visibilidad de tus notas como \"S | |||||||
| alwaysMarkSensitive: "Marcar los medios de comunicación como contenido sensible por defecto" | alwaysMarkSensitive: "Marcar los medios de comunicación como contenido sensible por defecto" | ||||||
| loadRawImages: "Cargar las imágenes originales en lugar de mostrar las miniaturas" | loadRawImages: "Cargar las imágenes originales en lugar de mostrar las miniaturas" | ||||||
| disableShowingAnimatedImages: "No reproducir imágenes animadas" | disableShowingAnimatedImages: "No reproducir imágenes animadas" | ||||||
|  | highlightSensitiveMedia: "Resaltar medios marcados como sensibles" | ||||||
| verificationEmailSent: "Se le ha enviado un correo electrónico de confirmación. Por favor, acceda al enlace proporcionado en el correo electrónico para completar la configuración." | verificationEmailSent: "Se le ha enviado un correo electrónico de confirmación. Por favor, acceda al enlace proporcionado en el correo electrónico para completar la configuración." | ||||||
| notSet: "Sin especificar" | notSet: "Sin especificar" | ||||||
| emailVerified: "Su dirección de correo electrónico ha sido verificada." | emailVerified: "Su dirección de correo electrónico ha sido verificada." | ||||||
| @@ -1023,7 +1026,7 @@ retryAllQueuesConfirmText: "La carga del servidor está incrementándose tempora | |||||||
| enableChartsForRemoteUser: "Generar gráficas de usuarios remotos." | enableChartsForRemoteUser: "Generar gráficas de usuarios remotos." | ||||||
| enableChartsForFederatedInstances: "Generar gráficos de servidores remotos" | enableChartsForFederatedInstances: "Generar gráficos de servidores remotos" | ||||||
| showClipButtonInNoteFooter: "Añadir \"Clip\" al menú de notas" | showClipButtonInNoteFooter: "Añadir \"Clip\" al menú de notas" | ||||||
| largeNoteReactions: "Agrandar las reacciones de las notas" | reactionsDisplaySize: "Tamaño de las reacciones" | ||||||
| noteIdOrUrl: "ID o URL de la nota" | noteIdOrUrl: "ID o URL de la nota" | ||||||
| video: "Video" | video: "Video" | ||||||
| videos: "Video" | videos: "Video" | ||||||
| @@ -1105,6 +1108,19 @@ forYou: "Para ti" | |||||||
| currentAnnouncements: "Anuncios actuales" | currentAnnouncements: "Anuncios actuales" | ||||||
| pastAnnouncements: "Anuncios anteriores" | pastAnnouncements: "Anuncios anteriores" | ||||||
| youHaveUnreadAnnouncements: "Hay anuncios sin leer" | youHaveUnreadAnnouncements: "Hay anuncios sin leer" | ||||||
|  | useSecurityKey: "Por favor, sigue las instrucciones de tu dispositivo o navegador para usar tu clave de seguridad o tu clave de paso." | ||||||
|  | replies: "Responder" | ||||||
|  | renotes: "Renotar" | ||||||
|  | loadReplies: "Ver respuestas" | ||||||
|  | loadConversation: "Ver conversación" | ||||||
|  | pinnedList: "Lista fijada" | ||||||
|  | keepScreenOn: "Mantener pantalla encendida" | ||||||
|  | verifiedLink: "Propiedad del enlace verificada" | ||||||
|  | notifyNotes: "Notificar nuevas notas" | ||||||
|  | unnotifyNotes: "Dejar de notificar nuevas notas" | ||||||
|  | authentication: "Autenticación" | ||||||
|  | authenticationRequiredToContinue: "Por favor, autentifícate para continuar" | ||||||
|  | dateAndTime: "Fecha y hora" | ||||||
| _announcement: | _announcement: | ||||||
|   forExistingUsers: "Solo para usuarios registrados" |   forExistingUsers: "Solo para usuarios registrados" | ||||||
|   forExistingUsersDescription: "Este anuncio solo se mostrará a aquellos usuarios registrados en el momento de su publicación. Si se deshabilita esta opción, aquellos usuarios que se registren tras su publicación también lo verán." |   forExistingUsersDescription: "Este anuncio solo se mostrará a aquellos usuarios registrados en el momento de su publicación. Si se deshabilita esta opción, aquellos usuarios que se registren tras su publicación también lo verán." | ||||||
| @@ -1131,6 +1147,15 @@ _initialAccountSetting: | |||||||
|   laterAreYouSure: "¿Realmente quieres configurar tu perfil después?" |   laterAreYouSure: "¿Realmente quieres configurar tu perfil después?" | ||||||
| _serverRules: | _serverRules: | ||||||
|   description: "Un conjunto de reglas que serán mostradas antes del registro. Configurar un sumario de términos de servicio es recomendado." |   description: "Un conjunto de reglas que serán mostradas antes del registro. Configurar un sumario de términos de servicio es recomendado." | ||||||
|  | _serverSettings: | ||||||
|  |   iconUrl: "URL del ícono" | ||||||
|  |   appIconDescription: "Indica el icono que se va a usar cuando {host} se muestre como una app." | ||||||
|  |   appIconUsageExample: "Por ejemplo, como PWA o cuando se muestre como un marcador en la pantalla inicial del dispositivo" | ||||||
|  |   appIconStyleRecommendation: "Como el icono puede ser recortado como un cuadrado o un círculo, se recomienda un icono con un margen coloreado alrededor del contenido." | ||||||
|  |   appIconResolutionMustBe: "La resolución mínima es {resolution}." | ||||||
|  |   manifestJsonOverride: "Sobreescribir manifest.json" | ||||||
|  |   shortName: "Nombre corto" | ||||||
|  |   shortNameDescription: "Forma corta del nombre de la instancia que puede mostrarse si el nombre completo es demasiado largo." | ||||||
| _accountMigration: | _accountMigration: | ||||||
|   moveFrom: "Trasladar de otra cuenta a ésta" |   moveFrom: "Trasladar de otra cuenta a ésta" | ||||||
|   moveFromSub: "Crear un alias para otra cuenta." |   moveFromSub: "Crear un alias para otra cuenta." | ||||||
| @@ -1385,6 +1410,9 @@ _achievements: | |||||||
|       title: "Brain Diver" |       title: "Brain Diver" | ||||||
|       description: "Publicaste un vínculo a \"Brain Diver\"" |       description: "Publicaste un vínculo a \"Brain Diver\"" | ||||||
|       flavor: "Misskey-Misskey La-Tu-Ma" |       flavor: "Misskey-Misskey La-Tu-Ma" | ||||||
|  |     _smashTestNotificationButton: | ||||||
|  |       title: "Sobrecarga de pruebas" | ||||||
|  |       description: "Envía muchas notificaciones de prueba en un corto espacio de tiempo" | ||||||
| _role: | _role: | ||||||
|   new: "Crear rol" |   new: "Crear rol" | ||||||
|   edit: "Editar rol" |   edit: "Editar rol" | ||||||
| @@ -1508,6 +1536,7 @@ _plugin: | |||||||
|   install: "Instalar plugins" |   install: "Instalar plugins" | ||||||
|   installWarn: "Por favor no instale plugins que no son de confianza" |   installWarn: "Por favor no instale plugins que no son de confianza" | ||||||
|   manage: "Gestionar plugins" |   manage: "Gestionar plugins" | ||||||
|  |   viewSource: "Ver la fuente" | ||||||
| _preferencesBackups: | _preferencesBackups: | ||||||
|   list: "Respaldos creados" |   list: "Respaldos creados" | ||||||
|   saveNew: "Guardar nuevo respaldo" |   saveNew: "Guardar nuevo respaldo" | ||||||
| @@ -1695,7 +1724,6 @@ _timelineTutorial: | |||||||
| _2fa: | _2fa: | ||||||
|   alreadyRegistered: "Ya has completado la configuración." |   alreadyRegistered: "Ya has completado la configuración." | ||||||
|   registerTOTP: "Registrar aplicación autenticadora" |   registerTOTP: "Registrar aplicación autenticadora" | ||||||
|   passwordToTOTP: "Ingresa tu contraseña" |  | ||||||
|   step1: "Primero, instale en su dispositivo la aplicación de autenticación {a} o {b} u otra." |   step1: "Primero, instale en su dispositivo la aplicación de autenticación {a} o {b} u otra." | ||||||
|   step2: "Luego, escanee con la aplicación el código QR mostrado en pantalla." |   step2: "Luego, escanee con la aplicación el código QR mostrado en pantalla." | ||||||
|   step2Click: "Clicking on this QR code will allow you to register 2FA to your security key or phone authenticator app.\nTocar este código QR te permitirá registrar la autenticación 2FA a tu llave de seguridad o aplicación autenticadora." |   step2Click: "Clicking on this QR code will allow you to register 2FA to your security key or phone authenticator app.\nTocar este código QR te permitirá registrar la autenticación 2FA a tu llave de seguridad o aplicación autenticadora." | ||||||
| @@ -1707,7 +1735,6 @@ _2fa: | |||||||
|   securityKeyNotSupported: "Tu navegador no soporta claves de autenticación." |   securityKeyNotSupported: "Tu navegador no soporta claves de autenticación." | ||||||
|   registerTOTPBeforeKey: "Please set up an authenticator app to register a security or pass key.\npor favor. configura una aplicación de autenticación para registrar una llave de seguridad." |   registerTOTPBeforeKey: "Please set up an authenticator app to register a security or pass key.\npor favor. configura una aplicación de autenticación para registrar una llave de seguridad." | ||||||
|   securityKeyInfo: "Se puede configurar el inicio de sesión usando una clave de seguridad de hardware que soporte FIDO2 o con un certificado de huella digital o con un PIN" |   securityKeyInfo: "Se puede configurar el inicio de sesión usando una clave de seguridad de hardware que soporte FIDO2 o con un certificado de huella digital o con un PIN" | ||||||
|   chromePasskeyNotSupported: "Las llaves de seguridad de Chrome no son soportadas por el momento." |  | ||||||
|   registerSecurityKey: "Registrar una llave de seguridad" |   registerSecurityKey: "Registrar una llave de seguridad" | ||||||
|   securityKeyName: "Ingresa un nombre para la clave" |   securityKeyName: "Ingresa un nombre para la clave" | ||||||
|   tapSecurityKey: "Por favor, sigue tu navegador para registrar una llave de seguridad" |   tapSecurityKey: "Por favor, sigue tu navegador para registrar una llave de seguridad" | ||||||
| @@ -1775,6 +1802,7 @@ _antennaSources: | |||||||
|   homeTimeline: "Notas de los usuarios que sigues" |   homeTimeline: "Notas de los usuarios que sigues" | ||||||
|   users: "Notas de un usuario o varios" |   users: "Notas de un usuario o varios" | ||||||
|   userList: "Notas de los usuarios de una lista" |   userList: "Notas de los usuarios de una lista" | ||||||
|  |   userBlacklist: "Todas las notas excepto aquellas de uno o más usuarios especificados" | ||||||
| _weekday: | _weekday: | ||||||
|   sunday: "Domingo" |   sunday: "Domingo" | ||||||
|   monday: "Lunes" |   monday: "Lunes" | ||||||
| @@ -1874,6 +1902,7 @@ _profile: | |||||||
|   metadataContent: "Contenido" |   metadataContent: "Contenido" | ||||||
|   changeAvatar: "Cambiar avatar" |   changeAvatar: "Cambiar avatar" | ||||||
|   changeBanner: "Cambiar banner" |   changeBanner: "Cambiar banner" | ||||||
|  |   verifiedLinkDescription: "Introduciendo una URL que contiene un enlace a tu perfil, se puede mostrar un icono de verificación de propiedad al lado del campo." | ||||||
| _exportOrImport: | _exportOrImport: | ||||||
|   allNotes: "Todas las notas" |   allNotes: "Todas las notas" | ||||||
|   favoritedNotes: "Notas favoritas" |   favoritedNotes: "Notas favoritas" | ||||||
| @@ -1992,11 +2021,17 @@ _notification: | |||||||
|   youReceivedFollowRequest: "Has mandado una solicitud de seguimiento" |   youReceivedFollowRequest: "Has mandado una solicitud de seguimiento" | ||||||
|   yourFollowRequestAccepted: "Tu solicitud de seguimiento fue aceptada" |   yourFollowRequestAccepted: "Tu solicitud de seguimiento fue aceptada" | ||||||
|   pollEnded: "Estan disponibles los resultados de la encuesta" |   pollEnded: "Estan disponibles los resultados de la encuesta" | ||||||
|  |   newNote: "Nueva nota" | ||||||
|   unreadAntennaNote: "Antena {name}" |   unreadAntennaNote: "Antena {name}" | ||||||
|   emptyPushNotificationMessage: "Se han actualizado las notificaciones push" |   emptyPushNotificationMessage: "Se han actualizado las notificaciones push" | ||||||
|   achievementEarned: "Logro desbloqueado" |   achievementEarned: "Logro desbloqueado" | ||||||
|  |   testNotification: "Notificación de prueba" | ||||||
|  |   checkNotificationBehavior: "Comprobar comportamiento de la notificación" | ||||||
|  |   sendTestNotification: "Enviar notificación de prueba" | ||||||
|  |   notificationWillBeDisplayedLikeThis: "Las notificaciones tendrán este aspecto" | ||||||
|   _types: |   _types: | ||||||
|     all: "Todo" |     all: "Todo" | ||||||
|  |     note: "Nuevas notas" | ||||||
|     follow: "Siguiendo" |     follow: "Siguiendo" | ||||||
|     mention: "Menciones" |     mention: "Menciones" | ||||||
|     reply: "Respuestas" |     reply: "Respuestas" | ||||||
| @@ -2066,3 +2101,31 @@ _webhookSettings: | |||||||
|     renote: "Cuando reciba un \"re-note\"" |     renote: "Cuando reciba un \"re-note\"" | ||||||
|     reaction: "Cuando se recibe una reacción" |     reaction: "Cuando se recibe una reacción" | ||||||
|     mention: "Cuando hay una mención" |     mention: "Cuando hay una mención" | ||||||
|  | _moderationLogTypes: | ||||||
|  |   createRole: "Rol creado" | ||||||
|  |   deleteRole: "Rol eliminado" | ||||||
|  |   updateRole: "Rol actualizado" | ||||||
|  |   assignRole: "Rol asignado" | ||||||
|  |   unassignRole: "Rol retirado" | ||||||
|  |   suspend: "Suspender" | ||||||
|  |   unsuspend: "Suspensión retirada" | ||||||
|  |   addCustomEmoji: "Añadido emoji personalizado" | ||||||
|  |   updateCustomEmoji: "Emoji personalizado actualizado" | ||||||
|  |   deleteCustomEmoji: "Emoji personalizado eliminado" | ||||||
|  |   updateServerSettings: "Ajustes de servidor actualizados" | ||||||
|  |   updateUserNote: "Nota de moderación actualizada" | ||||||
|  |   deleteDriveFile: "Archivo eliminado" | ||||||
|  |   deleteNote: "Nota eliminada" | ||||||
|  |   createGlobalAnnouncement: "Anuncio global creado" | ||||||
|  |   createUserAnnouncement: "Anuncio de usuario creado" | ||||||
|  |   updateGlobalAnnouncement: "Anuncio global actualizado" | ||||||
|  |   updateUserAnnouncement: "Anuncio de usuario actualizado" | ||||||
|  |   deleteGlobalAnnouncement: "Anuncio global eliminado" | ||||||
|  |   deleteUserAnnouncement: "Anuncio de usuario eliminado" | ||||||
|  |   resetPassword: "Resetear contraseña" | ||||||
|  |   suspendRemoteInstance: "Instancia remota suspendida" | ||||||
|  |   unsuspendRemoteInstance: "Suspensión de instancia remota retirada" | ||||||
|  |   markSensitiveDriveFile: "Archivo marcado como sensible" | ||||||
|  |   unmarkSensitiveDriveFile: "Archivo marcado como no sensible" | ||||||
|  |   resolveAbuseReport: "Reporte resuelto" | ||||||
|  |   createInvitation: "Generar invitación" | ||||||
|   | |||||||
| @@ -47,7 +47,7 @@ copyContent: "Copier le contenu" | |||||||
| copyLink: "Copier le lien" | copyLink: "Copier le lien" | ||||||
| delete: "Supprimer" | delete: "Supprimer" | ||||||
| deleteAndEdit: "Supprimer et réécrire" | deleteAndEdit: "Supprimer et réécrire" | ||||||
| deleteAndEditConfirm: "Êtes-vous sûr·e de vouloir supprimer cette note et la reformuler ? Vous perdrez toutes les réactions, renotes et réponses y afférentes." | deleteAndEditConfirm: "Êtes-vous sûr de vouloir effacer cette note et la modifier ? Vous perdrez toutes les réactions, renotes et réponses." | ||||||
| addToList: "Ajouter à une liste" | addToList: "Ajouter à une liste" | ||||||
| addToAntenna: "Ajouter à l’antenne" | addToAntenna: "Ajouter à l’antenne" | ||||||
| sendMessage: "Envoyer un message" | sendMessage: "Envoyer un message" | ||||||
| @@ -272,6 +272,7 @@ startMessaging: "Commencer à discuter" | |||||||
| nUsersRead: "Lu par {n} personnes" | nUsersRead: "Lu par {n} personnes" | ||||||
| agreeTo: "J’accepte {0}" | agreeTo: "J’accepte {0}" | ||||||
| agree: "Accepter" | agree: "Accepter" | ||||||
|  | agreeBelow: "J’accepte ce qui suit" | ||||||
| basicNotesBeforeCreateAccount: "Notes importantes" | basicNotesBeforeCreateAccount: "Notes importantes" | ||||||
| termsOfService: "Conditions d'utilisation" | termsOfService: "Conditions d'utilisation" | ||||||
| start: "Commencer" | start: "Commencer" | ||||||
| @@ -351,7 +352,6 @@ invite: "Inviter" | |||||||
| driveCapacityPerLocalAccount: "Volume du Drive par utilisateur local" | driveCapacityPerLocalAccount: "Volume du Drive par utilisateur local" | ||||||
| driveCapacityPerRemoteAccount: "Volume du Drive par utilisateur distant" | driveCapacityPerRemoteAccount: "Volume du Drive par utilisateur distant" | ||||||
| inMb: "en mégaoctets" | inMb: "en mégaoctets" | ||||||
| iconUrl: "URL de l'icône" |  | ||||||
| bannerUrl: "URL de l’image de la bannière" | bannerUrl: "URL de l’image de la bannière" | ||||||
| backgroundImageUrl: "URL de l'image d'arrière-plan" | backgroundImageUrl: "URL de l'image d'arrière-plan" | ||||||
| basicInfo: "Informations basiques" | basicInfo: "Informations basiques" | ||||||
| @@ -407,11 +407,15 @@ aboutMisskey: "À propos de Misskey" | |||||||
| administrator: "Administrateur" | administrator: "Administrateur" | ||||||
| token: "Jeton" | token: "Jeton" | ||||||
| 2fa: "Authentification à deux facteurs" | 2fa: "Authentification à deux facteurs" | ||||||
|  | setupOf2fa: "Configuration de l’authentification à deux facteurs" | ||||||
| totp: "Application d'authentification" | totp: "Application d'authentification" | ||||||
| totpDescription: "Entrez un mot de passe à usage unique à l'aide d'une application d'authentification" | totpDescription: "Entrez un mot de passe à usage unique à l'aide d'une application d'authentification" | ||||||
| moderator: "Modérateur·rice·s" | moderator: "Modérateur·rice·s" | ||||||
| moderation: "Modérations" | moderation: "Modérations" | ||||||
|  | moderationNote: "Note de modération" | ||||||
|  | addModerationNote: "Ajouter une note de modération" | ||||||
| nUsersMentioned: "{n} utilisateur·rice·s mentionné·e·s" | nUsersMentioned: "{n} utilisateur·rice·s mentionné·e·s" | ||||||
|  | securityKeyAndPasskey: "Sécurité et clés de sécurité" | ||||||
| securityKey: "Clé de sécurité" | securityKey: "Clé de sécurité" | ||||||
| lastUsed: "Dernier utilisé" | lastUsed: "Dernier utilisé" | ||||||
| lastUsedAt: "Dernière utilisation : {t}" | lastUsedAt: "Dernière utilisation : {t}" | ||||||
| @@ -479,6 +483,7 @@ createAccount: "Créer un compte" | |||||||
| existingAccount: "Compte existant" | existingAccount: "Compte existant" | ||||||
| regenerate: "Générer à nouveau" | regenerate: "Générer à nouveau" | ||||||
| fontSize: "Taille de la police" | fontSize: "Taille de la police" | ||||||
|  | limitTo: "Limiter à {x}" | ||||||
| noFollowRequests: "Vous n’avez aucune demande d’abonnement en attente" | noFollowRequests: "Vous n’avez aucune demande d’abonnement en attente" | ||||||
| openImageInNewTab: "Ouvrir les images dans un nouvel onglet" | openImageInNewTab: "Ouvrir les images dans un nouvel onglet" | ||||||
| dashboard: "Tableau de bord" | dashboard: "Tableau de bord" | ||||||
| @@ -795,6 +800,7 @@ popularPosts: "Les plus consultées" | |||||||
| shareWithNote: "Partager dans une note" | shareWithNote: "Partager dans une note" | ||||||
| ads: "Publicité" | ads: "Publicité" | ||||||
| expiration: "Échéance" | expiration: "Échéance" | ||||||
|  | startingperiod: "Commencer" | ||||||
| memo: "Pense-bête" | memo: "Pense-bête" | ||||||
| priority: "Priorité" | priority: "Priorité" | ||||||
| high: "Haute" | high: "Haute" | ||||||
| @@ -821,6 +827,7 @@ translatedFrom: "Traduit depuis {x}" | |||||||
| accountDeletionInProgress: "La suppression de votre compte est en cours" | accountDeletionInProgress: "La suppression de votre compte est en cours" | ||||||
| usernameInfo: "C'est un nom qui identifie votre compte sur l'instance de manière unique. Vous pouvez utiliser des lettres de l'alphabet (minuscules et majuscules), des chiffres (de 0 à 9), ou bien le tiret « _ ». Vous ne pourrez pas modifier votre nom d'utilisateur·rice par la suite." | usernameInfo: "C'est un nom qui identifie votre compte sur l'instance de manière unique. Vous pouvez utiliser des lettres de l'alphabet (minuscules et majuscules), des chiffres (de 0 à 9), ou bien le tiret « _ ». Vous ne pourrez pas modifier votre nom d'utilisateur·rice par la suite." | ||||||
| aiChanMode: "Mode Ai" | aiChanMode: "Mode Ai" | ||||||
|  | devMode: "Mode développement" | ||||||
| keepCw: "Garder le CW" | keepCw: "Garder le CW" | ||||||
| pubSub: "Comptes Pub/Sub" | pubSub: "Comptes Pub/Sub" | ||||||
| lastCommunication: "Dernière communication" | lastCommunication: "Dernière communication" | ||||||
| @@ -940,6 +947,7 @@ roles: "Rôles" | |||||||
| role: "Rôles" | role: "Rôles" | ||||||
| noRole: "Aucun rôle" | noRole: "Aucun rôle" | ||||||
| normalUser: "Simple utilisateur·rice" | normalUser: "Simple utilisateur·rice" | ||||||
|  | undefined: "Non défini" | ||||||
| assign: "Attribuer" | assign: "Attribuer" | ||||||
| color: "Couleur" | color: "Couleur" | ||||||
| manageCustomEmojis: "Gestion des émojis personnalisés" | manageCustomEmojis: "Gestion des émojis personnalisés" | ||||||
| @@ -947,17 +955,24 @@ preset: "Préréglage" | |||||||
| selectFromPresets: "Sélectionner à partir des préréglages" | selectFromPresets: "Sélectionner à partir des préréglages" | ||||||
| achievements: "Accomplissements" | achievements: "Accomplissements" | ||||||
| thisPostMayBeAnnoying: "Cette note peut gêner d'autres personnes." | thisPostMayBeAnnoying: "Cette note peut gêner d'autres personnes." | ||||||
|  | thisPostMayBeAnnoyingHome: "Publier vers le fil principal" | ||||||
| thisPostMayBeAnnoyingCancel: "Annuler" | thisPostMayBeAnnoyingCancel: "Annuler" | ||||||
| thisPostMayBeAnnoyingIgnore: "Publier quand-même" | thisPostMayBeAnnoyingIgnore: "Publier quand-même" | ||||||
| internalServerError: "Erreur interne du serveur" | internalServerError: "Erreur interne du serveur" | ||||||
|  | copyErrorInfo: "Copier les détails de l’erreur" | ||||||
|  | exploreOtherServers: "Trouver une autre instance" | ||||||
| disableFederationOk: "Désactiver" | disableFederationOk: "Désactiver" | ||||||
|  | likeOnly: "Les favoris uniquement" | ||||||
| license: "Licence" | license: "Licence" | ||||||
| video: "Vidéo" | video: "Vidéo" | ||||||
| videos: "Vidéos" | videos: "Vidéos" | ||||||
| dataSaver: "Économiseur de données" | dataSaver: "Économiseur de données" | ||||||
| accountMigration: "Migration de compte" | accountMigration: "Migration de compte" | ||||||
| accountMoved: "Cet·te utilisateur·rice a migré son compte vers :" | accountMoved: "Cet·te utilisateur·rice a migré son compte vers :" | ||||||
|  | accountMovedShort: "Ce compte a migré" | ||||||
|  | operationForbidden: "Opération non autorisée" | ||||||
| addMemo: "Ajouter un mémo" | addMemo: "Ajouter un mémo" | ||||||
|  | reactionsList: "Réactions" | ||||||
| notificationDisplay: "Style des notifications" | notificationDisplay: "Style des notifications" | ||||||
| leftTop: "En haut à gauche" | leftTop: "En haut à gauche" | ||||||
| rightTop: "En haut à droite" | rightTop: "En haut à droite" | ||||||
| @@ -966,7 +981,9 @@ rightBottom: "En bas à droite" | |||||||
| vertical: "Vertical" | vertical: "Vertical" | ||||||
| horizontal: "Latéral" | horizontal: "Latéral" | ||||||
| serverRules: "Règles du serveur" | serverRules: "Règles du serveur" | ||||||
|  | archive: "Archive" | ||||||
| youFollowing: "Abonné·e" | youFollowing: "Abonné·e" | ||||||
|  | options: "Options" | ||||||
| later: "Plus tard" | later: "Plus tard" | ||||||
| goToMisskey: "Retour vers Misskey" | goToMisskey: "Retour vers Misskey" | ||||||
| expirationDate: "Date d’expiration" | expirationDate: "Date d’expiration" | ||||||
| @@ -977,12 +994,26 @@ expired: "Expiré" | |||||||
| doYouAgree: "Êtes-vous d’accord ?" | doYouAgree: "Êtes-vous d’accord ?" | ||||||
| icon: "Avatar" | icon: "Avatar" | ||||||
| forYou: "Pour vous" | forYou: "Pour vous" | ||||||
|  | replies: "Répondre" | ||||||
|  | renotes: "Renoter" | ||||||
|  | loadReplies: "Inclure les réponses" | ||||||
|  | pinnedList: "Liste épinglée" | ||||||
|  | notifyNotes: "Notifier à propos des nouvelles notes" | ||||||
|  | authentication: "Authentification" | ||||||
|  | authenticationRequiredToContinue: "Veuillez vous authentifier pour continuer" | ||||||
| _announcement: | _announcement: | ||||||
|   readConfirmTitle: "Marquer comme lu ?" |   readConfirmTitle: "Marquer comme lu ?" | ||||||
| _initialAccountSetting: | _initialAccountSetting: | ||||||
|   profileSetting: "Paramètres du profil" |   profileSetting: "Paramètres du profil" | ||||||
|   privacySetting: "Paramètres de confidentialité" |   privacySetting: "Paramètres de confidentialité" | ||||||
|  |   initialAccountSettingCompleted: "Configuration du profil terminée avec succès !" | ||||||
|  |   ifYouNeedLearnMore: "Si vous voulez en savoir plus comment utiliser {name}(Misskey), veuillez visiter {link}." | ||||||
|  |   skipAreYouSure: "Désirez-vous ignorer la configuration du profile ?" | ||||||
|  | _serverSettings: | ||||||
|  |   iconUrl: "URL de l’icône" | ||||||
| _accountMigration: | _accountMigration: | ||||||
|  |   moveFrom: "Migrer un autre compte vers le présent compte" | ||||||
|  |   moveFromSub: "Créer un alias vers un autre compte" | ||||||
|   moveToLabel: "Compte vers lequel vous migrez :" |   moveToLabel: "Compte vers lequel vous migrez :" | ||||||
|   startMigration: "Migrer" |   startMigration: "Migrer" | ||||||
|   movedTo: "Compte vers lequel vous migrez :" |   movedTo: "Compte vers lequel vous migrez :" | ||||||
| @@ -1039,20 +1070,33 @@ _achievements: | |||||||
|     _login1000: |     _login1000: | ||||||
|       flavor: "Merci d'utiliser Misskey !" |       flavor: "Merci d'utiliser Misskey !" | ||||||
|     _profileFilled: |     _profileFilled: | ||||||
|  |       title: "Bien préparé" | ||||||
|       description: "Configuration de votre profil" |       description: "Configuration de votre profil" | ||||||
|     _markedAsCat: |     _markedAsCat: | ||||||
|       title: "Je suis un chat" |       title: "Je suis un chat" | ||||||
|  |       description: "Rendre votre compte comme un chat" | ||||||
|       flavor: "Je n'ai pas encore de nom" |       flavor: "Je n'ai pas encore de nom" | ||||||
|  |     _following1: | ||||||
|  |       title: "Vous suivez votre premier utilisateur·rice" | ||||||
|     _following50: |     _following50: | ||||||
|       title: "Beaucoup d'amis" |       title: "Beaucoup d'amis" | ||||||
|     _followers10: |     _followers10: | ||||||
|       title: "Abonnez-moi !" |       title: "Abonnez-moi !" | ||||||
|  |     _followers100: | ||||||
|  |       title: "Populaire" | ||||||
|  |     _followers500: | ||||||
|  |       title: "Tour radio" | ||||||
|  |     _followers1000: | ||||||
|  |       title: "Influenceur·euse" | ||||||
|     _iLoveMisskey: |     _iLoveMisskey: | ||||||
|       title: "J’adore Misskey" |       title: "J’adore Misskey" | ||||||
|       description: "Publication « J’❤ #Misskey »" |       description: "Publication « J’❤ #Misskey »" | ||||||
|  |       flavor: "L'équipe de développement de Misskey apprécie vraiment votre aide !" | ||||||
|     _foundTreasure: |     _foundTreasure: | ||||||
|       title: "Chasse au trésor" |       title: "Chasse au trésor" | ||||||
|       description: "Vous avez trouvé le trésor caché" |       description: "Vous avez trouvé le trésor caché" | ||||||
|  |     _client30min: | ||||||
|  |       title: "Pause bien méritée" | ||||||
|     _postedAtLateNight: |     _postedAtLateNight: | ||||||
|       flavor: "C’est l’heure d’aller au lit." |       flavor: "C’est l’heure d’aller au lit." | ||||||
|     _postedAt0min0sec: |     _postedAt0min0sec: | ||||||
| @@ -1061,18 +1105,45 @@ _achievements: | |||||||
|       flavor: "Tic tac, tic tac, tic tac, ding !" |       flavor: "Tic tac, tic tac, tic tac, ding !" | ||||||
|     _viewInstanceChart: |     _viewInstanceChart: | ||||||
|       title: "Analyste" |       title: "Analyste" | ||||||
|  |     _outputHelloWorldOnScratchpad: | ||||||
|  |       title: "Bonjour tout le monde !" | ||||||
|  |     _open3windows: | ||||||
|  |       title: "Multi-fenêtres" | ||||||
|  |     _driveFolderCircularReference: | ||||||
|  |       title: "Référence circulaire" | ||||||
|  |     _setNameToSyuilo: | ||||||
|  |       description: "Vous avez spécifié « syuilo » comme nom" | ||||||
|  |     _passedSinceAccountCreated1: | ||||||
|  |       title: "Premier anniversaire" | ||||||
|  |     _passedSinceAccountCreated2: | ||||||
|  |       title: "Second anniversaire" | ||||||
|  |     _passedSinceAccountCreated3: | ||||||
|  |       title: "3ème anniversaire" | ||||||
|     _loggedInOnBirthday: |     _loggedInOnBirthday: | ||||||
|       title: "Joyeux Anniversaire !" |       title: "Joyeux Anniversaire !" | ||||||
|  |       description: "Vous vous êtes connecté à la date de votre anniversaire" | ||||||
|     _loggedInOnNewYearsDay: |     _loggedInOnNewYearsDay: | ||||||
|       title: "Bonne année !" |       title: "Bonne année !" | ||||||
|     _cookieClicked: |     _cookieClicked: | ||||||
|       flavor: "Attendez une minute, vous êtes sur le mauvais site web ?" |       flavor: "Attendez une minute, vous êtes sur le mauvais site web ?" | ||||||
|  |     _brainDiver: | ||||||
|  |       flavor: "Misskey-Misskey La-Tu-Ma" | ||||||
| _role: | _role: | ||||||
|  |   new: "Nouveau rôle" | ||||||
|  |   edit: "Modifier le rôle" | ||||||
|   name: "Nom du rôle" |   name: "Nom du rôle" | ||||||
|   description: "Description du rôle" |   description: "Description du rôle" | ||||||
|   permission: "Rôle et autorisations" |   permission: "Rôle et autorisations" | ||||||
|   assignTarget: "Attribuer" |   assignTarget: "Attribuer" | ||||||
|   condition: "Condition" |   condition: "Condition" | ||||||
|  |   isPublic: "Rôle public" | ||||||
|  |   options: "Options" | ||||||
|  |   policies: "Stratégies" | ||||||
|  |   baseRole: "Modèle de rôle" | ||||||
|  |   useBaseValue: "Utiliser la valeur du modèle de rôle" | ||||||
|  |   chooseRoleToAssign: "Sélectionner le rôle à assigner" | ||||||
|  |   iconUrl: "URL de l’icône" | ||||||
|  |   displayOrder: "Classement" | ||||||
|   priority: "Priorité" |   priority: "Priorité" | ||||||
|   _priority: |   _priority: | ||||||
|     low: "Basse" |     low: "Basse" | ||||||
| @@ -1131,6 +1202,7 @@ _plugin: | |||||||
|   install: "Installation de plugin" |   install: "Installation de plugin" | ||||||
|   installWarn: "N’installez que des extensions provenant de sources de confiance." |   installWarn: "N’installez que des extensions provenant de sources de confiance." | ||||||
|   manage: "Gestion des plugins" |   manage: "Gestion des plugins" | ||||||
|  |   viewSource: "Afficher la source" | ||||||
| _preferencesBackups: | _preferencesBackups: | ||||||
|   list: "Sauvegardes créées" |   list: "Sauvegardes créées" | ||||||
|   saveNew: "Nouvelle sauvegarde" |   saveNew: "Nouvelle sauvegarde" | ||||||
| @@ -1317,6 +1389,7 @@ _2fa: | |||||||
|   securityKeyNotSupported: "Votre navigateur ne prend pas en charge les clés de sécurité." |   securityKeyNotSupported: "Votre navigateur ne prend pas en charge les clés de sécurité." | ||||||
|   securityKeyInfo: "Vous pouvez configurer l'authentification WebAuthN pour sécuriser davantage le processus de connexion grâce à une clé de sécurité matérielle qui prend en charge FIDO2, ou bien en configurant l'authentification par empreinte digitale ou par code PIN sur votre appareil." |   securityKeyInfo: "Vous pouvez configurer l'authentification WebAuthN pour sécuriser davantage le processus de connexion grâce à une clé de sécurité matérielle qui prend en charge FIDO2, ou bien en configurant l'authentification par empreinte digitale ou par code PIN sur votre appareil." | ||||||
|   securityKeyName: "Nom de la clé" |   securityKeyName: "Nom de la clé" | ||||||
|  |   removeKey: "Supprimer la clé de sécurité" | ||||||
|   removeKeyConfirm: "Voulez-vous supprimer {name} ?" |   removeKeyConfirm: "Voulez-vous supprimer {name} ?" | ||||||
|   renewTOTPOk: "Reconfigurer" |   renewTOTPOk: "Reconfigurer" | ||||||
|   renewTOTPCancel: "Pas maintenant" |   renewTOTPCancel: "Pas maintenant" | ||||||
| @@ -1618,3 +1691,6 @@ _deck: | |||||||
| _webhookSettings: | _webhookSettings: | ||||||
|   name: "Nom" |   name: "Nom" | ||||||
|   active: "Activé" |   active: "Activé" | ||||||
|  | _moderationLogTypes: | ||||||
|  |   suspend: "Suspendre" | ||||||
|  |   resetPassword: "Réinitialiser le mot de passe" | ||||||
|   | |||||||
| @@ -1,6 +1,11 @@ | |||||||
| import * as fs from 'node:fs'; | import * as fs from 'node:fs'; | ||||||
|  | import { fileURLToPath } from 'node:url'; | ||||||
|  | import { dirname } from 'node:path'; | ||||||
| import * as yaml from 'js-yaml'; | import * as yaml from 'js-yaml'; | ||||||
| import * as ts from 'typescript'; | import ts from 'typescript'; | ||||||
|  |  | ||||||
|  | const __filename = fileURLToPath(import.meta.url); | ||||||
|  | const __dirname = dirname(__filename); | ||||||
|  |  | ||||||
| function createMembers(record) { | function createMembers(record) { | ||||||
| 	return Object.entries(record) | 	return Object.entries(record) | ||||||
|   | |||||||
| @@ -77,6 +77,7 @@ smtpUser: "Felhasználónév" | |||||||
| smtpPass: "Jelszó" | smtpPass: "Jelszó" | ||||||
| user: "Felhasználók" | user: "Felhasználók" | ||||||
| searchByGoogle: "Keresés" | searchByGoogle: "Keresés" | ||||||
|  | renotes: "Renote" | ||||||
| _theme: | _theme: | ||||||
|   keys: |   keys: | ||||||
|     renote: "Renote" |     renote: "Renote" | ||||||
|   | |||||||
| @@ -354,7 +354,6 @@ invite: "Undang" | |||||||
| driveCapacityPerLocalAccount: "Kapasitas drive per pengguna lokal" | driveCapacityPerLocalAccount: "Kapasitas drive per pengguna lokal" | ||||||
| driveCapacityPerRemoteAccount: "Kapasitas drive per pengguna remote" | driveCapacityPerRemoteAccount: "Kapasitas drive per pengguna remote" | ||||||
| inMb: "dalam Megabytes" | inMb: "dalam Megabytes" | ||||||
| iconUrl: "URL Gambar ikon" |  | ||||||
| bannerUrl: "URL Banner" | bannerUrl: "URL Banner" | ||||||
| backgroundImageUrl: "URL Gambar latar" | backgroundImageUrl: "URL Gambar latar" | ||||||
| basicInfo: "Informasi Umum" | basicInfo: "Informasi Umum" | ||||||
| @@ -1019,7 +1018,6 @@ retryAllQueuesConfirmText: "Hal ini akan meningkatkan beban sementara ke peladen | |||||||
| enableChartsForRemoteUser: "Buat bagan data pengguna instansi luar" | enableChartsForRemoteUser: "Buat bagan data pengguna instansi luar" | ||||||
| enableChartsForFederatedInstances: "Buat bagan data peladen instansi luar" | enableChartsForFederatedInstances: "Buat bagan data peladen instansi luar" | ||||||
| showClipButtonInNoteFooter: "Tambahkan \"Klip\" ke menu aksi catatan" | showClipButtonInNoteFooter: "Tambahkan \"Klip\" ke menu aksi catatan" | ||||||
| largeNoteReactions: "Besarkan reaksi yang ditampilkan" |  | ||||||
| noteIdOrUrl: "ID catatan atau URL" | noteIdOrUrl: "ID catatan atau URL" | ||||||
| video: "Video" | video: "Video" | ||||||
| videos: "Video" | videos: "Video" | ||||||
| @@ -1100,6 +1098,9 @@ icon: "Avatar" | |||||||
| forYou: "Untuk Anda" | forYou: "Untuk Anda" | ||||||
| currentAnnouncements: "Pengumuman Saat Ini" | currentAnnouncements: "Pengumuman Saat Ini" | ||||||
| pastAnnouncements: "Pengumuman Terdahulu" | pastAnnouncements: "Pengumuman Terdahulu" | ||||||
|  | replies: "Balas" | ||||||
|  | renotes: "Renote" | ||||||
|  | dateAndTime: "Tanggal dan Waktu" | ||||||
| _initialAccountSetting: | _initialAccountSetting: | ||||||
|   accountCreated: "Akun kamu telah sukses dibuat!" |   accountCreated: "Akun kamu telah sukses dibuat!" | ||||||
|   letsStartAccountSetup: "Untuk pemula, ayo atur profilmu dulu." |   letsStartAccountSetup: "Untuk pemula, ayo atur profilmu dulu." | ||||||
| @@ -1117,6 +1118,8 @@ _initialAccountSetting: | |||||||
|   laterAreYouSure: "Yakin banget untuk atur profil nanti?" |   laterAreYouSure: "Yakin banget untuk atur profil nanti?" | ||||||
| _serverRules: | _serverRules: | ||||||
|   description: "Daftar peraturan akan ditampilkan sebelum pendaftaran. Mengatur ringkasan dari Syarat dan Ketentuan sangat direkomendasikan." |   description: "Daftar peraturan akan ditampilkan sebelum pendaftaran. Mengatur ringkasan dari Syarat dan Ketentuan sangat direkomendasikan." | ||||||
|  | _serverSettings: | ||||||
|  |   iconUrl: "URL ikon" | ||||||
| _accountMigration: | _accountMigration: | ||||||
|   moveFrom: "Pindahkan akun lain ke akun ini" |   moveFrom: "Pindahkan akun lain ke akun ini" | ||||||
|   moveFromSub: "Buat alias ke akun lain" |   moveFromSub: "Buat alias ke akun lain" | ||||||
| @@ -1494,6 +1497,7 @@ _plugin: | |||||||
|   install: "Memasang plugin" |   install: "Memasang plugin" | ||||||
|   installWarn: "Mohon jangan memasang plugin yang tidak dapat dipercayai." |   installWarn: "Mohon jangan memasang plugin yang tidak dapat dipercayai." | ||||||
|   manage: "Manajemen plugin" |   manage: "Manajemen plugin" | ||||||
|  |   viewSource: "Lihat sumber" | ||||||
| _preferencesBackups: | _preferencesBackups: | ||||||
|   list: "Cadangan yang dibuat" |   list: "Cadangan yang dibuat" | ||||||
|   saveNew: "Simpan cadangan baru" |   saveNew: "Simpan cadangan baru" | ||||||
| @@ -1681,7 +1685,6 @@ _timelineTutorial: | |||||||
| _2fa: | _2fa: | ||||||
|   alreadyRegistered: "Kamu telah mendaftarkan perangkat otentikasi dua faktor." |   alreadyRegistered: "Kamu telah mendaftarkan perangkat otentikasi dua faktor." | ||||||
|   registerTOTP: "Daftarkan aplikasi autentikator" |   registerTOTP: "Daftarkan aplikasi autentikator" | ||||||
|   passwordToTOTP: "Masukkan kata sandimu" |  | ||||||
|   step1: "Pertama, pasang aplikasi otentikasi (seperti {a} atau {b}) di perangkat kamu." |   step1: "Pertama, pasang aplikasi otentikasi (seperti {a} atau {b}) di perangkat kamu." | ||||||
|   step2: "Lalu, pindai kode QR yang ada di layar." |   step2: "Lalu, pindai kode QR yang ada di layar." | ||||||
|   step2Click: "Mengeklik kode QR ini akan membolehkanmu untuk mendaftarkan 2FA ke security-key atau aplikasi autentikator ponsel." |   step2Click: "Mengeklik kode QR ini akan membolehkanmu untuk mendaftarkan 2FA ke security-key atau aplikasi autentikator ponsel." | ||||||
| @@ -1691,7 +1694,6 @@ _2fa: | |||||||
|   securityKeyNotSupported: "Peramban kamu tidak mendukung security key." |   securityKeyNotSupported: "Peramban kamu tidak mendukung security key." | ||||||
|   registerTOTPBeforeKey: "Mohon atur aplikasi autentikator untuk mendaftarkan security key atau passkey." |   registerTOTPBeforeKey: "Mohon atur aplikasi autentikator untuk mendaftarkan security key atau passkey." | ||||||
|   securityKeyInfo: "Kamu dapat memasang otentikasi WebAuthN untuk mengamankan proses login lebih lanjut dengan tidak hanya perangkat keras kunci keamanan yang mendukung FIDO2, namun juga sidik jari atau otentikasi PIN pada perangkatmu." |   securityKeyInfo: "Kamu dapat memasang otentikasi WebAuthN untuk mengamankan proses login lebih lanjut dengan tidak hanya perangkat keras kunci keamanan yang mendukung FIDO2, namun juga sidik jari atau otentikasi PIN pada perangkatmu." | ||||||
|   chromePasskeyNotSupported: "Passkey Chrome saat ini tidak didukung." |  | ||||||
|   registerSecurityKey: "Daftarkan security key atau passkey." |   registerSecurityKey: "Daftarkan security key atau passkey." | ||||||
|   securityKeyName: "Masukkan nama key." |   securityKeyName: "Masukkan nama key." | ||||||
|   tapSecurityKey: "Mohon ikuti peramban kamu untuk mendaftarkan security key atau passkey" |   tapSecurityKey: "Mohon ikuti peramban kamu untuk mendaftarkan security key atau passkey" | ||||||
| @@ -2040,3 +2042,7 @@ _webhookSettings: | |||||||
|     renote: "Ketika direnote" |     renote: "Ketika direnote" | ||||||
|     reaction: "Ketika menerima reaksi" |     reaction: "Ketika menerima reaksi" | ||||||
|     mention: "Ketika sedang disebut" |     mention: "Ketika sedang disebut" | ||||||
|  | _moderationLogTypes: | ||||||
|  |   suspend: "Tangguhkan" | ||||||
|  |   resetPassword: "Atur ulang kata sandi" | ||||||
|  |   createInvitation: "Buat kode undangan" | ||||||
|   | |||||||
							
								
								
									
										91
									
								
								locales/index.d.ts
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										91
									
								
								locales/index.d.ts
									
									
									
									
										vendored
									
									
								
							| @@ -359,7 +359,6 @@ export interface Locale { | |||||||
|     "driveCapacityPerLocalAccount": string; |     "driveCapacityPerLocalAccount": string; | ||||||
|     "driveCapacityPerRemoteAccount": string; |     "driveCapacityPerRemoteAccount": string; | ||||||
|     "inMb": string; |     "inMb": string; | ||||||
|     "iconUrl": string; |  | ||||||
|     "bannerUrl": string; |     "bannerUrl": string; | ||||||
|     "backgroundImageUrl": string; |     "backgroundImageUrl": string; | ||||||
|     "basicInfo": string; |     "basicInfo": string; | ||||||
| @@ -420,6 +419,9 @@ export interface Locale { | |||||||
|     "totpDescription": string; |     "totpDescription": string; | ||||||
|     "moderator": string; |     "moderator": string; | ||||||
|     "moderation": string; |     "moderation": string; | ||||||
|  |     "moderationNote": string; | ||||||
|  |     "addModerationNote": string; | ||||||
|  |     "moderationLogs": string; | ||||||
|     "nUsersMentioned": string; |     "nUsersMentioned": string; | ||||||
|     "securityKeyAndPasskey": string; |     "securityKeyAndPasskey": string; | ||||||
|     "securityKey": string; |     "securityKey": string; | ||||||
| @@ -712,6 +714,7 @@ export interface Locale { | |||||||
|     "alwaysMarkSensitive": string; |     "alwaysMarkSensitive": string; | ||||||
|     "loadRawImages": string; |     "loadRawImages": string; | ||||||
|     "disableShowingAnimatedImages": string; |     "disableShowingAnimatedImages": string; | ||||||
|  |     "highlightSensitiveMedia": string; | ||||||
|     "verificationEmailSent": string; |     "verificationEmailSent": string; | ||||||
|     "notSet": string; |     "notSet": string; | ||||||
|     "emailVerified": string; |     "emailVerified": string; | ||||||
| @@ -1026,7 +1029,7 @@ export interface Locale { | |||||||
|     "enableChartsForRemoteUser": string; |     "enableChartsForRemoteUser": string; | ||||||
|     "enableChartsForFederatedInstances": string; |     "enableChartsForFederatedInstances": string; | ||||||
|     "showClipButtonInNoteFooter": string; |     "showClipButtonInNoteFooter": string; | ||||||
|     "largeNoteReactions": string; |     "reactionsDisplaySize": string; | ||||||
|     "noteIdOrUrl": string; |     "noteIdOrUrl": string; | ||||||
|     "video": string; |     "video": string; | ||||||
|     "videos": string; |     "videos": string; | ||||||
| @@ -1108,6 +1111,26 @@ export interface Locale { | |||||||
|     "currentAnnouncements": string; |     "currentAnnouncements": string; | ||||||
|     "pastAnnouncements": string; |     "pastAnnouncements": string; | ||||||
|     "youHaveUnreadAnnouncements": string; |     "youHaveUnreadAnnouncements": string; | ||||||
|  |     "useSecurityKey": string; | ||||||
|  |     "replies": string; | ||||||
|  |     "renotes": string; | ||||||
|  |     "loadReplies": string; | ||||||
|  |     "loadConversation": string; | ||||||
|  |     "pinnedList": string; | ||||||
|  |     "keepScreenOn": string; | ||||||
|  |     "verifiedLink": string; | ||||||
|  |     "notifyNotes": string; | ||||||
|  |     "unnotifyNotes": string; | ||||||
|  |     "authentication": string; | ||||||
|  |     "authenticationRequiredToContinue": string; | ||||||
|  |     "dateAndTime": string; | ||||||
|  |     "showRenotes": string; | ||||||
|  |     "edited": string; | ||||||
|  |     "notificationRecieveConfig": string; | ||||||
|  |     "mutualFollow": string; | ||||||
|  |     "fileAttachedOnly": string; | ||||||
|  |     "showRepliesToOthersInTimeline": string; | ||||||
|  |     "hideRepliesToOthersInTimeline": string; | ||||||
|     "_announcement": { |     "_announcement": { | ||||||
|         "forExistingUsers": string; |         "forExistingUsers": string; | ||||||
|         "forExistingUsersDescription": string; |         "forExistingUsersDescription": string; | ||||||
| @@ -1137,6 +1160,16 @@ export interface Locale { | |||||||
|     "_serverRules": { |     "_serverRules": { | ||||||
|         "description": string; |         "description": string; | ||||||
|     }; |     }; | ||||||
|  |     "_serverSettings": { | ||||||
|  |         "iconUrl": string; | ||||||
|  |         "appIconDescription": string; | ||||||
|  |         "appIconUsageExample": string; | ||||||
|  |         "appIconStyleRecommendation": string; | ||||||
|  |         "appIconResolutionMustBe": string; | ||||||
|  |         "manifestJsonOverride": string; | ||||||
|  |         "shortName": string; | ||||||
|  |         "shortNameDescription": string; | ||||||
|  |     }; | ||||||
|     "_accountMigration": { |     "_accountMigration": { | ||||||
|         "moveFrom": string; |         "moveFrom": string; | ||||||
|         "moveFromSub": string; |         "moveFromSub": string; | ||||||
| @@ -1466,6 +1499,10 @@ export interface Locale { | |||||||
|                 "description": string; |                 "description": string; | ||||||
|                 "flavor": string; |                 "flavor": string; | ||||||
|             }; |             }; | ||||||
|  |             "_smashTestNotificationButton": { | ||||||
|  |                 "title": string; | ||||||
|  |                 "description": string; | ||||||
|  |             }; | ||||||
|         }; |         }; | ||||||
|     }; |     }; | ||||||
|     "_role": { |     "_role": { | ||||||
| @@ -1507,6 +1544,7 @@ export interface Locale { | |||||||
|             "gtlAvailable": string; |             "gtlAvailable": string; | ||||||
|             "ltlAvailable": string; |             "ltlAvailable": string; | ||||||
|             "canPublicNote": string; |             "canPublicNote": string; | ||||||
|  |             "canEditNote": string; | ||||||
|             "canInvite": string; |             "canInvite": string; | ||||||
|             "inviteLimit": string; |             "inviteLimit": string; | ||||||
|             "inviteLimitCycle": string; |             "inviteLimitCycle": string; | ||||||
| @@ -1526,6 +1564,7 @@ export interface Locale { | |||||||
|             "descriptionOfRateLimitFactor": string; |             "descriptionOfRateLimitFactor": string; | ||||||
|             "canHideAds": string; |             "canHideAds": string; | ||||||
|             "canSearchNotes": string; |             "canSearchNotes": string; | ||||||
|  |             "canUseTranslator": string; | ||||||
|         }; |         }; | ||||||
|         "_condition": { |         "_condition": { | ||||||
|             "isLocal": string; |             "isLocal": string; | ||||||
| @@ -1606,6 +1645,7 @@ export interface Locale { | |||||||
|         "install": string; |         "install": string; | ||||||
|         "installWarn": string; |         "installWarn": string; | ||||||
|         "manage": string; |         "manage": string; | ||||||
|  |         "viewSource": string; | ||||||
|     }; |     }; | ||||||
|     "_preferencesBackups": { |     "_preferencesBackups": { | ||||||
|         "list": string; |         "list": string; | ||||||
| @@ -1681,11 +1721,6 @@ export interface Locale { | |||||||
|         "muteWords": string; |         "muteWords": string; | ||||||
|         "muteWordsDescription": string; |         "muteWordsDescription": string; | ||||||
|         "muteWordsDescription2": string; |         "muteWordsDescription2": string; | ||||||
|         "softDescription": string; |  | ||||||
|         "hardDescription": string; |  | ||||||
|         "soft": string; |  | ||||||
|         "hard": string; |  | ||||||
|         "mutedNotes": string; |  | ||||||
|     }; |     }; | ||||||
|     "_instanceMute": { |     "_instanceMute": { | ||||||
|         "instanceMuteDescription": string; |         "instanceMuteDescription": string; | ||||||
| @@ -1810,7 +1845,6 @@ export interface Locale { | |||||||
|     "_2fa": { |     "_2fa": { | ||||||
|         "alreadyRegistered": string; |         "alreadyRegistered": string; | ||||||
|         "registerTOTP": string; |         "registerTOTP": string; | ||||||
|         "passwordToTOTP": string; |  | ||||||
|         "step1": string; |         "step1": string; | ||||||
|         "step2": string; |         "step2": string; | ||||||
|         "step2Click": string; |         "step2Click": string; | ||||||
| @@ -1822,7 +1856,6 @@ export interface Locale { | |||||||
|         "securityKeyNotSupported": string; |         "securityKeyNotSupported": string; | ||||||
|         "registerTOTPBeforeKey": string; |         "registerTOTPBeforeKey": string; | ||||||
|         "securityKeyInfo": string; |         "securityKeyInfo": string; | ||||||
|         "chromePasskeyNotSupported": string; |  | ||||||
|         "registerSecurityKey": string; |         "registerSecurityKey": string; | ||||||
|         "securityKeyName": string; |         "securityKeyName": string; | ||||||
|         "tapSecurityKey": string; |         "tapSecurityKey": string; | ||||||
| @@ -1893,6 +1926,7 @@ export interface Locale { | |||||||
|         "homeTimeline": string; |         "homeTimeline": string; | ||||||
|         "users": string; |         "users": string; | ||||||
|         "userList": string; |         "userList": string; | ||||||
|  |         "userBlacklist": string; | ||||||
|     }; |     }; | ||||||
|     "_weekday": { |     "_weekday": { | ||||||
|         "sunday": string; |         "sunday": string; | ||||||
| @@ -2001,6 +2035,7 @@ export interface Locale { | |||||||
|         "metadataContent": string; |         "metadataContent": string; | ||||||
|         "changeAvatar": string; |         "changeAvatar": string; | ||||||
|         "changeBanner": string; |         "changeBanner": string; | ||||||
|  |         "verifiedLinkDescription": string; | ||||||
|     }; |     }; | ||||||
|     "_exportOrImport": { |     "_exportOrImport": { | ||||||
|         "allNotes": string; |         "allNotes": string; | ||||||
| @@ -2129,11 +2164,17 @@ export interface Locale { | |||||||
|         "youReceivedFollowRequest": string; |         "youReceivedFollowRequest": string; | ||||||
|         "yourFollowRequestAccepted": string; |         "yourFollowRequestAccepted": string; | ||||||
|         "pollEnded": string; |         "pollEnded": string; | ||||||
|  |         "newNote": string; | ||||||
|         "unreadAntennaNote": string; |         "unreadAntennaNote": string; | ||||||
|         "emptyPushNotificationMessage": string; |         "emptyPushNotificationMessage": string; | ||||||
|         "achievementEarned": string; |         "achievementEarned": string; | ||||||
|  |         "testNotification": string; | ||||||
|  |         "checkNotificationBehavior": string; | ||||||
|  |         "sendTestNotification": string; | ||||||
|  |         "notificationWillBeDisplayedLikeThis": string; | ||||||
|         "_types": { |         "_types": { | ||||||
|             "all": string; |             "all": string; | ||||||
|  |             "note": string; | ||||||
|             "follow": string; |             "follow": string; | ||||||
|             "mention": string; |             "mention": string; | ||||||
|             "reply": string; |             "reply": string; | ||||||
| @@ -2213,6 +2254,38 @@ export interface Locale { | |||||||
|             "mention": string; |             "mention": string; | ||||||
|         }; |         }; | ||||||
|     }; |     }; | ||||||
|  |     "_moderationLogTypes": { | ||||||
|  |         "createRole": string; | ||||||
|  |         "deleteRole": string; | ||||||
|  |         "updateRole": string; | ||||||
|  |         "assignRole": string; | ||||||
|  |         "unassignRole": string; | ||||||
|  |         "suspend": string; | ||||||
|  |         "unsuspend": string; | ||||||
|  |         "addCustomEmoji": string; | ||||||
|  |         "updateCustomEmoji": string; | ||||||
|  |         "deleteCustomEmoji": string; | ||||||
|  |         "updateServerSettings": string; | ||||||
|  |         "updateUserNote": string; | ||||||
|  |         "deleteDriveFile": string; | ||||||
|  |         "deleteNote": string; | ||||||
|  |         "createGlobalAnnouncement": string; | ||||||
|  |         "createUserAnnouncement": string; | ||||||
|  |         "updateGlobalAnnouncement": string; | ||||||
|  |         "updateUserAnnouncement": string; | ||||||
|  |         "deleteGlobalAnnouncement": string; | ||||||
|  |         "deleteUserAnnouncement": string; | ||||||
|  |         "resetPassword": string; | ||||||
|  |         "suspendRemoteInstance": string; | ||||||
|  |         "unsuspendRemoteInstance": string; | ||||||
|  |         "markSensitiveDriveFile": string; | ||||||
|  |         "unmarkSensitiveDriveFile": string; | ||||||
|  |         "resolveAbuseReport": string; | ||||||
|  |         "createInvitation": string; | ||||||
|  |         "createAd": string; | ||||||
|  |         "deleteAd": string; | ||||||
|  |         "updateAd": string; | ||||||
|  |     }; | ||||||
| } | } | ||||||
| declare const locales: { | declare const locales: { | ||||||
|     [lang: string]: Locale; |     [lang: string]: Locale; | ||||||
|   | |||||||
| @@ -106,7 +106,7 @@ unfollow: "Non seguire" | |||||||
| followRequestPending: "Richiesta in approvazione" | followRequestPending: "Richiesta in approvazione" | ||||||
| enterEmoji: "Inserisci emoji" | enterEmoji: "Inserisci emoji" | ||||||
| renote: "Rinota" | renote: "Rinota" | ||||||
| unrenote: "Annulla rinota" | unrenote: "Elimina la Rinota" | ||||||
| renoted: "Rinotato!" | renoted: "Rinotato!" | ||||||
| cantRenote: "È impossibile rinotare questa nota." | cantRenote: "È impossibile rinotare questa nota." | ||||||
| cantReRenote: "È impossibile rinotare una Rinota." | cantReRenote: "È impossibile rinotare una Rinota." | ||||||
| @@ -117,7 +117,7 @@ pinnedNote: "Nota fissata" | |||||||
| pinned: "Fissa sul profilo" | pinned: "Fissa sul profilo" | ||||||
| you: "Tu" | you: "Tu" | ||||||
| clickToShow: "Clicca per visualizzare" | clickToShow: "Clicca per visualizzare" | ||||||
| sensitive: "Contenuto sensibile" | sensitive: "Esplicito" | ||||||
| add: "Aggiungi" | add: "Aggiungi" | ||||||
| reaction: "Reazioni" | reaction: "Reazioni" | ||||||
| reactions: "Reazioni" | reactions: "Reazioni" | ||||||
| @@ -125,16 +125,16 @@ reactionSetting: "Reazioni visualizzate sul pannello" | |||||||
| reactionSettingDescription2: "Trascina per riorganizzare, clicca per cancellare, usa il pulsante \"+\" per aggiungere." | reactionSettingDescription2: "Trascina per riorganizzare, clicca per cancellare, usa il pulsante \"+\" per aggiungere." | ||||||
| rememberNoteVisibility: "Ricordare le impostazioni di visibilità delle note" | rememberNoteVisibility: "Ricordare le impostazioni di visibilità delle note" | ||||||
| attachCancel: "Rimuovi allegato" | attachCancel: "Rimuovi allegato" | ||||||
| markAsSensitive: "Segna come sensibile" | markAsSensitive: "Segna come esplicito" | ||||||
| unmarkAsSensitive: "Segna come non sensibile" | unmarkAsSensitive: "Non segnare come esplicito " | ||||||
| enterFileName: "Nome del file" | enterFileName: "Nome del file" | ||||||
| mute: "Silenzia" | mute: "Silenzia" | ||||||
| unmute: "Riattiva l'audio" | unmute: "Riattiva l'audio" | ||||||
| renoteMute: "Silenzia i Rinota" | renoteMute: "Silenzia le Rinota" | ||||||
| renoteUnmute: "Non silenziare i Rinota" | renoteUnmute: "Non silenziare le Rinota" | ||||||
| block: "Blocca" | block: "Blocca" | ||||||
| unblock: "Sblocca" | unblock: "Sblocca" | ||||||
| suspend: "Sospendi" | suspend: "Sospensione" | ||||||
| unsuspend: "Revoca la sospensione" | unsuspend: "Revoca la sospensione" | ||||||
| blockConfirm: "Vuoi davvero bloccare il profilo?" | blockConfirm: "Vuoi davvero bloccare il profilo?" | ||||||
| unblockConfirm: "Vuoi davvero sbloccare il profilo?" | unblockConfirm: "Vuoi davvero sbloccare il profilo?" | ||||||
| @@ -148,7 +148,7 @@ editAntenna: "Modifica Antenna" | |||||||
| selectWidget: "Seleziona il riquadro" | selectWidget: "Seleziona il riquadro" | ||||||
| editWidgets: "Modifica i riquadri" | editWidgets: "Modifica i riquadri" | ||||||
| editWidgetsExit: "Conferma le modifiche" | editWidgetsExit: "Conferma le modifiche" | ||||||
| customEmojis: "Emoji personalizzati" | customEmojis: "Emoji personalizzate" | ||||||
| emoji: "Emoji" | emoji: "Emoji" | ||||||
| emojis: "Emoji" | emojis: "Emoji" | ||||||
| emojiName: "Nome dell'emoji" | emojiName: "Nome dell'emoji" | ||||||
| @@ -158,8 +158,8 @@ settingGuide: "Configurazione suggerita" | |||||||
| cacheRemoteFiles: "Memorizza i file remoti nella cache" | cacheRemoteFiles: "Memorizza i file remoti nella cache" | ||||||
| cacheRemoteFilesDescription: "Disabilitando questa opzione, i file remoti verranno linkati direttamente senza essere memorizzati nella cache. Sarà possibile risparmiare spazio di archiviazione sul server, ma il traffico aumenterà in quanto non verranno generate anteprime." | cacheRemoteFilesDescription: "Disabilitando questa opzione, i file remoti verranno linkati direttamente senza essere memorizzati nella cache. Sarà possibile risparmiare spazio di archiviazione sul server, ma il traffico aumenterà in quanto non verranno generate anteprime." | ||||||
| youCanCleanRemoteFilesCache: "Puoi svuotare tutta la cache cliccando il bottone 🗑️ nella gestione file" | youCanCleanRemoteFilesCache: "Puoi svuotare tutta la cache cliccando il bottone 🗑️ nella gestione file" | ||||||
| cacheRemoteSensitiveFiles: "Memorizza nella cache i file sensibili remoti" | cacheRemoteSensitiveFiles: "Copia nella cache locale i file espliciti remoti" | ||||||
| cacheRemoteSensitiveFilesDescription: "Disattivando questa opzione, i file sensibili verranno caricati direttamente dall'istanza remota senza essere salvati dal server." | cacheRemoteSensitiveFilesDescription: "Disattivando questa opzione, i file espliciti verranno richiesti direttamente all'istanza remota senza essere salvati nel server locale." | ||||||
| flagAsBot: "Io sono un robot" | flagAsBot: "Io sono un robot" | ||||||
| flagAsBotDescription: "Attiva questo campo se il profilo esegue principalmente operazioni automatiche. L'attivazione segnala agli altri sviluppatori come comportarsi per evitare catene d’interazione infinite con altri bot. I sistemi interni di Misskey si adegueranno al fine di trattare questo profilo come bot." | flagAsBotDescription: "Attiva questo campo se il profilo esegue principalmente operazioni automatiche. L'attivazione segnala agli altri sviluppatori come comportarsi per evitare catene d’interazione infinite con altri bot. I sistemi interni di Misskey si adegueranno al fine di trattare questo profilo come bot." | ||||||
| flagAsCat: "Sono un gatto" | flagAsCat: "Sono un gatto" | ||||||
| @@ -180,7 +180,7 @@ youHaveNoLists: "Non hai ancora creato nessuna lista" | |||||||
| followConfirm: "Vuoi seguire {name}?" | followConfirm: "Vuoi seguire {name}?" | ||||||
| proxyAccount: "Profilo proxy" | proxyAccount: "Profilo proxy" | ||||||
| proxyAccountDescription: "Un profilo proxy funziona come follower per i profili remoti, sotto certe condizioni. Ad esempio, quando un profilo locale ne inserisce uno remoto in una lista (senza seguirlo), se nessun altro segue quel profilo remoto, le attività non possono essere distribuite. Dunque, il profilo proxy le seguirà per tutti." | proxyAccountDescription: "Un profilo proxy funziona come follower per i profili remoti, sotto certe condizioni. Ad esempio, quando un profilo locale ne inserisce uno remoto in una lista (senza seguirlo), se nessun altro segue quel profilo remoto, le attività non possono essere distribuite. Dunque, il profilo proxy le seguirà per tutti." | ||||||
| host: "Server remoto" | host: "Host" | ||||||
| selectUser: "Seleziona profilo" | selectUser: "Seleziona profilo" | ||||||
| recipient: "Destinatario" | recipient: "Destinatario" | ||||||
| annotation: "Annotazione preventiva" | annotation: "Annotazione preventiva" | ||||||
| @@ -287,7 +287,7 @@ images: "Immagini" | |||||||
| image: "Immagini" | image: "Immagini" | ||||||
| birthday: "Compleanno" | birthday: "Compleanno" | ||||||
| yearsOld: "{age} anni" | yearsOld: "{age} anni" | ||||||
| registeredDate: "Iscrizione a.." | registeredDate: "Data iscrizione" | ||||||
| location: "Posizione" | location: "Posizione" | ||||||
| theme: "Tema" | theme: "Tema" | ||||||
| themeForLightMode: "Tema da utilizzare per il modo chiaro" | themeForLightMode: "Tema da utilizzare per il modo chiaro" | ||||||
| @@ -321,7 +321,7 @@ copyUrl: "Copia URL" | |||||||
| rename: "Modifica nome" | rename: "Modifica nome" | ||||||
| avatar: "Foto del profilo" | avatar: "Foto del profilo" | ||||||
| banner: "Intestazione" | banner: "Intestazione" | ||||||
| displayOfSensitiveMedia: "Visibilità dei media sensibili" | displayOfSensitiveMedia: "Visibilità dei media espliciti" | ||||||
| whenServerDisconnected: "Quando la connessione col server è persa" | whenServerDisconnected: "Quando la connessione col server è persa" | ||||||
| disconnectedFromServer: "Il server si è disconnesso" | disconnectedFromServer: "Il server si è disconnesso" | ||||||
| reload: "Ricarica" | reload: "Ricarica" | ||||||
| @@ -356,7 +356,6 @@ invite: "Invita" | |||||||
| driveCapacityPerLocalAccount: "Capienza del Drive per profilo locale" | driveCapacityPerLocalAccount: "Capienza del Drive per profilo locale" | ||||||
| driveCapacityPerRemoteAccount: "Capienza del Drive per profilo remoto" | driveCapacityPerRemoteAccount: "Capienza del Drive per profilo remoto" | ||||||
| inMb: "in Megabytes" | inMb: "in Megabytes" | ||||||
| iconUrl: "URL di icona (favicon, ecc.)" |  | ||||||
| bannerUrl: "URL dell'immagine d'intestazione" | bannerUrl: "URL dell'immagine d'intestazione" | ||||||
| backgroundImageUrl: "URL dello sfondo" | backgroundImageUrl: "URL dello sfondo" | ||||||
| basicInfo: "Informazioni fondamentali" | basicInfo: "Informazioni fondamentali" | ||||||
| @@ -417,6 +416,9 @@ totp: "App di autenticazione" | |||||||
| totpDescription: "Inserisci un codice OTP tramite un'app di autenticazione" | totpDescription: "Inserisci un codice OTP tramite un'app di autenticazione" | ||||||
| moderator: "Moderatore" | moderator: "Moderatore" | ||||||
| moderation: "moderazione" | moderation: "moderazione" | ||||||
|  | moderationNote: "Promemoria di moderazione" | ||||||
|  | addModerationNote: "Aggiungi promemoria di moderazione" | ||||||
|  | moderationLogs: "Cronologia di moderazione" | ||||||
| nUsersMentioned: "{n} profili menzionati" | nUsersMentioned: "{n} profili menzionati" | ||||||
| securityKeyAndPasskey: "Chiave di sicurezza e accesso" | securityKeyAndPasskey: "Chiave di sicurezza e accesso" | ||||||
| securityKey: "Chiave di sicurezza" | securityKey: "Chiave di sicurezza" | ||||||
| @@ -495,7 +497,7 @@ noFollowRequests: "Non hai alcuna richiesta di follow" | |||||||
| openImageInNewTab: "Apri le immagini in un nuovo tab" | openImageInNewTab: "Apri le immagini in un nuovo tab" | ||||||
| dashboard: "Pannello di controllo" | dashboard: "Pannello di controllo" | ||||||
| local: "Locale" | local: "Locale" | ||||||
| remote: "Remoto" | remote: "Remota" | ||||||
| total: "Totale" | total: "Totale" | ||||||
| weekOverWeekChanges: "Settimanale" | weekOverWeekChanges: "Settimanale" | ||||||
| dayOverDayChanges: "Giornaliero" | dayOverDayChanges: "Giornaliero" | ||||||
| @@ -550,8 +552,8 @@ installedDate: "Data installazione" | |||||||
| lastUsedDate: "Data di ultimo uso" | lastUsedDate: "Data di ultimo uso" | ||||||
| state: "Stato" | state: "Stato" | ||||||
| sort: "Ordina per" | sort: "Ordina per" | ||||||
| ascendingOrder: "Ascendente" | ascendingOrder: "Aumenta" | ||||||
| descendingOrder: "Discendente" | descendingOrder: "Diminuisce" | ||||||
| scratchpad: "ScratchPad" | scratchpad: "ScratchPad" | ||||||
| scratchpadDescription: "Lo Scratchpad offre un ambiente per esperimenti di AiScript. È possibile scrivere, eseguire e confermare i risultati dell'interazione del codice con Misskey." | scratchpadDescription: "Lo Scratchpad offre un ambiente per esperimenti di AiScript. È possibile scrivere, eseguire e confermare i risultati dell'interazione del codice con Misskey." | ||||||
| output: "Uscita" | output: "Uscita" | ||||||
| @@ -620,7 +622,7 @@ emailConfigInfo: "Utilizzato per verificare il tuo indirizzo di posta elettronic | |||||||
| email: "Email" | email: "Email" | ||||||
| emailAddress: "Indirizzo di posta elettronica" | emailAddress: "Indirizzo di posta elettronica" | ||||||
| smtpConfig: "Impostazioni del server SMTP" | smtpConfig: "Impostazioni del server SMTP" | ||||||
| smtpHost: "Server remoto" | smtpHost: "Host SMTP" | ||||||
| smtpPort: "Porta" | smtpPort: "Porta" | ||||||
| smtpUser: "Nome utente" | smtpUser: "Nome utente" | ||||||
| smtpPass: "Password" | smtpPass: "Password" | ||||||
| @@ -706,9 +708,10 @@ driveUsage: "Utilizzazione del Drive" | |||||||
| noCrawle: "Rifiuta l'indicizzazione dai robot." | noCrawle: "Rifiuta l'indicizzazione dai robot." | ||||||
| noCrawleDescription: "Richiedi che i motori di ricerca non indicizzino la tua pagina di profilo, le tue note, pagine, ecc." | noCrawleDescription: "Richiedi che i motori di ricerca non indicizzino la tua pagina di profilo, le tue note, pagine, ecc." | ||||||
| lockedAccountInfo: "A meno che non imposti la visibilità delle tue note su \"Solo ai follower\", le tue note sono visibili da tutti, anche se hai configurato l'account per confermare manualmente le richieste di follow." | lockedAccountInfo: "A meno che non imposti la visibilità delle tue note su \"Solo ai follower\", le tue note sono visibili da tutti, anche se hai configurato l'account per confermare manualmente le richieste di follow." | ||||||
| alwaysMarkSensitive: "Segnare i media come sensibili per impostazione predefinita" | alwaysMarkSensitive: "Segnare gli allegati come espliciti come opzione predefinita" | ||||||
| loadRawImages: "Visualizza le intere immagini allegate invece delle miniature." | loadRawImages: "Visualizza le intere immagini allegate invece delle miniature." | ||||||
| disableShowingAnimatedImages: "Disabilita le immagini animate" | disableShowingAnimatedImages: "Disabilita le immagini animate" | ||||||
|  | highlightSensitiveMedia: "Evidenzia i media espliciti" | ||||||
| verificationEmailSent: "Una mail di verifica è stata inviata. Si prega di accedere al collegamento per compiere la verifica." | verificationEmailSent: "Una mail di verifica è stata inviata. Si prega di accedere al collegamento per compiere la verifica." | ||||||
| notSet: "Non impostato" | notSet: "Non impostato" | ||||||
| emailVerified: "Il tuo indirizzo email è stato verificato" | emailVerified: "Il tuo indirizzo email è stato verificato" | ||||||
| @@ -925,7 +928,7 @@ type: "Tipo" | |||||||
| speed: "Velocità" | speed: "Velocità" | ||||||
| slow: "Lento" | slow: "Lento" | ||||||
| fast: "Veloce" | fast: "Veloce" | ||||||
| sensitiveMediaDetection: "Rilevamento dei contenuti sensibili." | sensitiveMediaDetection: "Rilevamento dei contenuti espliciti" | ||||||
| localOnly: "Soltanto locale" | localOnly: "Soltanto locale" | ||||||
| remoteOnly: "Solo remoto" | remoteOnly: "Solo remoto" | ||||||
| failedToUpload: "errore di caricamento" | failedToUpload: "errore di caricamento" | ||||||
| @@ -988,7 +991,7 @@ thisPostMayBeAnnoying: "Questa nota potrebbe essere offensiva" | |||||||
| thisPostMayBeAnnoyingHome: "Pubblica sulla timeline principale" | thisPostMayBeAnnoyingHome: "Pubblica sulla timeline principale" | ||||||
| thisPostMayBeAnnoyingCancel: "Annulla" | thisPostMayBeAnnoyingCancel: "Annulla" | ||||||
| thisPostMayBeAnnoyingIgnore: "Pubblica lo stesso" | thisPostMayBeAnnoyingIgnore: "Pubblica lo stesso" | ||||||
| collapseRenotes: "Comprimi i Rinota già letti" | collapseRenotes: "Comprimi le Rinota già viste" | ||||||
| internalServerError: "Errore interno del server" | internalServerError: "Errore interno del server" | ||||||
| internalServerErrorDescription: "Si è verificato un errore imprevisto all'interno del server" | internalServerErrorDescription: "Si è verificato un errore imprevisto all'interno del server" | ||||||
| copyErrorInfo: "Copia le informazioni sull'errore" | copyErrorInfo: "Copia le informazioni sull'errore" | ||||||
| @@ -1005,11 +1008,11 @@ cannotBeChangedLater: "Non sarà più modificabile" | |||||||
| reactionAcceptance: "Reazioni consentite" | reactionAcceptance: "Reazioni consentite" | ||||||
| likeOnly: "Solo i Like" | likeOnly: "Solo i Like" | ||||||
| likeOnlyForRemote: "Solo Like remoti" | likeOnlyForRemote: "Solo Like remoti" | ||||||
| nonSensitiveOnly: "Solamente non sensibili" | nonSensitiveOnly: "Soltanto non espliciti" | ||||||
| nonSensitiveOnlyForLocalLikeOnlyForRemote: "Solamente non sensibili (solo Mi piace remoti)" | nonSensitiveOnlyForLocalLikeOnlyForRemote: "Soltanto non espliciti (reazioni remote)" | ||||||
| rolesAssignedToMe: "I miei ruoli" | rolesAssignedToMe: "I miei ruoli" | ||||||
| resetPasswordConfirm: "Vuoi davvero ripristinare la password?" | resetPasswordConfirm: "Vuoi davvero ripristinare la password?" | ||||||
| sensitiveWords: "Parole sensibili" | sensitiveWords: "Parole esplicite" | ||||||
| sensitiveWordsDescription: "Imposta automaticamente \"Home\" alla visibilità delle Note che contengono una qualsiasi parola tra queste configurate. Puoi separarle per riga." | sensitiveWordsDescription: "Imposta automaticamente \"Home\" alla visibilità delle Note che contengono una qualsiasi parola tra queste configurate. Puoi separarle per riga." | ||||||
| sensitiveWordsDescription2: "Gli spazi creano la relazione \"E\" tra parole (questo E quello). Racchiudere una parola nelle slash \"/\" la trasforma in Espressione Regolare." | sensitiveWordsDescription2: "Gli spazi creano la relazione \"E\" tra parole (questo E quello). Racchiudere una parola nelle slash \"/\" la trasforma in Espressione Regolare." | ||||||
| notesSearchNotAvailable: "Non è possibile cercare tra le Note." | notesSearchNotAvailable: "Non è possibile cercare tra le Note." | ||||||
| @@ -1023,7 +1026,7 @@ retryAllQueuesConfirmText: "Potrebbe sovraccaricare il server temporaneamente." | |||||||
| enableChartsForRemoteUser: "Abilita i grafici per i profili remoti" | enableChartsForRemoteUser: "Abilita i grafici per i profili remoti" | ||||||
| enableChartsForFederatedInstances: "Abilita i grafici per le istanze federate" | enableChartsForFederatedInstances: "Abilita i grafici per le istanze federate" | ||||||
| showClipButtonInNoteFooter: "Aggiungi il bottone Clip tra le azioni delle Note" | showClipButtonInNoteFooter: "Aggiungi il bottone Clip tra le azioni delle Note" | ||||||
| largeNoteReactions: "Ingrandisci le reazioni" | reactionsDisplaySize: "Grandezza delle reazioni" | ||||||
| noteIdOrUrl: "ID della Nota o URL" | noteIdOrUrl: "ID della Nota o URL" | ||||||
| video: "Video" | video: "Video" | ||||||
| videos: "Video" | videos: "Video" | ||||||
| @@ -1105,6 +1108,20 @@ forYou: "Per te" | |||||||
| currentAnnouncements: "Annunci attuali" | currentAnnouncements: "Annunci attuali" | ||||||
| pastAnnouncements: "Annunci precedenti" | pastAnnouncements: "Annunci precedenti" | ||||||
| youHaveUnreadAnnouncements: "Ci sono Annunci non letti" | youHaveUnreadAnnouncements: "Ci sono Annunci non letti" | ||||||
|  | useSecurityKey: "Per utilizzare la chiave di sicurezza o la passkey, segui le indicazioni del dispositivo" | ||||||
|  | replies: "Rispondi" | ||||||
|  | renotes: "Rinota" | ||||||
|  | loadReplies: "Leggi le risposte" | ||||||
|  | loadConversation: "Leggi la conversazione" | ||||||
|  | pinnedList: "Elenco in primo piano" | ||||||
|  | keepScreenOn: "Mantieni lo schermo acceso" | ||||||
|  | verifiedLink: "Abbiamo confermato la validità di questo collegamento" | ||||||
|  | notifyNotes: "Notifica nuove Note" | ||||||
|  | unnotifyNotes: "Interrompi le notifiche di nuove Note" | ||||||
|  | authentication: "Autenticazione" | ||||||
|  | authenticationRequiredToContinue: "Per procedere, è richiesta l'autenticazione" | ||||||
|  | dateAndTime: "Data e Ora" | ||||||
|  | showRenotes: "Leggi le Rinota" | ||||||
| _announcement: | _announcement: | ||||||
|   forExistingUsers: "Solo ai profili attuali" |   forExistingUsers: "Solo ai profili attuali" | ||||||
|   forExistingUsersDescription: "L'annuncio sarà visibile solo ai profili esistenti in questo momento. Se disabilitato, sarà visibile anche ai profili che verranno creati dopo la pubblicazione di questo annuncio." |   forExistingUsersDescription: "L'annuncio sarà visibile solo ai profili esistenti in questo momento. Se disabilitato, sarà visibile anche ai profili che verranno creati dopo la pubblicazione di questo annuncio." | ||||||
| @@ -1131,6 +1148,15 @@ _initialAccountSetting: | |||||||
|   laterAreYouSure: "Vuoi davvero rimandare la configurazione iniziale?" |   laterAreYouSure: "Vuoi davvero rimandare la configurazione iniziale?" | ||||||
| _serverRules: | _serverRules: | ||||||
|   description: "In Europa è necessario mostrare l'informativa sul trattamento dei dati personali, prima della registrazione al servizio." |   description: "In Europa è necessario mostrare l'informativa sul trattamento dei dati personali, prima della registrazione al servizio." | ||||||
|  | _serverSettings: | ||||||
|  |   iconUrl: "URL dell'icona" | ||||||
|  |   appIconDescription: "Indicare l'icona da usare quando {host} viene salvata come App." | ||||||
|  |   appIconUsageExample: "Ad esempio quando si aggiunge il segnalibro alla PWA (Progressive Web App), oppure alla schermata iniziale del dispositivo mobile " | ||||||
|  |   appIconStyleRecommendation: "Poiché l'icona potrebbe essere ritagliata in un quadrato o in un cerchio, si raccomanda che abbia un margine colorato." | ||||||
|  |   appIconResolutionMustBe: "La risoluzione minima è {resolution}" | ||||||
|  |   manifestJsonOverride: "Sostituire il file manifest.json" | ||||||
|  |   shortName: "Abbreviazione" | ||||||
|  |   shortNameDescription: "Un'abbreviazione o un nome comune che può essere visualizzato al posto del nome ufficiale lungo del server." | ||||||
| _accountMigration: | _accountMigration: | ||||||
|   moveFrom: "Migra un altro profilo dentro a questo" |   moveFrom: "Migra un altro profilo dentro a questo" | ||||||
|   moveFromSub: "Crea un alias verso un altro profilo remoto" |   moveFromSub: "Crea un alias verso un altro profilo remoto" | ||||||
| @@ -1385,6 +1411,9 @@ _achievements: | |||||||
|       title: "Brain Diver" |       title: "Brain Diver" | ||||||
|       description: "Pubblica un link a Brain Diver" |       description: "Pubblica un link a Brain Diver" | ||||||
|       flavor: "Sulle note di Brain Diver" |       flavor: "Sulle note di Brain Diver" | ||||||
|  |     _smashTestNotificationButton: | ||||||
|  |       title: "Prove eccessive" | ||||||
|  |       description: "Hai provato le notifiche consecutivamente in un periodo di tempo molto breve" | ||||||
| _role: | _role: | ||||||
|   new: "Nuovo ruolo" |   new: "Nuovo ruolo" | ||||||
|   edit: "Modifica ruolo" |   edit: "Modifica ruolo" | ||||||
| @@ -1445,10 +1474,10 @@ _role: | |||||||
|   _condition: |   _condition: | ||||||
|     isLocal: "Profilo locale" |     isLocal: "Profilo locale" | ||||||
|     isRemote: "Profilo remoto" |     isRemote: "Profilo remoto" | ||||||
|     createdLessThan: "Creato meno di" |     createdLessThan: "Profilo creato da meno di N" | ||||||
|     createdMoreThan: "Creato più di" |     createdMoreThan: "Profilo creato da più di N" | ||||||
|     followersLessThanOrEq: "Ha meno di N follower" |     followersLessThanOrEq: "Profilo con N follower o meno" | ||||||
|     followersMoreThanOrEq: "Ha più di N follower" |     followersMoreThanOrEq: "Profilo con N follower o più" | ||||||
|     followingLessThanOrEq: "Segue N profili o meno" |     followingLessThanOrEq: "Segue N profili o meno" | ||||||
|     followingMoreThanOrEq: "Segue N profili o più" |     followingMoreThanOrEq: "Segue N profili o più" | ||||||
|     notesLessThanOrEq: "Conteggio Note inferiore o uguale a" |     notesLessThanOrEq: "Conteggio Note inferiore o uguale a" | ||||||
| @@ -1457,9 +1486,9 @@ _role: | |||||||
|     or: "O" |     or: "O" | ||||||
|     not: "NON" |     not: "NON" | ||||||
| _sensitiveMediaDetection: | _sensitiveMediaDetection: | ||||||
|   description: "L'apprendimento automatico può essere utilizzato per individuare automaticamente i media sensibili da moderare. Il carico del server aumenta leggermente." |   description: "Utilizzare l'apprendimento automatico (machine learning) per riconoscere media espliciti e sottoporli alla moderazione. Aumenterà lievemente il carico del server." | ||||||
|   sensitivity: "Sensibilità di rilevamento" |   sensitivity: "Sensibilità del rilevamento" | ||||||
|   sensitivityDescription: "Una minore sensibilità riduce i falsi positivi (false positività). Una maggiore sensibilità riduce le omissioni (falsi negativi)." |   sensitivityDescription: "Abbassando la sensibilità si riducono i falsi positivi (rilevazioni errate). Aumentando la sensibilità si riduce il numero di rilevazioni mancate. (rilevazioni ignorate)." | ||||||
|   setSensitiveFlagAutomatically: "Impostare il flag NSFW." |   setSensitiveFlagAutomatically: "Impostare il flag NSFW." | ||||||
|   setSensitiveFlagAutomaticallyDescription: "Anche se questa impostazione è disattivata, il risultato della decisione viene conservato internamente." |   setSensitiveFlagAutomaticallyDescription: "Anche se questa impostazione è disattivata, il risultato della decisione viene conservato internamente." | ||||||
|   analyzeVideos: "Abilitazione dell'analisi video." |   analyzeVideos: "Abilitazione dell'analisi video." | ||||||
| @@ -1508,6 +1537,7 @@ _plugin: | |||||||
|   install: "Installa estensioni" |   install: "Installa estensioni" | ||||||
|   installWarn: "Si prega di installare soltanto estensioni che provengono da fonti affidabili." |   installWarn: "Si prega di installare soltanto estensioni che provengono da fonti affidabili." | ||||||
|   manage: "Gestisci estensioni" |   manage: "Gestisci estensioni" | ||||||
|  |   viewSource: "Visualizza sorgente" | ||||||
| _preferencesBackups: | _preferencesBackups: | ||||||
|   list: "Elenco di impostazioni salvate in precedenza" |   list: "Elenco di impostazioni salvate in precedenza" | ||||||
|   saveNew: "Nuovo salvataggio" |   saveNew: "Nuovo salvataggio" | ||||||
| @@ -1542,8 +1572,8 @@ _aboutMisskey: | |||||||
|   morePatrons: "Apprezziamo sinceramente il supporto di tante altre persone. Grazie mille! 🥰" |   morePatrons: "Apprezziamo sinceramente il supporto di tante altre persone. Grazie mille! 🥰" | ||||||
|   patrons: "Sostenitori" |   patrons: "Sostenitori" | ||||||
| _displayOfSensitiveMedia: | _displayOfSensitiveMedia: | ||||||
|   respect: "Nascondere i media sensibili" |   respect: "Nascondere i media espliciti" | ||||||
|   ignore: "Non nascondere i media sensibili" |   ignore: "Non nascondere i media espliciti" | ||||||
|   force: "Nascondi tutti i media" |   force: "Nascondi tutti i media" | ||||||
| _instanceTicker: | _instanceTicker: | ||||||
|   none: "Nascondi" |   none: "Nascondi" | ||||||
| @@ -1695,7 +1725,6 @@ _timelineTutorial: | |||||||
| _2fa: | _2fa: | ||||||
|   alreadyRegistered: "La configurazione è stata già completata." |   alreadyRegistered: "La configurazione è stata già completata." | ||||||
|   registerTOTP: "Registra un'app di autenticazione" |   registerTOTP: "Registra un'app di autenticazione" | ||||||
|   passwordToTOTP: "Inserire la password" |  | ||||||
|   step1: "Innanzitutto, installare sul dispositivo un'applicazione di autenticazione come {a} o {b}." |   step1: "Innanzitutto, installare sul dispositivo un'applicazione di autenticazione come {a} o {b}." | ||||||
|   step2: "Quindi, scansionare il codice QR visualizzato con l'app." |   step2: "Quindi, scansionare il codice QR visualizzato con l'app." | ||||||
|   step2Click: "Cliccando sul codice QR, puoi registrarlo con l'app di autenticazione o il portachiavi installato sul tuo dispositivo." |   step2Click: "Cliccando sul codice QR, puoi registrarlo con l'app di autenticazione o il portachiavi installato sul tuo dispositivo." | ||||||
| @@ -1707,7 +1736,6 @@ _2fa: | |||||||
|   securityKeyNotSupported: "Il tuo browser non supporta le chiavi di sicurezza." |   securityKeyNotSupported: "Il tuo browser non supporta le chiavi di sicurezza." | ||||||
|   registerTOTPBeforeKey: "Ti occorre un'app di autenticazione con OTP, prima di registrare la chiave di sicurezza." |   registerTOTPBeforeKey: "Ti occorre un'app di autenticazione con OTP, prima di registrare la chiave di sicurezza." | ||||||
|   securityKeyInfo: "È possibile impostare il dispositivo per accedere utilizzando una chiave di sicurezza hardware che supporta FIDO2 o un'impronta digitale o un PIN sul dispositivo." |   securityKeyInfo: "È possibile impostare il dispositivo per accedere utilizzando una chiave di sicurezza hardware che supporta FIDO2 o un'impronta digitale o un PIN sul dispositivo." | ||||||
|   chromePasskeyNotSupported: "Le passkey di Chrome non sono attualmente supportate." |  | ||||||
|   registerSecurityKey: "Registra la chiave di sicurezza" |   registerSecurityKey: "Registra la chiave di sicurezza" | ||||||
|   securityKeyName: "Inserisci il nome della chiave" |   securityKeyName: "Inserisci il nome della chiave" | ||||||
|   tapSecurityKey: "Segui le istruzioni del browser e registra la chiave di sicurezza." |   tapSecurityKey: "Segui le istruzioni del browser e registra la chiave di sicurezza." | ||||||
| @@ -1775,6 +1803,7 @@ _antennaSources: | |||||||
|   homeTimeline: "Note dagli utenti che segui" |   homeTimeline: "Note dagli utenti che segui" | ||||||
|   users: "Note dagli utenti selezionati" |   users: "Note dagli utenti selezionati" | ||||||
|   userList: "Note dagli utenti della lista selezionata" |   userList: "Note dagli utenti della lista selezionata" | ||||||
|  |   userBlacklist: "Tutte le Note tranne quelle di uno o più profili specificati" | ||||||
| _weekday: | _weekday: | ||||||
|   sunday: "Domenica" |   sunday: "Domenica" | ||||||
|   monday: "Lunedì" |   monday: "Lunedì" | ||||||
| @@ -1874,6 +1903,7 @@ _profile: | |||||||
|   metadataContent: "Contenuto" |   metadataContent: "Contenuto" | ||||||
|   changeAvatar: "Modifica immagine profilo" |   changeAvatar: "Modifica immagine profilo" | ||||||
|   changeBanner: "Cambia intestazione" |   changeBanner: "Cambia intestazione" | ||||||
|  |   verifiedLinkDescription: "Puoi verificare il tuo profilo mostrando una icona. Devi inserire la URL alla pagina che contiene un link al tuo profilo." | ||||||
| _exportOrImport: | _exportOrImport: | ||||||
|   allNotes: "Tutte le note" |   allNotes: "Tutte le note" | ||||||
|   favoritedNotes: "Note preferite" |   favoritedNotes: "Note preferite" | ||||||
| @@ -1992,12 +2022,18 @@ _notification: | |||||||
|   youReceivedFollowRequest: "Hai ricevuto una richiesta di follow" |   youReceivedFollowRequest: "Hai ricevuto una richiesta di follow" | ||||||
|   yourFollowRequestAccepted: "La tua richiesta di follow è stata accettata" |   yourFollowRequestAccepted: "La tua richiesta di follow è stata accettata" | ||||||
|   pollEnded: "Risultati del sondaggio." |   pollEnded: "Risultati del sondaggio." | ||||||
|  |   newNote: "Nuove Note" | ||||||
|   unreadAntennaNote: "Antenna {name}" |   unreadAntennaNote: "Antenna {name}" | ||||||
|   emptyPushNotificationMessage: "Le notifiche push sono state aggiornate." |   emptyPushNotificationMessage: "Le notifiche push sono state aggiornate." | ||||||
|   achievementEarned: "Obiettivo raggiunto" |   achievementEarned: "Obiettivo raggiunto" | ||||||
|  |   testNotification: "Prova la notifica" | ||||||
|  |   checkNotificationBehavior: "Prova il comportamento della notifica" | ||||||
|  |   sendTestNotification: "Spedisci una notifica di prova" | ||||||
|  |   notificationWillBeDisplayedLikeThis: "La notifica apparirà così" | ||||||
|   _types: |   _types: | ||||||
|     all: "Tutto" |     all: "Tutto" | ||||||
|     follow: "Novità follower" |     note: "Nuove Note" | ||||||
|  |     follow: "Nuovi profili follower" | ||||||
|     mention: "Menzioni" |     mention: "Menzioni" | ||||||
|     reply: "Risposte" |     reply: "Risposte" | ||||||
|     renote: "Rinota" |     renote: "Rinota" | ||||||
| @@ -2066,3 +2102,31 @@ _webhookSettings: | |||||||
|     renote: "Quando la Nota è Rinotata" |     renote: "Quando la Nota è Rinotata" | ||||||
|     reaction: "Quando ricevo una reazione" |     reaction: "Quando ricevo una reazione" | ||||||
|     mention: "Quando mi menzionano" |     mention: "Quando mi menzionano" | ||||||
|  | _moderationLogTypes: | ||||||
|  |   createRole: "Ruolo creato" | ||||||
|  |   deleteRole: "Ruolo eliminato" | ||||||
|  |   updateRole: "Ruolo aggiornato" | ||||||
|  |   assignRole: "Ruolo assegnato" | ||||||
|  |   unassignRole: "Ruolo disassegnato" | ||||||
|  |   suspend: "Sospensione" | ||||||
|  |   unsuspend: "Sospensione rimossa" | ||||||
|  |   addCustomEmoji: "Emoji personalizzata aggiunta" | ||||||
|  |   updateCustomEmoji: "Emoji personalizzata aggiornata" | ||||||
|  |   deleteCustomEmoji: "Emoji personalizzata eliminata" | ||||||
|  |   updateServerSettings: "Impostazioni del server aggiornate" | ||||||
|  |   updateUserNote: "Promemoria di moderazione aggiornato" | ||||||
|  |   deleteDriveFile: "File da Drive eliminato" | ||||||
|  |   deleteNote: "Nota eliminata" | ||||||
|  |   createGlobalAnnouncement: "Annuncio globale creato" | ||||||
|  |   createUserAnnouncement: "Annuncio ai profili iscritti creato" | ||||||
|  |   updateGlobalAnnouncement: "Annuncio globale aggiornato" | ||||||
|  |   updateUserAnnouncement: "Annuncio ai profili iscritti aggiornato" | ||||||
|  |   deleteGlobalAnnouncement: "Annuncio globale eliminato" | ||||||
|  |   deleteUserAnnouncement: "Annuncio ai profili iscritti eliminato" | ||||||
|  |   resetPassword: "Password azzerata" | ||||||
|  |   suspendRemoteInstance: "Istanza remota sospesa" | ||||||
|  |   unsuspendRemoteInstance: "Istanza remota riattivata" | ||||||
|  |   markSensitiveDriveFile: "File nel Drive segnato come esplicito" | ||||||
|  |   unmarkSensitiveDriveFile: "File nel Drive segnato come non esplicito" | ||||||
|  |   resolveAbuseReport: "Segnalazione risolta" | ||||||
|  |   createInvitation: "Genera codice di invito" | ||||||
|   | |||||||
| @@ -15,7 +15,7 @@ gotIt: "わかった" | |||||||
| cancel: "キャンセル" | cancel: "キャンセル" | ||||||
| noThankYou: "やめておく" | noThankYou: "やめておく" | ||||||
| enterUsername: "ユーザー名を入力" | enterUsername: "ユーザー名を入力" | ||||||
| renotedBy: "{user}がRenote" | renotedBy: "{user}がリノート" | ||||||
| noNotes: "ノートはありません" | noNotes: "ノートはありません" | ||||||
| noNotifications: "通知はありません" | noNotifications: "通知はありません" | ||||||
| instance: "サーバー" | instance: "サーバー" | ||||||
| @@ -45,10 +45,10 @@ pin: "ピン留め" | |||||||
| unpin: "ピン留め解除" | unpin: "ピン留め解除" | ||||||
| copyContent: "内容をコピー" | copyContent: "内容をコピー" | ||||||
| copyLink: "リンクをコピー" | copyLink: "リンクをコピー" | ||||||
| copyLinkRenote: "Renoteのリンクをコピー" | copyLinkRenote: "リノートのリンクをコピー" | ||||||
| delete: "削除" | delete: "削除" | ||||||
| deleteAndEdit: "削除して編集" | deleteAndEdit: "削除して編集" | ||||||
| deleteAndEditConfirm: "このノートを削除してもう一度編集しますか?このノートへのリアクション、Renote、返信も全て削除されます。" | deleteAndEditConfirm: "このノートを削除してもう一度編集しますか?このノートへのリアクション、リノート、返信も全て削除されます。" | ||||||
| addToList: "リストに追加" | addToList: "リストに追加" | ||||||
| addToAntenna: "アンテナに追加" | addToAntenna: "アンテナに追加" | ||||||
| sendMessage: "メッセージを送信" | sendMessage: "メッセージを送信" | ||||||
| @@ -105,13 +105,13 @@ followRequests: "フォロー申請" | |||||||
| unfollow: "フォロー解除" | unfollow: "フォロー解除" | ||||||
| followRequestPending: "フォロー許可待ち" | followRequestPending: "フォロー許可待ち" | ||||||
| enterEmoji: "絵文字を入力" | enterEmoji: "絵文字を入力" | ||||||
| renote: "Renote" | renote: "リノート" | ||||||
| unrenote: "Renote解除" | unrenote: "リノート解除" | ||||||
| renoted: "Renoteしました。" | renoted: "リノートしました。" | ||||||
| cantRenote: "この投稿はRenoteできません。" | cantRenote: "この投稿はリノートできません。" | ||||||
| cantReRenote: "RenoteをRenoteすることはできません。" | cantReRenote: "リノートをリノートすることはできません。" | ||||||
| quote: "引用" | quote: "引用" | ||||||
| inChannelRenote: "チャンネル内Renote" | inChannelRenote: "チャンネル内リノート" | ||||||
| inChannelQuote: "チャンネル内引用" | inChannelQuote: "チャンネル内引用" | ||||||
| pinnedNote: "ピン留めされたノート" | pinnedNote: "ピン留めされたノート" | ||||||
| pinned: "ピン留め" | pinned: "ピン留め" | ||||||
| @@ -356,7 +356,6 @@ invite: "招待" | |||||||
| driveCapacityPerLocalAccount: "ローカルユーザーひとりあたりのドライブ容量" | driveCapacityPerLocalAccount: "ローカルユーザーひとりあたりのドライブ容量" | ||||||
| driveCapacityPerRemoteAccount: "リモートユーザーひとりあたりのドライブ容量" | driveCapacityPerRemoteAccount: "リモートユーザーひとりあたりのドライブ容量" | ||||||
| inMb: "メガバイト単位" | inMb: "メガバイト単位" | ||||||
| iconUrl: "アイコン画像のURL (faviconなど)" |  | ||||||
| bannerUrl: "バナー画像のURL" | bannerUrl: "バナー画像のURL" | ||||||
| backgroundImageUrl: "背景画像のURL" | backgroundImageUrl: "背景画像のURL" | ||||||
| basicInfo: "基本情報" | basicInfo: "基本情報" | ||||||
| @@ -417,6 +416,9 @@ totp: "認証アプリ" | |||||||
| totpDescription: "認証アプリを使ってワンタイムパスワードを入力" | totpDescription: "認証アプリを使ってワンタイムパスワードを入力" | ||||||
| moderator: "モデレーター" | moderator: "モデレーター" | ||||||
| moderation: "モデレーション" | moderation: "モデレーション" | ||||||
|  | moderationNote: "モデレーションノート" | ||||||
|  | addModerationNote: "モデレーションノートを追加する" | ||||||
|  | moderationLogs: "モデログ" | ||||||
| nUsersMentioned: "{n}人が投稿" | nUsersMentioned: "{n}人が投稿" | ||||||
| securityKeyAndPasskey: "セキュリティキー・パスキー" | securityKeyAndPasskey: "セキュリティキー・パスキー" | ||||||
| securityKey: "セキュリティキー" | securityKey: "セキュリティキー" | ||||||
| @@ -656,7 +658,7 @@ behavior: "動作" | |||||||
| sample: "サンプル" | sample: "サンプル" | ||||||
| abuseReports: "通報" | abuseReports: "通報" | ||||||
| reportAbuse: "通報" | reportAbuse: "通報" | ||||||
| reportAbuseRenote: "Renoteを通報" | reportAbuseRenote: "リノートを通報" | ||||||
| reportAbuseOf: "{name}を通報する" | reportAbuseOf: "{name}を通報する" | ||||||
| fillAbuseReportDescription: "通報理由の詳細を記入してください。対象のノートがある場合はそのURLも記入してください。" | fillAbuseReportDescription: "通報理由の詳細を記入してください。対象のノートがある場合はそのURLも記入してください。" | ||||||
| abuseReported: "内容が送信されました。ご報告ありがとうございました。" | abuseReported: "内容が送信されました。ご報告ありがとうございました。" | ||||||
| @@ -690,9 +692,9 @@ manageAccessTokens: "アクセストークンの管理" | |||||||
| accountInfo: "アカウント情報" | accountInfo: "アカウント情報" | ||||||
| notesCount: "ノートの数" | notesCount: "ノートの数" | ||||||
| repliesCount: "返信した数" | repliesCount: "返信した数" | ||||||
| renotesCount: "Renoteした数" | renotesCount: "リノートした数" | ||||||
| repliedCount: "返信された数" | repliedCount: "返信された数" | ||||||
| renotedCount: "Renoteされた数" | renotedCount: "リノートされた数" | ||||||
| followingCount: "フォロー数" | followingCount: "フォロー数" | ||||||
| followersCount: "フォロワー数" | followersCount: "フォロワー数" | ||||||
| sentReactionsCount: "リアクションした数" | sentReactionsCount: "リアクションした数" | ||||||
| @@ -709,6 +711,7 @@ lockedAccountInfo: "フォローを承認制にしても、ノートの公開範 | |||||||
| alwaysMarkSensitive: "デフォルトでメディアをセンシティブ設定にする" | alwaysMarkSensitive: "デフォルトでメディアをセンシティブ設定にする" | ||||||
| loadRawImages: "添付画像のサムネイルをオリジナル画質にする" | loadRawImages: "添付画像のサムネイルをオリジナル画質にする" | ||||||
| disableShowingAnimatedImages: "アニメーション画像を再生しない" | disableShowingAnimatedImages: "アニメーション画像を再生しない" | ||||||
|  | highlightSensitiveMedia: "メディアがセンシティブであることを分かりやすく表示" | ||||||
| verificationEmailSent: "確認のメールを送信しました。メールに記載されたリンクにアクセスして、設定を完了してください。" | verificationEmailSent: "確認のメールを送信しました。メールに記載されたリンクにアクセスして、設定を完了してください。" | ||||||
| notSet: "未設定" | notSet: "未設定" | ||||||
| emailVerified: "メールアドレスが確認されました" | emailVerified: "メールアドレスが確認されました" | ||||||
| @@ -988,7 +991,7 @@ thisPostMayBeAnnoying: "この投稿は迷惑になる可能性があります | |||||||
| thisPostMayBeAnnoyingHome: "ホームに投稿" | thisPostMayBeAnnoyingHome: "ホームに投稿" | ||||||
| thisPostMayBeAnnoyingCancel: "やめる" | thisPostMayBeAnnoyingCancel: "やめる" | ||||||
| thisPostMayBeAnnoyingIgnore: "このまま投稿" | thisPostMayBeAnnoyingIgnore: "このまま投稿" | ||||||
| collapseRenotes: "見たことのあるRenoteを省略して表示" | collapseRenotes: "見たことのあるリノートを省略して表示" | ||||||
| internalServerError: "サーバー内部エラー" | internalServerError: "サーバー内部エラー" | ||||||
| internalServerErrorDescription: "サーバー内部で予期しないエラーが発生しました。" | internalServerErrorDescription: "サーバー内部で予期しないエラーが発生しました。" | ||||||
| copyErrorInfo: "エラー情報をコピー" | copyErrorInfo: "エラー情報をコピー" | ||||||
| @@ -1023,7 +1026,7 @@ retryAllQueuesConfirmText: "一時的にサーバーの負荷が増大するこ | |||||||
| enableChartsForRemoteUser: "リモートユーザーのチャートを生成" | enableChartsForRemoteUser: "リモートユーザーのチャートを生成" | ||||||
| enableChartsForFederatedInstances: "リモートサーバーのチャートを生成" | enableChartsForFederatedInstances: "リモートサーバーのチャートを生成" | ||||||
| showClipButtonInNoteFooter: "ノートのアクションにクリップを追加" | showClipButtonInNoteFooter: "ノートのアクションにクリップを追加" | ||||||
| largeNoteReactions: "ノートのリアクションを大きく表示" | reactionsDisplaySize: "リアクションの表示サイズ" | ||||||
| noteIdOrUrl: "ノートIDまたはURL" | noteIdOrUrl: "ノートIDまたはURL" | ||||||
| video: "動画" | video: "動画" | ||||||
| videos: "動画" | videos: "動画" | ||||||
| @@ -1036,7 +1039,7 @@ forceShowAds: "常に広告を表示する" | |||||||
| addMemo: "メモを追加" | addMemo: "メモを追加" | ||||||
| editMemo: "メモを編集" | editMemo: "メモを編集" | ||||||
| reactionsList: "リアクション一覧" | reactionsList: "リアクション一覧" | ||||||
| renotesList: "Renote一覧" | renotesList: "リノート一覧" | ||||||
| notificationDisplay: "通知の表示" | notificationDisplay: "通知の表示" | ||||||
| leftTop: "左上" | leftTop: "左上" | ||||||
| rightTop: "右上" | rightTop: "右上" | ||||||
| @@ -1105,6 +1108,26 @@ forYou: "あなたへ" | |||||||
| currentAnnouncements: "現在のお知らせ" | currentAnnouncements: "現在のお知らせ" | ||||||
| pastAnnouncements: "過去のお知らせ" | pastAnnouncements: "過去のお知らせ" | ||||||
| youHaveUnreadAnnouncements: "未読のお知らせがあります。" | youHaveUnreadAnnouncements: "未読のお知らせがあります。" | ||||||
|  | useSecurityKey: "ブラウザまたはデバイスの指示に従って、セキュリティキーまたはパスキーを使用してください。" | ||||||
|  | replies: "返信" | ||||||
|  | renotes: "リノート" | ||||||
|  | loadReplies: "返信を見る" | ||||||
|  | loadConversation: "会話を見る" | ||||||
|  | pinnedList: "ピン留めされたリスト" | ||||||
|  | keepScreenOn: "デバイスの画面を常にオンにする" | ||||||
|  | verifiedLink: "このリンク先の所有者であることが確認されました" | ||||||
|  | notifyNotes: "投稿を通知" | ||||||
|  | unnotifyNotes: "投稿の通知を解除" | ||||||
|  | authentication: "認証" | ||||||
|  | authenticationRequiredToContinue: "続けるには認証を行ってください" | ||||||
|  | dateAndTime: "日時" | ||||||
|  | showRenotes: "リノートを表示" | ||||||
|  | edited: "編集済み" | ||||||
|  | notificationRecieveConfig: "通知の受信設定" | ||||||
|  | mutualFollow: "相互フォロー" | ||||||
|  | fileAttachedOnly: "ファイル付きのみ" | ||||||
|  | showRepliesToOthersInTimeline: "TLに他の人への返信を含める" | ||||||
|  | hideRepliesToOthersInTimeline: "TLに他の人への返信を含めない" | ||||||
|  |  | ||||||
| _announcement: | _announcement: | ||||||
|   forExistingUsers: "既存ユーザーのみ" |   forExistingUsers: "既存ユーザーのみ" | ||||||
| @@ -1135,6 +1158,16 @@ _initialAccountSetting: | |||||||
| _serverRules: | _serverRules: | ||||||
|   description: "新規登録前に表示する、サーバーの簡潔なルールを設定します。内容は利用規約の要約とすることを推奨します。" |   description: "新規登録前に表示する、サーバーの簡潔なルールを設定します。内容は利用規約の要約とすることを推奨します。" | ||||||
|  |  | ||||||
|  | _serverSettings: | ||||||
|  |   iconUrl: "アイコン画像のURL" | ||||||
|  |   appIconDescription: "{host}がアプリとして表示される際のアイコンを指定します。" | ||||||
|  |   appIconUsageExample: "例: PWAや、スマートフォンのホーム画面にブックマークとして追加された時など" | ||||||
|  |   appIconStyleRecommendation: "円形もしくは角丸にクロップされる場合があるため、塗り潰された余白のある背景を持つことが推奨されます。" | ||||||
|  |   appIconResolutionMustBe: "解像度は必ず{resolution}である必要があります。" | ||||||
|  |   manifestJsonOverride: "manifest.jsonのオーバーライド" | ||||||
|  |   shortName: "略称" | ||||||
|  |   shortNameDescription: "サーバーの正式名称が長い場合に、代わりに表示することのできる略称や通称。" | ||||||
|  |  | ||||||
| _accountMigration: | _accountMigration: | ||||||
|   moveFrom: "別のアカウントからこのアカウントに移行" |   moveFrom: "別のアカウントからこのアカウントに移行" | ||||||
|   moveFromSub: "別のアカウントへエイリアスを作成" |   moveFromSub: "別のアカウントへエイリアスを作成" | ||||||
| @@ -1390,6 +1423,9 @@ _achievements: | |||||||
|       title: "Brain Diver" |       title: "Brain Diver" | ||||||
|       description: "Brain Diverへのリンクを投稿した" |       description: "Brain Diverへのリンクを投稿した" | ||||||
|       flavor: "Misskey-Misskey La-Tu-Ma" |       flavor: "Misskey-Misskey La-Tu-Ma" | ||||||
|  |     _smashTestNotificationButton: | ||||||
|  |       title: "テスト過剰" | ||||||
|  |       description: "通知のテストをごく短時間のうちに連続して行った" | ||||||
|  |  | ||||||
| _role: | _role: | ||||||
|   new: "ロールの作成" |   new: "ロールの作成" | ||||||
| @@ -1429,6 +1465,7 @@ _role: | |||||||
|     gtlAvailable: "グローバルタイムラインの閲覧" |     gtlAvailable: "グローバルタイムラインの閲覧" | ||||||
|     ltlAvailable: "ローカルタイムラインの閲覧" |     ltlAvailable: "ローカルタイムラインの閲覧" | ||||||
|     canPublicNote: "パブリック投稿の許可" |     canPublicNote: "パブリック投稿の許可" | ||||||
|  |     canEditNote: "ノートの編集" | ||||||
|     canInvite: "サーバー招待コードの発行" |     canInvite: "サーバー招待コードの発行" | ||||||
|     inviteLimit: "招待コードの作成可能数" |     inviteLimit: "招待コードの作成可能数" | ||||||
|     inviteLimitCycle: "招待コードの発行間隔" |     inviteLimitCycle: "招待コードの発行間隔" | ||||||
| @@ -1447,7 +1484,8 @@ _role: | |||||||
|     rateLimitFactor: "レートリミット" |     rateLimitFactor: "レートリミット" | ||||||
|     descriptionOfRateLimitFactor: "小さいほど制限が緩和され、大きいほど制限が強化されます。" |     descriptionOfRateLimitFactor: "小さいほど制限が緩和され、大きいほど制限が強化されます。" | ||||||
|     canHideAds: "広告の非表示" |     canHideAds: "広告の非表示" | ||||||
|     canSearchNotes: "ノート検索の利用可否" |     canSearchNotes: "ノート検索の利用" | ||||||
|  |     canUseTranslator: "翻訳機能の利用" | ||||||
|   _condition: |   _condition: | ||||||
|     isLocal: "ローカルユーザー" |     isLocal: "ローカルユーザー" | ||||||
|     isRemote: "リモートユーザー" |     isRemote: "リモートユーザー" | ||||||
| @@ -1524,6 +1562,7 @@ _plugin: | |||||||
|   install: "プラグインのインストール" |   install: "プラグインのインストール" | ||||||
|   installWarn: "信頼できないプラグインはインストールしないでください。" |   installWarn: "信頼できないプラグインはインストールしないでください。" | ||||||
|   manage: "プラグインの管理" |   manage: "プラグインの管理" | ||||||
|  |   viewSource: "ソースを表示" | ||||||
|  |  | ||||||
| _preferencesBackups: | _preferencesBackups: | ||||||
|   list: "作成したバックアップ" |   list: "作成したバックアップ" | ||||||
| @@ -1599,11 +1638,6 @@ _wordMute: | |||||||
|   muteWords: "ミュートするワード" |   muteWords: "ミュートするワード" | ||||||
|   muteWordsDescription: "スペースで区切るとAND指定になり、改行で区切るとOR指定になります。" |   muteWordsDescription: "スペースで区切るとAND指定になり、改行で区切るとOR指定になります。" | ||||||
|   muteWordsDescription2: "キーワードをスラッシュで囲むと正規表現になります。" |   muteWordsDescription2: "キーワードをスラッシュで囲むと正規表現になります。" | ||||||
|   softDescription: "指定した条件のノートをタイムラインから隠します。" |  | ||||||
|   hardDescription: "指定した条件のノートをタイムラインに追加しないようにします。追加されなかったノートは、条件を変更しても除外されたままになります。" |  | ||||||
|   soft: "ソフト" |  | ||||||
|   hard: "ハード" |  | ||||||
|   mutedNotes: "ミュートされたノート" |  | ||||||
|  |  | ||||||
| _instanceMute: | _instanceMute: | ||||||
|   instanceMuteDescription: "ミュートしたサーバーのユーザーへの返信を含めて、設定したサーバーの全てのノートとRenoteをミュートします。" |   instanceMuteDescription: "ミュートしたサーバーのユーザーへの返信を含めて、設定したサーバーの全てのノートとRenoteをミュートします。" | ||||||
| @@ -1728,7 +1762,6 @@ _timelineTutorial: | |||||||
| _2fa: | _2fa: | ||||||
|   alreadyRegistered: "既に設定は完了しています。" |   alreadyRegistered: "既に設定は完了しています。" | ||||||
|   registerTOTP: "認証アプリの設定を開始" |   registerTOTP: "認証アプリの設定を開始" | ||||||
|   passwordToTOTP: "パスワードを入力してください" |  | ||||||
|   step1: "まず、{a}や{b}などの認証アプリをお使いのデバイスにインストールします。" |   step1: "まず、{a}や{b}などの認証アプリをお使いのデバイスにインストールします。" | ||||||
|   step2: "次に、表示されているQRコードをアプリでスキャンします。" |   step2: "次に、表示されているQRコードをアプリでスキャンします。" | ||||||
|   step2Click: "QRコードをクリックすると、お使いの端末にインストールされている認証アプリやキーリングに登録できます。" |   step2Click: "QRコードをクリックすると、お使いの端末にインストールされている認証アプリやキーリングに登録できます。" | ||||||
| @@ -1740,7 +1773,6 @@ _2fa: | |||||||
|   securityKeyNotSupported: "お使いのブラウザはセキュリティキーに対応していません。" |   securityKeyNotSupported: "お使いのブラウザはセキュリティキーに対応していません。" | ||||||
|   registerTOTPBeforeKey: "セキュリティキー・パスキーを登録するには、まず認証アプリの設定を行なってください。" |   registerTOTPBeforeKey: "セキュリティキー・パスキーを登録するには、まず認証アプリの設定を行なってください。" | ||||||
|   securityKeyInfo: "FIDO2をサポートするハードウェアセキュリティキー、端末の生体認証やPINロック、パスキーといった、WebAuthn由来の鍵を登録します。" |   securityKeyInfo: "FIDO2をサポートするハードウェアセキュリティキー、端末の生体認証やPINロック、パスキーといった、WebAuthn由来の鍵を登録します。" | ||||||
|   chromePasskeyNotSupported: "Chromeのパスキーは現在サポートしていません。" |  | ||||||
|   registerSecurityKey: "セキュリティキー・パスキーを登録する" |   registerSecurityKey: "セキュリティキー・パスキーを登録する" | ||||||
|   securityKeyName: "キーの名前を入力" |   securityKeyName: "キーの名前を入力" | ||||||
|   tapSecurityKey: "ブラウザの指示に従い、セキュリティキーやパスキーを登録してください" |   tapSecurityKey: "ブラウザの指示に従い、セキュリティキーやパスキーを登録してください" | ||||||
| @@ -1811,6 +1843,7 @@ _antennaSources: | |||||||
|   homeTimeline: "フォローしているユーザーのノート" |   homeTimeline: "フォローしているユーザーのノート" | ||||||
|   users: "指定した一人または複数のユーザーのノート" |   users: "指定した一人または複数のユーザーのノート" | ||||||
|   userList: "指定したリストのユーザーのノート" |   userList: "指定したリストのユーザーのノート" | ||||||
|  |   userBlacklist: "指定した一人または複数のユーザーを除いた全てのノート" | ||||||
|  |  | ||||||
| _weekday: | _weekday: | ||||||
|   sunday: "日曜日" |   sunday: "日曜日" | ||||||
| @@ -1917,6 +1950,7 @@ _profile: | |||||||
|   metadataContent: "内容" |   metadataContent: "内容" | ||||||
|   changeAvatar: "アイコン画像を変更" |   changeAvatar: "アイコン画像を変更" | ||||||
|   changeBanner: "バナー画像を変更" |   changeBanner: "バナー画像を変更" | ||||||
|  |   verifiedLinkDescription: "内容にURLを設定すると、リンク先のWebサイトに自分のプロフィールへのリンクが含まれている場合に所有者確認済みアイコンを表示させることができます。" | ||||||
|  |  | ||||||
| _exportOrImport: | _exportOrImport: | ||||||
|   allNotes: "全てのノート" |   allNotes: "全てのノート" | ||||||
| @@ -2044,12 +2078,18 @@ _notification: | |||||||
|   youReceivedFollowRequest: "フォローリクエストが来ました" |   youReceivedFollowRequest: "フォローリクエストが来ました" | ||||||
|   yourFollowRequestAccepted: "フォローリクエストが承認されました" |   yourFollowRequestAccepted: "フォローリクエストが承認されました" | ||||||
|   pollEnded: "アンケートの結果が出ました" |   pollEnded: "アンケートの結果が出ました" | ||||||
|  |   newNote: "新しい投稿" | ||||||
|   unreadAntennaNote: "アンテナ {name}" |   unreadAntennaNote: "アンテナ {name}" | ||||||
|   emptyPushNotificationMessage: "プッシュ通知の更新をしました" |   emptyPushNotificationMessage: "プッシュ通知の更新をしました" | ||||||
|   achievementEarned: "実績を獲得" |   achievementEarned: "実績を獲得" | ||||||
|  |   testNotification: "通知テスト" | ||||||
|  |   checkNotificationBehavior: "通知の表示を確かめる" | ||||||
|  |   sendTestNotification: "テスト通知を送信する" | ||||||
|  |   notificationWillBeDisplayedLikeThis: "通知はこのように表示されます" | ||||||
|  |  | ||||||
|   _types: |   _types: | ||||||
|     all: "すべて" |     all: "すべて" | ||||||
|  |     note: "ユーザーの新規投稿" | ||||||
|     follow: "フォロー" |     follow: "フォロー" | ||||||
|     mention: "メンション" |     mention: "メンション" | ||||||
|     reply: "リプライ" |     reply: "リプライ" | ||||||
| @@ -2126,3 +2166,35 @@ _webhookSettings: | |||||||
|     renote: "Renoteされたとき" |     renote: "Renoteされたとき" | ||||||
|     reaction: "リアクションがあったとき" |     reaction: "リアクションがあったとき" | ||||||
|     mention: "メンションされたとき" |     mention: "メンションされたとき" | ||||||
|  |  | ||||||
|  | _moderationLogTypes: | ||||||
|  |   createRole: "ロールを作成" | ||||||
|  |   deleteRole: "ロールを削除" | ||||||
|  |   updateRole: "ロールを更新" | ||||||
|  |   assignRole: "ロールへアサイン" | ||||||
|  |   unassignRole: "ロールのアサイン解除" | ||||||
|  |   suspend: "凍結" | ||||||
|  |   unsuspend: "凍結解除" | ||||||
|  |   addCustomEmoji: "カスタム絵文字追加" | ||||||
|  |   updateCustomEmoji: "カスタム絵文字更新" | ||||||
|  |   deleteCustomEmoji: "カスタム絵文字削除" | ||||||
|  |   updateServerSettings: "サーバー設定更新" | ||||||
|  |   updateUserNote: "モデレーションノート更新" | ||||||
|  |   deleteDriveFile: "ファイルを削除" | ||||||
|  |   deleteNote: "ノートを削除" | ||||||
|  |   createGlobalAnnouncement: "全体のお知らせを作成" | ||||||
|  |   createUserAnnouncement: "ユーザーへお知らせを作成" | ||||||
|  |   updateGlobalAnnouncement: "全体のお知らせを更新" | ||||||
|  |   updateUserAnnouncement: "ユーザーのお知らせを更新" | ||||||
|  |   deleteGlobalAnnouncement: "全体のお知らせを削除" | ||||||
|  |   deleteUserAnnouncement: "ユーザーのお知らせを削除" | ||||||
|  |   resetPassword: "パスワードをリセット" | ||||||
|  |   suspendRemoteInstance: "リモートサーバーを停止" | ||||||
|  |   unsuspendRemoteInstance: "リモートサーバーを再開" | ||||||
|  |   markSensitiveDriveFile: "ファイルをセンシティブ付与" | ||||||
|  |   unmarkSensitiveDriveFile: "ファイルをセンシティブ解除" | ||||||
|  |   resolveAbuseReport: "通報を解決" | ||||||
|  |   createInvitation: "招待コードを作成" | ||||||
|  |   createAd: "広告を作成" | ||||||
|  |   deleteAd: "広告を削除" | ||||||
|  |   updateAd: "広告を更新" | ||||||
|   | |||||||
| @@ -355,7 +355,6 @@ invite: "来てや" | |||||||
| driveCapacityPerLocalAccount: "ローカルユーザーはんひとりあたりのドライブ容量" | driveCapacityPerLocalAccount: "ローカルユーザーはんひとりあたりのドライブ容量" | ||||||
| driveCapacityPerRemoteAccount: "リモートユーザーはんひとりあたりのドライブ容量" | driveCapacityPerRemoteAccount: "リモートユーザーはんひとりあたりのドライブ容量" | ||||||
| inMb: "メガバイト単位" | inMb: "メガバイト単位" | ||||||
| iconUrl: "アイコン画像のURL" |  | ||||||
| bannerUrl: "バナー画像のURL" | bannerUrl: "バナー画像のURL" | ||||||
| backgroundImageUrl: "背景画像のURL" | backgroundImageUrl: "背景画像のURL" | ||||||
| basicInfo: "基本情報" | basicInfo: "基本情報" | ||||||
| @@ -415,6 +414,8 @@ totp: "認証アプリ" | |||||||
| totpDescription: "認証アプリ使うてワンタイムパスワードを入れる" | totpDescription: "認証アプリ使うてワンタイムパスワードを入れる" | ||||||
| moderator: "モデレーター" | moderator: "モデレーター" | ||||||
| moderation: "モデレーション" | moderation: "モデレーション" | ||||||
|  | moderationNote: "モデレーションノート" | ||||||
|  | addModerationNote: "モデレーションノートを追加するで" | ||||||
| nUsersMentioned: "{n}人が投稿" | nUsersMentioned: "{n}人が投稿" | ||||||
| securityKeyAndPasskey: "セキュリティキー・パスキー" | securityKeyAndPasskey: "セキュリティキー・パスキー" | ||||||
| securityKey: "セキュリティキー" | securityKey: "セキュリティキー" | ||||||
| @@ -1020,7 +1021,6 @@ retryAllQueuesConfirmText: "一時的にサーバー重なるかもしれへん | |||||||
| enableChartsForRemoteUser: "リモートユーザーのチャートを作る" | enableChartsForRemoteUser: "リモートユーザーのチャートを作る" | ||||||
| enableChartsForFederatedInstances: "リモートサーバーのチャートを作る" | enableChartsForFederatedInstances: "リモートサーバーのチャートを作る" | ||||||
| showClipButtonInNoteFooter: "ノートのアクションにクリップを追加" | showClipButtonInNoteFooter: "ノートのアクションにクリップを追加" | ||||||
| largeNoteReactions: "ノートのツッコミを大きする" |  | ||||||
| noteIdOrUrl: "ノートIDかURL" | noteIdOrUrl: "ノートIDかURL" | ||||||
| video: "動画" | video: "動画" | ||||||
| videos: "動画" | videos: "動画" | ||||||
| @@ -1102,6 +1102,13 @@ forYou: "あんたへ" | |||||||
| currentAnnouncements: "現在のお知らせやで" | currentAnnouncements: "現在のお知らせやで" | ||||||
| pastAnnouncements: "過去のお知らせやで" | pastAnnouncements: "過去のお知らせやで" | ||||||
| youHaveUnreadAnnouncements: "あんたまだこのお知らせ読んどらんやろ。" | youHaveUnreadAnnouncements: "あんたまだこのお知らせ読んどらんやろ。" | ||||||
|  | useSecurityKey: "ブラウザまたはデバイスの言う通りに、セキュリティキーまたはパスキーを使ってや。" | ||||||
|  | replies: "返事" | ||||||
|  | renotes: "Renote" | ||||||
|  | loadReplies: "返信を見るで" | ||||||
|  | loadConversation: "会話を見るで" | ||||||
|  | verifiedLink: "このリンク先の所有者であることが確認されたで。" | ||||||
|  | authenticationRequiredToContinue: "続けるには認証をやってや。" | ||||||
| _announcement: | _announcement: | ||||||
|   forExistingUsers: "もうおるユーザーのみ" |   forExistingUsers: "もうおるユーザーのみ" | ||||||
|   forExistingUsersDescription: "有効にすると、このお知らせ作成時点でおるユーザーにのみお知らせが表示されます。無効にすると、このお知らせ作成後にアカウントを作成したユーザーにもお知らせが表示されます。" |   forExistingUsersDescription: "有効にすると、このお知らせ作成時点でおるユーザーにのみお知らせが表示されます。無効にすると、このお知らせ作成後にアカウントを作成したユーザーにもお知らせが表示されます。" | ||||||
| @@ -1128,6 +1135,13 @@ _initialAccountSetting: | |||||||
|   laterAreYouSure: "初期設定あとでやり直すん?" |   laterAreYouSure: "初期設定あとでやり直すん?" | ||||||
| _serverRules: | _serverRules: | ||||||
|   description: "新規登録前に見せる、サーバーの簡潔なルールを設定すんで。内容は使うための決め事の要約とすることを推奨するわ。" |   description: "新規登録前に見せる、サーバーの簡潔なルールを設定すんで。内容は使うための決め事の要約とすることを推奨するわ。" | ||||||
|  | _serverSettings: | ||||||
|  |   iconUrl: "アイコン画像のURL" | ||||||
|  |   appIconDescription: "{host}がアプリとして表示してるんやつをアイコンを指定すんで。" | ||||||
|  |   appIconUsageExample: "PWAや、スマートフォンのホーム画面にブックマークとして追加された時など" | ||||||
|  |   appIconStyleRecommendation: "円形もしくは角丸にクロップされる場合があるさかいに、塗り潰された余白のある背景があるものが推奨されるで。" | ||||||
|  |   appIconResolutionMustBe: "解像度は必ず{resolution}である必要があるで。" | ||||||
|  |   shortNameDescription: "サーバーの名前が長い時に、代わりに表示することのできるあだ名。" | ||||||
| _accountMigration: | _accountMigration: | ||||||
|   moveFrom: "別のアカウントからこのアカウントに引っ越す" |   moveFrom: "別のアカウントからこのアカウントに引っ越す" | ||||||
|   moveFromSub: "別のアカウントへエイリアスを作る" |   moveFromSub: "別のアカウントへエイリアスを作る" | ||||||
| @@ -1505,6 +1519,7 @@ _plugin: | |||||||
|   install: "プラグインのインストール" |   install: "プラグインのインストール" | ||||||
|   installWarn: "信頼できへんプラグインはインストールせんとってな" |   installWarn: "信頼できへんプラグインはインストールせんとってな" | ||||||
|   manage: "プラグインの管理" |   manage: "プラグインの管理" | ||||||
|  |   viewSource: "ソースを表示" | ||||||
| _preferencesBackups: | _preferencesBackups: | ||||||
|   list: "作ったバックアップ" |   list: "作ったバックアップ" | ||||||
|   saveNew: "新しく保存" |   saveNew: "新しく保存" | ||||||
| @@ -1692,17 +1707,16 @@ _timelineTutorial: | |||||||
| _2fa: | _2fa: | ||||||
|   alreadyRegistered: "もう設定終わっとるわ。" |   alreadyRegistered: "もう設定終わっとるわ。" | ||||||
|   registerTOTP: "認証アプリの設定はじめる" |   registerTOTP: "認証アプリの設定はじめる" | ||||||
|   passwordToTOTP: "パスワードを入れてーや" |  | ||||||
|   step1: "ほんなら、{a}や{b}とかの認証アプリを使っとるデバイスにインストールしてな。" |   step1: "ほんなら、{a}や{b}とかの認証アプリを使っとるデバイスにインストールしてな。" | ||||||
|   step2: "次に、ここにあるQRコードをアプリでスキャンしてな~。" |   step2: "次に、ここにあるQRコードをアプリでスキャンしてな~。" | ||||||
|   step2Click: "QRコードをクリックすると、今使とる端末に入っとる認証アプリとかキーリングに登録できるで。" |   step2Click: "QRコードをクリックすると、今使とる端末に入っとる認証アプリとかキーリングに登録できるで。" | ||||||
|   step3Title: "確認コードを入れてーや" |   step3Title: "確認コードを入れてーや" | ||||||
|   step3: "アプリに表示されているトークンを入力して終わりや。" |   step3: "アプリに表示されているトークンを入力して終わりや。" | ||||||
|  |   setupCompleted: "設定が完了したで。" | ||||||
|   step4: "これからログインするときも、同じようにトークンを入力するんやで" |   step4: "これからログインするときも、同じようにトークンを入力するんやで" | ||||||
|   securityKeyNotSupported: "今使とるブラウザはセキュリティキーに対応してへんのやってさ。" |   securityKeyNotSupported: "今使とるブラウザはセキュリティキーに対応してへんのやってさ。" | ||||||
|   registerTOTPBeforeKey: "セキュリティキー・パスキーを登録するんやったら、まず認証アプリを設定してーな。" |   registerTOTPBeforeKey: "セキュリティキー・パスキーを登録するんやったら、まず認証アプリを設定してーな。" | ||||||
|   securityKeyInfo: "FIDO2をサポートするハードウェアセキュリティキーか端末の指紋認証やPINを使ってログインするように設定できるで。" |   securityKeyInfo: "FIDO2をサポートするハードウェアセキュリティキーか端末の指紋認証やPINを使ってログインするように設定できるで。" | ||||||
|   chromePasskeyNotSupported: "Chromeのパスキーは今んとこ対応してないねん。" |  | ||||||
|   registerSecurityKey: "セキュリティキー・パスキーを登録するわ" |   registerSecurityKey: "セキュリティキー・パスキーを登録するわ" | ||||||
|   securityKeyName: "キーの名前を入れてーや" |   securityKeyName: "キーの名前を入れてーや" | ||||||
|   tapSecurityKey: "ブラウザが言うこと聞いて、セキュリティキーとかパスキー登録しといでや" |   tapSecurityKey: "ブラウザが言うこと聞いて、セキュリティキーとかパスキー登録しといでや" | ||||||
| @@ -1713,6 +1727,10 @@ _2fa: | |||||||
|   renewTOTPConfirm: "今までの認証アプリの確認コードは使えんくなるけどええか?" |   renewTOTPConfirm: "今までの認証アプリの確認コードは使えんくなるけどええか?" | ||||||
|   renewTOTPOk: "もっかい設定する" |   renewTOTPOk: "もっかい設定する" | ||||||
|   renewTOTPCancel: "やめとく" |   renewTOTPCancel: "やめとく" | ||||||
|  |   checkBackupCodesBeforeCloseThisWizard: "このウィザードを閉じる前に、したのバックアップコードを確認しいや。" | ||||||
|  |   backupCodesDescription: "認証アプリが使用できんなった場合、以下のバックアップコードを使ってアカウントにアクセスできるで。これらのコードは必ず安全な場所に置いときや。各コードは一回だけ使用できるで。" | ||||||
|  |   backupCodeUsedWarning: "バックアップコードが使用されたで。認証アプリが使えなくなってるん場合、なるべく早く認証アプリを再設定しや。" | ||||||
|  |   backupCodesExhaustedWarning: "バックアップコードが全て使用されたで。認証アプリを利用できん場合、これ以上アカウントにアクセスできなくなるで。認証アプリを再登録しや。" | ||||||
| _permissions: | _permissions: | ||||||
|   "read:account": "アカウントの情報を見るで" |   "read:account": "アカウントの情報を見るで" | ||||||
|   "write:account": "アカウントの情報を変更するで" |   "write:account": "アカウントの情報を変更するで" | ||||||
| @@ -1985,6 +2003,9 @@ _notification: | |||||||
|   unreadAntennaNote: "アンテナ {name}" |   unreadAntennaNote: "アンテナ {name}" | ||||||
|   emptyPushNotificationMessage: "プッシュ通知の更新をしといたで" |   emptyPushNotificationMessage: "プッシュ通知の更新をしといたで" | ||||||
|   achievementEarned: "実績を獲得しとるで" |   achievementEarned: "実績を獲得しとるで" | ||||||
|  |   checkNotificationBehavior: "通知の表示を確かめるで" | ||||||
|  |   sendTestNotification: "テスト通知を送信するで" | ||||||
|  |   notificationWillBeDisplayedLikeThis: "通知はこのように表示されるで" | ||||||
|   _types: |   _types: | ||||||
|     all: "すべて" |     all: "すべて" | ||||||
|     follow: "フォロー" |     follow: "フォロー" | ||||||
| @@ -2020,6 +2041,7 @@ _deck: | |||||||
|   introduction2: "画面の右にある + を押して、いつでもカラムを追加できるで。" |   introduction2: "画面の右にある + を押して、いつでもカラムを追加できるで。" | ||||||
|   widgetsIntroduction: "カラムのメニューから、「ウィジェットの編集」を選んでウィジェットを追加してなー" |   widgetsIntroduction: "カラムのメニューから、「ウィジェットの編集」を選んでウィジェットを追加してなー" | ||||||
|   useSimpleUiForNonRootPages: "非ルートページは簡易UIで表示" |   useSimpleUiForNonRootPages: "非ルートページは簡易UIで表示" | ||||||
|  |   usedAsMinWidthWhenFlexible: "「幅を自動調整」が有効の場合、これが幅の最小値となるで" | ||||||
|   _columns: |   _columns: | ||||||
|     main: "メイン" |     main: "メイン" | ||||||
|     widgets: "ウィジェット" |     widgets: "ウィジェット" | ||||||
| @@ -2054,3 +2076,7 @@ _webhookSettings: | |||||||
|     renote: "Renoteされるとき~!" |     renote: "Renoteされるとき~!" | ||||||
|     reaction: "ツッコミがあるとき~!" |     reaction: "ツッコミがあるとき~!" | ||||||
|     mention: "メンションがあるとき~!" |     mention: "メンションがあるとき~!" | ||||||
|  | _moderationLogTypes: | ||||||
|  |   suspend: "凍結" | ||||||
|  |   resetPassword: "パスワードをリセット" | ||||||
|  |   createInvitation: "招待コードを作成" | ||||||
|   | |||||||
| @@ -56,6 +56,7 @@ accounts: "Imiḍan" | |||||||
| searchByGoogle: "Nadi" | searchByGoogle: "Nadi" | ||||||
| file: "Ifuyla" | file: "Ifuyla" | ||||||
| account: "Imiḍan" | account: "Imiḍan" | ||||||
|  | replies: "Err" | ||||||
| _email: | _email: | ||||||
|   _follow: |   _follow: | ||||||
|     title: "Yeṭṭafaṛ-ik·em-id" |     title: "Yeṭṭafaṛ-ik·em-id" | ||||||
|   | |||||||
| @@ -61,6 +61,7 @@ smtpPass: "ಗುಪ್ತಪದ" | |||||||
| user: "ಬಳಕೆದಾರ" | user: "ಬಳಕೆದಾರ" | ||||||
| searchByGoogle: "ಹುಡುಕು" | searchByGoogle: "ಹುಡುಕು" | ||||||
| file: "ಕಡತಗಳು" | file: "ಕಡತಗಳು" | ||||||
|  | replies: "ಉತ್ತರಿಸು" | ||||||
| _email: | _email: | ||||||
|   _follow: |   _follow: | ||||||
|     title: "ಹಿಂಬಾಲಿಸಿದರು" |     title: "ಹಿಂಬಾಲಿಸಿದರು" | ||||||
|   | |||||||
| @@ -45,6 +45,7 @@ pin: "프로필에 고정" | |||||||
| unpin: "프로필에서 고정 해제" | unpin: "프로필에서 고정 해제" | ||||||
| copyContent: "내용 복사" | copyContent: "내용 복사" | ||||||
| copyLink: "링크 복사" | copyLink: "링크 복사" | ||||||
|  | copyLinkRenote: "Renote 링크 복사" | ||||||
| delete: "삭제" | delete: "삭제" | ||||||
| deleteAndEdit: "삭제 후 편집" | deleteAndEdit: "삭제 후 편집" | ||||||
| deleteAndEditConfirm: "이 노트를 삭제한 뒤 다시 편집하시겠습니까? 이 노트에 대한 리액션, 리노트, 답글 또한 모두 삭제됩니다." | deleteAndEditConfirm: "이 노트를 삭제한 뒤 다시 편집하시겠습니까? 이 노트에 대한 리액션, 리노트, 답글 또한 모두 삭제됩니다." | ||||||
| @@ -355,7 +356,6 @@ invite: "초대" | |||||||
| driveCapacityPerLocalAccount: "로컬 유저 한 명당 드라이브 용량" | driveCapacityPerLocalAccount: "로컬 유저 한 명당 드라이브 용량" | ||||||
| driveCapacityPerRemoteAccount: "리모트 유저 한 명당 드라이브 용량" | driveCapacityPerRemoteAccount: "리모트 유저 한 명당 드라이브 용량" | ||||||
| inMb: "메가바이트 단위" | inMb: "메가바이트 단위" | ||||||
| iconUrl: "아이콘 URL" |  | ||||||
| bannerUrl: "배너 이미지 URL" | bannerUrl: "배너 이미지 URL" | ||||||
| backgroundImageUrl: "배경 이미지 URL" | backgroundImageUrl: "배경 이미지 URL" | ||||||
| basicInfo: "기본 정보" | basicInfo: "기본 정보" | ||||||
| @@ -416,6 +416,9 @@ totp: "인증 앱" | |||||||
| totpDescription: "인증 앱을 사용하여 일회성 비밀번호 입력" | totpDescription: "인증 앱을 사용하여 일회성 비밀번호 입력" | ||||||
| moderator: "모더레이터" | moderator: "모더레이터" | ||||||
| moderation: "모더레이션" | moderation: "모더레이션" | ||||||
|  | moderationNote: "모더레이션 노트" | ||||||
|  | addModerationNote: "모더레이션 노트 추가하기" | ||||||
|  | moderationLogs: "모더레이션 로그" | ||||||
| nUsersMentioned: "{n}명이 언급함" | nUsersMentioned: "{n}명이 언급함" | ||||||
| securityKeyAndPasskey: "보안 키 또는 패스 키" | securityKeyAndPasskey: "보안 키 또는 패스 키" | ||||||
| securityKey: "보안 키" | securityKey: "보안 키" | ||||||
| @@ -655,6 +658,7 @@ behavior: "동작" | |||||||
| sample: "예시" | sample: "예시" | ||||||
| abuseReports: "신고" | abuseReports: "신고" | ||||||
| reportAbuse: "신고" | reportAbuse: "신고" | ||||||
|  | reportAbuseRenote: "Renote를 신고" | ||||||
| reportAbuseOf: "{name}을 신고하기" | reportAbuseOf: "{name}을 신고하기" | ||||||
| fillAbuseReportDescription: "신고하려는 이유를 자세히 알려주세요. 특정 게시물을 신고할 때에는 게시물의 URL도 포함해 주세요." | fillAbuseReportDescription: "신고하려는 이유를 자세히 알려주세요. 특정 게시물을 신고할 때에는 게시물의 URL도 포함해 주세요." | ||||||
| abuseReported: "신고를 보냈습니다. 신고해 주셔서 감사합니다." | abuseReported: "신고를 보냈습니다. 신고해 주셔서 감사합니다." | ||||||
| @@ -1021,7 +1025,7 @@ retryAllQueuesConfirmText: "일시적으로 서버의 부하가 증가할 수  | |||||||
| enableChartsForRemoteUser: "리모트 유저의 차트를 생성" | enableChartsForRemoteUser: "리모트 유저의 차트를 생성" | ||||||
| enableChartsForFederatedInstances: "리모트 서버의 차트를 생성" | enableChartsForFederatedInstances: "리모트 서버의 차트를 생성" | ||||||
| showClipButtonInNoteFooter: "노트 동작에 클립을 추가" | showClipButtonInNoteFooter: "노트 동작에 클립을 추가" | ||||||
| largeNoteReactions: "노트의 리액션을 크게 표시" | reactionsDisplaySize: "리액션 표시 크기" | ||||||
| noteIdOrUrl: "노트 ID 및 URL" | noteIdOrUrl: "노트 ID 및 URL" | ||||||
| video: "동영상" | video: "동영상" | ||||||
| videos: "동영상" | videos: "동영상" | ||||||
| @@ -1103,6 +1107,21 @@ forYou: "당신에게" | |||||||
| currentAnnouncements: "현재 공지사항" | currentAnnouncements: "현재 공지사항" | ||||||
| pastAnnouncements: "과거 공지사항" | pastAnnouncements: "과거 공지사항" | ||||||
| youHaveUnreadAnnouncements: "읽지 않은 공지사항이 있습니다." | youHaveUnreadAnnouncements: "읽지 않은 공지사항이 있습니다." | ||||||
|  | useSecurityKey: "브라우저 또는 기기의 안내에 따라 보안 키 또는 패스키를 사용해 주십시오." | ||||||
|  | replies: "답글" | ||||||
|  | renotes: "리노트" | ||||||
|  | loadReplies: "답글 보기" | ||||||
|  | loadConversation: "대화 보기" | ||||||
|  | pinnedList: "고정해놓은 리스트" | ||||||
|  | keepScreenOn: "기기 화면을 항상 켜기" | ||||||
|  | verifiedLink: "이 링크의 소유자임이 확인되었습니다." | ||||||
|  | notifyNotes: "새 노트 알림 켜기" | ||||||
|  | unnotifyNotes: "새 노트 알림 끄기" | ||||||
|  | authentication: "인증" | ||||||
|  | showRenotes: "리노트 표시" | ||||||
|  | edited: "수정됨" | ||||||
|  | notificationRecieveConfig: "알림 설정" | ||||||
|  | mutualFollow: "맞팔로우" | ||||||
| _announcement: | _announcement: | ||||||
|   forExistingUsers: "기존 유저에게만 알림" |   forExistingUsers: "기존 유저에게만 알림" | ||||||
|   forExistingUsersDescription: "활성화하면 이 공지사항을 게시한 시점에서 이미 가입한 유저에게만 표시합니다. 비활성화하면 게시 후에 가입한 유저에게도 표시합니다." |   forExistingUsersDescription: "활성화하면 이 공지사항을 게시한 시점에서 이미 가입한 유저에게만 표시합니다. 비활성화하면 게시 후에 가입한 유저에게도 표시합니다." | ||||||
| @@ -1129,6 +1148,14 @@ _initialAccountSetting: | |||||||
|   laterAreYouSure: "초기 설정을 나중에 진행하시겠습니까?" |   laterAreYouSure: "초기 설정을 나중에 진행하시겠습니까?" | ||||||
| _serverRules: | _serverRules: | ||||||
|   description: "회원 가입 이전에 간단하게 표시할 서버 규칙입니다. 이용 약관의 요약으로 구성하는 것을 추천합니다." |   description: "회원 가입 이전에 간단하게 표시할 서버 규칙입니다. 이용 약관의 요약으로 구성하는 것을 추천합니다." | ||||||
|  | _serverSettings: | ||||||
|  |   iconUrl: "아이콘 URL" | ||||||
|  |   appIconUsageExample: "예를 들어, PWA나 스마트폰 홈 화면에 북마크로 추가되었을 때 등" | ||||||
|  |   appIconStyleRecommendation: "아이콘이 원형 또는 둥근 사각형으로 잘리는 경우가 있으므로, 가장자리 여백이 충분한 사진을 사용하는 것을 추천합니다." | ||||||
|  |   appIconResolutionMustBe: "해상도는 반드시 {resolution} 이어야 합니다." | ||||||
|  |   manifestJsonOverride: "manifest.json 오버라이드" | ||||||
|  |   shortName: "약칭" | ||||||
|  |   shortNameDescription: "서버의 정식 명칭이 긴 경우에, 대신에 표시할 수 있는 약칭이나 통칭." | ||||||
| _accountMigration: | _accountMigration: | ||||||
|   moveFrom: "다른 계정에서 이 계정으로 이사" |   moveFrom: "다른 계정에서 이 계정으로 이사" | ||||||
|   moveFromSub: "다른 계정에 대한 별칭을 생성" |   moveFromSub: "다른 계정에 대한 별칭을 생성" | ||||||
| @@ -1506,6 +1533,7 @@ _plugin: | |||||||
|   install: "플러그인 설치" |   install: "플러그인 설치" | ||||||
|   installWarn: "신뢰할 수 없는 플러그인은 설치하지 않는 것이 좋습니다." |   installWarn: "신뢰할 수 없는 플러그인은 설치하지 않는 것이 좋습니다." | ||||||
|   manage: "플러그인 관리" |   manage: "플러그인 관리" | ||||||
|  |   viewSource: "소스 보기" | ||||||
| _preferencesBackups: | _preferencesBackups: | ||||||
|   list: "생성한 백업" |   list: "생성한 백업" | ||||||
|   saveNew: "새 백업 만들기" |   saveNew: "새 백업 만들기" | ||||||
| @@ -1693,7 +1721,6 @@ _timelineTutorial: | |||||||
| _2fa: | _2fa: | ||||||
|   alreadyRegistered: "이미 설정이 완료되었습니다." |   alreadyRegistered: "이미 설정이 완료되었습니다." | ||||||
|   registerTOTP: "인증 앱 설정 시작" |   registerTOTP: "인증 앱 설정 시작" | ||||||
|   passwordToTOTP: "비밀번호를 입력하세요." |  | ||||||
|   step1: "먼저, {a}나 {b}등의 인증 앱을 사용 중인 디바이스에 설치합니다." |   step1: "먼저, {a}나 {b}등의 인증 앱을 사용 중인 디바이스에 설치합니다." | ||||||
|   step2: "그 후, 표시되어 있는 QR코드를 앱으로 스캔합니다." |   step2: "그 후, 표시되어 있는 QR코드를 앱으로 스캔합니다." | ||||||
|   step2Click: "QR 코드를 클릭하면 기기에 설치된 인증 앱에 등록할 수 있습니다." |   step2Click: "QR 코드를 클릭하면 기기에 설치된 인증 앱에 등록할 수 있습니다." | ||||||
| @@ -1705,7 +1732,6 @@ _2fa: | |||||||
|   securityKeyNotSupported: "이 브라우저는 보안 키를 지원하지 않습니다." |   securityKeyNotSupported: "이 브라우저는 보안 키를 지원하지 않습니다." | ||||||
|   registerTOTPBeforeKey: "보안 키 또는 패스키를 등록하려면 인증 앱을 등록하십시오." |   registerTOTPBeforeKey: "보안 키 또는 패스키를 등록하려면 인증 앱을 등록하십시오." | ||||||
|   securityKeyInfo: "FIDO2를 지원하는 하드웨어 보안 키 혹은 디바이스의 지문인식이나 화면잠금 PIN을 이용해서 로그인하도록 설정할 수 있습니다." |   securityKeyInfo: "FIDO2를 지원하는 하드웨어 보안 키 혹은 디바이스의 지문인식이나 화면잠금 PIN을 이용해서 로그인하도록 설정할 수 있습니다." | ||||||
|   chromePasskeyNotSupported: "현재 Chrome의 패스키는 지원되지 않습니다." |  | ||||||
|   registerSecurityKey: "보안 키 또는 패스키 등록" |   registerSecurityKey: "보안 키 또는 패스키 등록" | ||||||
|   securityKeyName: "키 이름 입력" |   securityKeyName: "키 이름 입력" | ||||||
|   tapSecurityKey: "브라우저의 지시에 따라 보안 키 또는 패스키를 등록하여 주십시오" |   tapSecurityKey: "브라우저의 지시에 따라 보안 키 또는 패스키를 등록하여 주십시오" | ||||||
| @@ -1993,6 +2019,10 @@ _notification: | |||||||
|   unreadAntennaNote: "안테나 {name}" |   unreadAntennaNote: "안테나 {name}" | ||||||
|   emptyPushNotificationMessage: "푸시 알림이 갱신되었습니다" |   emptyPushNotificationMessage: "푸시 알림이 갱신되었습니다" | ||||||
|   achievementEarned: "도전 과제를 달성했습니다" |   achievementEarned: "도전 과제를 달성했습니다" | ||||||
|  |   testNotification: "알림 테스트" | ||||||
|  |   checkNotificationBehavior: "알림 표시를 체크하기" | ||||||
|  |   sendTestNotification: "테스트 알림 보내기" | ||||||
|  |   notificationWillBeDisplayedLikeThis: "알림이 이렇게 표시됩니다" | ||||||
|   _types: |   _types: | ||||||
|     all: "전부" |     all: "전부" | ||||||
|     follow: "팔로잉" |     follow: "팔로잉" | ||||||
| @@ -2028,6 +2058,8 @@ _deck: | |||||||
|   introduction2: "나중에라도 화면 우측의 + 버튼을 눌러 새 칼럼을 추가할 수 있습니다." |   introduction2: "나중에라도 화면 우측의 + 버튼을 눌러 새 칼럼을 추가할 수 있습니다." | ||||||
|   widgetsIntroduction: "칼럼 메뉴의 \"위젯 편집\"에서 위젯을 추가해 주세요" |   widgetsIntroduction: "칼럼 메뉴의 \"위젯 편집\"에서 위젯을 추가해 주세요" | ||||||
|   useSimpleUiForNonRootPages: "루트 이외의 페이지로 접속한 경우 UI 간략화하기" |   useSimpleUiForNonRootPages: "루트 이외의 페이지로 접속한 경우 UI 간략화하기" | ||||||
|  |   usedAsMinWidthWhenFlexible: "'폭 자동 조정'이 활성화된 경우 최소 폭으로 사용됩니다" | ||||||
|  |   flexible: "폭 자동 조정" | ||||||
|   _columns: |   _columns: | ||||||
|     main: "메인" |     main: "메인" | ||||||
|     widgets: "위젯" |     widgets: "위젯" | ||||||
| @@ -2062,3 +2094,7 @@ _webhookSettings: | |||||||
|     renote: "누군가 내 글을 Renote했을 때" |     renote: "누군가 내 글을 Renote했을 때" | ||||||
|     reaction: "누군가 내 노트에 리액션했을 때" |     reaction: "누군가 내 노트에 리액션했을 때" | ||||||
|     mention: "누군가 나를 멘션했을 때" |     mention: "누군가 나를 멘션했을 때" | ||||||
|  | _moderationLogTypes: | ||||||
|  |   suspend: "정지" | ||||||
|  |   resetPassword: "비밀번호 재설정" | ||||||
|  |   createInvitation: "초대 코드 생성" | ||||||
|   | |||||||
| @@ -391,6 +391,8 @@ administration: "ການຈັດການ" | |||||||
| middle: "ປານກາງ" | middle: "ປານກາງ" | ||||||
| searchByGoogle: "ຄົ້ນຫາ" | searchByGoogle: "ຄົ້ນຫາ" | ||||||
| file: "ໄຟລ໌" | file: "ໄຟລ໌" | ||||||
|  | replies: "ຕອບໄປທີ" | ||||||
|  | renotes: "Renote" | ||||||
| _role: | _role: | ||||||
|   _priority: |   _priority: | ||||||
|     middle: "ປານກາງ" |     middle: "ປານກາງ" | ||||||
| @@ -461,3 +463,5 @@ _deck: | |||||||
|     mentions: "ກ່າວເຖິງ" |     mentions: "ກ່າວເຖິງ" | ||||||
| _webhookSettings: | _webhookSettings: | ||||||
|   name: "ຊື່" |   name: "ຊື່" | ||||||
|  | _moderationLogTypes: | ||||||
|  |   suspend: "ລະງັບ" | ||||||
|   | |||||||
| @@ -338,7 +338,6 @@ invite: "Uitnodigen" | |||||||
| driveCapacityPerLocalAccount: "Opslagruimte per lokale gebruiker" | driveCapacityPerLocalAccount: "Opslagruimte per lokale gebruiker" | ||||||
| driveCapacityPerRemoteAccount: "Opslagruimte per externe gebruiker" | driveCapacityPerRemoteAccount: "Opslagruimte per externe gebruiker" | ||||||
| inMb: "in megabytes" | inMb: "in megabytes" | ||||||
| iconUrl: "Pictogram URL" |  | ||||||
| bannerUrl: "Banner URL" | bannerUrl: "Banner URL" | ||||||
| backgroundImageUrl: "URL afbeelding" | backgroundImageUrl: "URL afbeelding" | ||||||
| basicInfo: "Basisinformatie" | basicInfo: "Basisinformatie" | ||||||
| @@ -427,6 +426,8 @@ windowMaximize: "Maximaliseren" | |||||||
| windowRestore: "Herstellen" | windowRestore: "Herstellen" | ||||||
| loggedInAsBot: "Momenteel als bot ingelogd" | loggedInAsBot: "Momenteel als bot ingelogd" | ||||||
| icon: "Avatar" | icon: "Avatar" | ||||||
|  | replies: "Antwoord" | ||||||
|  | renotes: "Herdelen" | ||||||
| _email: | _email: | ||||||
|   _follow: |   _follow: | ||||||
|     title: "volgde jou" |     title: "volgde jou" | ||||||
| @@ -493,3 +494,6 @@ _deck: | |||||||
|     mentions: "Vermeldingen" |     mentions: "Vermeldingen" | ||||||
| _webhookSettings: | _webhookSettings: | ||||||
|   name: "Naam" |   name: "Naam" | ||||||
|  | _moderationLogTypes: | ||||||
|  |   suspend: "Opschorten" | ||||||
|  |   resetPassword: "Wachtwoord terugzetten" | ||||||
|   | |||||||
| @@ -462,6 +462,8 @@ continue: "Fortsett" | |||||||
| youFollowing: "Følger" | youFollowing: "Følger" | ||||||
| options: "Alternativ" | options: "Alternativ" | ||||||
| icon: "Avatar" | icon: "Avatar" | ||||||
|  | replies: "Svar" | ||||||
|  | renotes: "Renote" | ||||||
| _initialAccountSetting: | _initialAccountSetting: | ||||||
|   theseSettingsCanEditLater: "Du kan endre disse innstillingene senere." |   theseSettingsCanEditLater: "Du kan endre disse innstillingene senere." | ||||||
| _achievements: | _achievements: | ||||||
| @@ -723,3 +725,5 @@ _deck: | |||||||
|     direct: "Direkte" |     direct: "Direkte" | ||||||
| _webhookSettings: | _webhookSettings: | ||||||
|   name: "Navn" |   name: "Navn" | ||||||
|  | _moderationLogTypes: | ||||||
|  |   suspend: "Suspender" | ||||||
|   | |||||||
| @@ -333,7 +333,6 @@ invite: "Zaproś" | |||||||
| driveCapacityPerLocalAccount: "Powierzchnia dyskowa na lokalnego użytkownika" | driveCapacityPerLocalAccount: "Powierzchnia dyskowa na lokalnego użytkownika" | ||||||
| driveCapacityPerRemoteAccount: "Powierzchnia dyskowa na zdalnego użytkownika" | driveCapacityPerRemoteAccount: "Powierzchnia dyskowa na zdalnego użytkownika" | ||||||
| inMb: "W megabajtach" | inMb: "W megabajtach" | ||||||
| iconUrl: "Adres URL ikony" |  | ||||||
| bannerUrl: "Adres URL banera" | bannerUrl: "Adres URL banera" | ||||||
| backgroundImageUrl: "Adres URL tła" | backgroundImageUrl: "Adres URL tła" | ||||||
| basicInfo: "Podstawowe informacje" | basicInfo: "Podstawowe informacje" | ||||||
| @@ -872,6 +871,8 @@ show: "Wyświetlanie" | |||||||
| color: "Kolor" | color: "Kolor" | ||||||
| youFollowing: "Śledzeni" | youFollowing: "Śledzeni" | ||||||
| icon: "Awatar" | icon: "Awatar" | ||||||
|  | replies: "Odpowiedz" | ||||||
|  | renotes: "Udostępnij" | ||||||
| _role: | _role: | ||||||
|   priority: "Priorytet" |   priority: "Priorytet" | ||||||
|   _priority: |   _priority: | ||||||
| @@ -924,6 +925,7 @@ _plugin: | |||||||
|   install: "Zainstaluj wtyczki" |   install: "Zainstaluj wtyczki" | ||||||
|   installWarn: "Nie instaluj niezaufanych wtyczek." |   installWarn: "Nie instaluj niezaufanych wtyczek." | ||||||
|   manage: "Zarządzanie wtyczkami" |   manage: "Zarządzanie wtyczkami" | ||||||
|  |   viewSource: "Zobacz źródło" | ||||||
| _preferencesBackups: | _preferencesBackups: | ||||||
|   list: "Utworzone kopie zapasowe" |   list: "Utworzone kopie zapasowe" | ||||||
|   saveNew: "Zapisz nową kopię zapasową" |   saveNew: "Zapisz nową kopię zapasową" | ||||||
| @@ -1400,3 +1402,6 @@ _webhookSettings: | |||||||
|     renote: "Po udostępnieniu wpisu" |     renote: "Po udostępnieniu wpisu" | ||||||
|     reaction: "Po otrzymaniu reakcji" |     reaction: "Po otrzymaniu reakcji" | ||||||
|     mention: "Po zostaniu wspomnianym" |     mention: "Po zostaniu wspomnianym" | ||||||
|  | _moderationLogTypes: | ||||||
|  |   suspend: "Zawieś" | ||||||
|  |   resetPassword: "Zresetuj hasło" | ||||||
|   | |||||||
| @@ -356,7 +356,6 @@ invite: "Convidar" | |||||||
| driveCapacityPerLocalAccount: "Capacidade do drive por usuário local" | driveCapacityPerLocalAccount: "Capacidade do drive por usuário local" | ||||||
| driveCapacityPerRemoteAccount: "Capacidade do drive por usuário remoto" | driveCapacityPerRemoteAccount: "Capacidade do drive por usuário remoto" | ||||||
| inMb: "Em ‘megabytes’" | inMb: "Em ‘megabytes’" | ||||||
| iconUrl: "URL da imagem do ícone (favicon, etc.)" |  | ||||||
| bannerUrl: "URL da imagem do ‘banner’" | bannerUrl: "URL da imagem do ‘banner’" | ||||||
| backgroundImageUrl: "URL da imagem de fundo" | backgroundImageUrl: "URL da imagem de fundo" | ||||||
| basicInfo: "Informações básicas" | basicInfo: "Informações básicas" | ||||||
| @@ -412,6 +411,7 @@ aboutMisskey: "Sobre Misskey" | |||||||
| administrator: "Administrador" | administrator: "Administrador" | ||||||
| token: "Símbolo" | token: "Símbolo" | ||||||
| 2fa: "Autenticação de dois fatores" | 2fa: "Autenticação de dois fatores" | ||||||
|  | setupOf2fa: "Configuração de autenticação de dois fatores" | ||||||
| totp: "Aplicativo Autenticador" | totp: "Aplicativo Autenticador" | ||||||
| totpDescription: "Digite a senha de uso único informado pelo aplicativo autenticador" | totpDescription: "Digite a senha de uso único informado pelo aplicativo autenticador" | ||||||
| moderator: "Moderador" | moderator: "Moderador" | ||||||
| @@ -919,6 +919,7 @@ pleaseSelect: "Por favor, selecione." | |||||||
| reverse: "Inversão" | reverse: "Inversão" | ||||||
| colored: "Colorido" | colored: "Colorido" | ||||||
| refreshInterval: "Intervalo de atualização" | refreshInterval: "Intervalo de atualização" | ||||||
|  | label: "Etiqueta" | ||||||
| type: "Tipo" | type: "Tipo" | ||||||
| speed: "Velocidade" | speed: "Velocidade" | ||||||
| slow: "Lento" | slow: "Lento" | ||||||
| @@ -984,6 +985,7 @@ rolesAssignedToMe: "Cargos atribuídos a mim" | |||||||
| unfavoriteConfirm: "Deseja realmente remover dos favoritos?" | unfavoriteConfirm: "Deseja realmente remover dos favoritos?" | ||||||
| drivecleaner: "Limpeza do drive" | drivecleaner: "Limpeza do drive" | ||||||
| retryAllQueuesConfirmTitle: "Gostaria de tentar novamente agora?" | retryAllQueuesConfirmTitle: "Gostaria de tentar novamente agora?" | ||||||
|  | reactionsDisplaySize: "Tamanho de exibição das reações" | ||||||
| reactionsList: "Reações" | reactionsList: "Reações" | ||||||
| renotesList: "Repostagens" | renotesList: "Repostagens" | ||||||
| leftTop: "Superior esquerdo" | leftTop: "Superior esquerdo" | ||||||
| @@ -1006,8 +1008,13 @@ rolesThatCanBeUsedThisEmojiAsReactionEmptyDescription: "Se nenhum cargo for espe | |||||||
| rolesThatCanBeUsedThisEmojiAsReactionPublicRoleWarn: "Estes cargos devem ser públicos." | rolesThatCanBeUsedThisEmojiAsReactionPublicRoleWarn: "Estes cargos devem ser públicos." | ||||||
| waitingForMailAuth: "Verificação de e-mail pendente " | waitingForMailAuth: "Verificação de e-mail pendente " | ||||||
| icon: "Avatar" | icon: "Avatar" | ||||||
|  | replies: "Responder" | ||||||
|  | renotes: "Repostar" | ||||||
|  | keepScreenOn: "Manter a tela do dispositivo sempre ligada" | ||||||
| _initialAccountSetting: | _initialAccountSetting: | ||||||
|   followUsers: "Siga usuários que lhe interessam para criar a sua linha do tempo." |   followUsers: "Siga usuários que lhe interessam para criar a sua linha do tempo." | ||||||
|  | _serverSettings: | ||||||
|  |   iconUrl: "URL do ícone" | ||||||
| _accountMigration: | _accountMigration: | ||||||
|   moveFromDescription: "Se você deseja migrar de outra conta para esta, é necessário criar um alias aqui. Por favor, insira a conta de origem da migração no seguinte formato: @username@server.example.com. Para excluir o alias, deixe o campo em branco e clique em salvar (não recomendado)." |   moveFromDescription: "Se você deseja migrar de outra conta para esta, é necessário criar um alias aqui. Por favor, insira a conta de origem da migração no seguinte formato: @username@server.example.com. Para excluir o alias, deixe o campo em branco e clique em salvar (não recomendado)." | ||||||
|   moveAccountDescription: "Você está migrando para uma nova conta.\n ・Seus seguidores irão automaticamente seguir a nova conta.\n ・Todas as suas conexões de seguidores nesta conta serão removidas.\n ・Você não poderá mais criar novas notas nesta conta.\n\nA migração dos seguidores é automática, mas a migração das pessoas que você segue deve ser feita manualmente. Antes de migrar, exporte quem você está seguindo nesta conta e, assim que migrar, importe essa lista na nova conta.\nO mesmo se aplica para listas, silenciamentos e bloqueios, que também devem ser migrados manualmente.\n\n(Esta descrição se refere ao comportamento do servidor Misskey v13.12.0 ou posterior. Outros softwares ActivityPub, como Mastodon, podem ter comportamentos diferentes.)" |   moveAccountDescription: "Você está migrando para uma nova conta.\n ・Seus seguidores irão automaticamente seguir a nova conta.\n ・Todas as suas conexões de seguidores nesta conta serão removidas.\n ・Você não poderá mais criar novas notas nesta conta.\n\nA migração dos seguidores é automática, mas a migração das pessoas que você segue deve ser feita manualmente. Antes de migrar, exporte quem você está seguindo nesta conta e, assim que migrar, importe essa lista na nova conta.\nO mesmo se aplica para listas, silenciamentos e bloqueios, que também devem ser migrados manualmente.\n\n(Esta descrição se refere ao comportamento do servidor Misskey v13.12.0 ou posterior. Outros softwares ActivityPub, como Mastodon, podem ter comportamentos diferentes.)" | ||||||
| @@ -1490,3 +1497,6 @@ _webhookSettings: | |||||||
|     follow: "Quando seguindo um usuário" |     follow: "Quando seguindo um usuário" | ||||||
|     followed: "Quando sendo seguido" |     followed: "Quando sendo seguido" | ||||||
|     renote: "Quando repostado" |     renote: "Quando repostado" | ||||||
|  | _moderationLogTypes: | ||||||
|  |   suspend: "Suspender" | ||||||
|  |   resetPassword: "Redefinir senha" | ||||||
|   | |||||||
| @@ -328,7 +328,6 @@ invite: "Invită" | |||||||
| driveCapacityPerLocalAccount: "Capacitatea Drive-ului per utilizator local" | driveCapacityPerLocalAccount: "Capacitatea Drive-ului per utilizator local" | ||||||
| driveCapacityPerRemoteAccount: "Capacitatea Drive-ului per utilizator extern" | driveCapacityPerRemoteAccount: "Capacitatea Drive-ului per utilizator extern" | ||||||
| inMb: "În megabytes" | inMb: "În megabytes" | ||||||
| iconUrl: "URL-ul iconiței" |  | ||||||
| bannerUrl: "URL-ul imaginii de banner" | bannerUrl: "URL-ul imaginii de banner" | ||||||
| backgroundImageUrl: "URL-ul imaginii de fundal" | backgroundImageUrl: "URL-ul imaginii de fundal" | ||||||
| basicInfo: "Informații de bază" | basicInfo: "Informații de bază" | ||||||
| @@ -631,6 +630,8 @@ searchByGoogle: "Caută" | |||||||
| file: "Fișiere" | file: "Fișiere" | ||||||
| show: "Arată" | show: "Arată" | ||||||
| icon: "Avatar" | icon: "Avatar" | ||||||
|  | replies: "Răspunde" | ||||||
|  | renotes: "Re-notează" | ||||||
| _role: | _role: | ||||||
|   _priority: |   _priority: | ||||||
|     middle: "Mediu" |     middle: "Mediu" | ||||||
| @@ -703,3 +704,6 @@ _deck: | |||||||
|     mentions: "Mențiuni" |     mentions: "Mențiuni" | ||||||
| _webhookSettings: | _webhookSettings: | ||||||
|   name: "Nume" |   name: "Nume" | ||||||
|  | _moderationLogTypes: | ||||||
|  |   suspend: "Suspendă" | ||||||
|  |   resetPassword: "Resetează parola" | ||||||
|   | |||||||
| @@ -354,7 +354,6 @@ invite: "Пригласить" | |||||||
| driveCapacityPerLocalAccount: "Объём диска на одного локального пользователя" | driveCapacityPerLocalAccount: "Объём диска на одного локального пользователя" | ||||||
| driveCapacityPerRemoteAccount: "Объём диска на одного пользователя с другого сайта" | driveCapacityPerRemoteAccount: "Объём диска на одного пользователя с другого сайта" | ||||||
| inMb: "В мегабайтах" | inMb: "В мегабайтах" | ||||||
| iconUrl: "Ссылка на аватар" |  | ||||||
| bannerUrl: "Ссылка на изображение в шапке" | bannerUrl: "Ссылка на изображение в шапке" | ||||||
| backgroundImageUrl: "Ссылка на фоновое изображение" | backgroundImageUrl: "Ссылка на фоновое изображение" | ||||||
| basicInfo: "Общая информация" | basicInfo: "Общая информация" | ||||||
| @@ -1017,7 +1016,6 @@ retryAllQueuesConfirmTitle: "Хотите попробовать ещё раз?" | |||||||
| retryAllQueuesConfirmText: "Нагрузка на сервер может увеличиться" | retryAllQueuesConfirmText: "Нагрузка на сервер может увеличиться" | ||||||
| enableChartsForRemoteUser: "Создание диаграмм для удалённых пользователей" | enableChartsForRemoteUser: "Создание диаграмм для удалённых пользователей" | ||||||
| enableChartsForFederatedInstances: "Создание диаграмм для удалённых серверов" | enableChartsForFederatedInstances: "Создание диаграмм для удалённых серверов" | ||||||
| largeNoteReactions: "Показывать большие реакции на заметки" |  | ||||||
| noteIdOrUrl: "ID или ссылка на заметку" | noteIdOrUrl: "ID или ссылка на заметку" | ||||||
| video: "Видео" | video: "Видео" | ||||||
| videos: "Видео" | videos: "Видео" | ||||||
| @@ -1067,6 +1065,8 @@ unused: "Неиспользуемый" | |||||||
| expired: "Срок действия приглашения истёк" | expired: "Срок действия приглашения истёк" | ||||||
| doYouAgree: "Согласны?" | doYouAgree: "Согласны?" | ||||||
| icon: "Аватар" | icon: "Аватар" | ||||||
|  | replies: "Ответить" | ||||||
|  | renotes: "Репост" | ||||||
| _initialAccountSetting: | _initialAccountSetting: | ||||||
|   accountCreated: "Аккаунт успешно создан!" |   accountCreated: "Аккаунт успешно создан!" | ||||||
|   letsStartAccountSetup: "Давайте настроим вашу учётную запись." |   letsStartAccountSetup: "Давайте настроим вашу учётную запись." | ||||||
| @@ -1074,6 +1074,8 @@ _initialAccountSetting: | |||||||
|   privacySetting: "Настройки конфиденциальности" |   privacySetting: "Настройки конфиденциальности" | ||||||
|   initialAccountSettingCompleted: "Первоначальная настройка успешно завершена!" |   initialAccountSettingCompleted: "Первоначальная настройка успешно завершена!" | ||||||
|   skipAreYouSure: "Пропустить настройку?" |   skipAreYouSure: "Пропустить настройку?" | ||||||
|  | _serverSettings: | ||||||
|  |   iconUrl: "Адрес на иконку роли" | ||||||
| _achievements: | _achievements: | ||||||
|   earnedAt: "Разблокировано в" |   earnedAt: "Разблокировано в" | ||||||
|   _types: |   _types: | ||||||
| @@ -1425,6 +1427,7 @@ _plugin: | |||||||
|   install: "Установка расширений" |   install: "Установка расширений" | ||||||
|   installWarn: "Пожалуйста, не устанавливайте расширения, которым не доверяете." |   installWarn: "Пожалуйста, не устанавливайте расширения, которым не доверяете." | ||||||
|   manage: "Управление расширениями" |   manage: "Управление расширениями" | ||||||
|  |   viewSource: "Просмотр исходника" | ||||||
| _preferencesBackups: | _preferencesBackups: | ||||||
|   list: "Существующие резервные копии" |   list: "Существующие резервные копии" | ||||||
|   saveNew: "Создать резервную копию" |   saveNew: "Создать резервную копию" | ||||||
| @@ -1606,7 +1609,6 @@ _timelineTutorial: | |||||||
| _2fa: | _2fa: | ||||||
|   alreadyRegistered: "Двухфакторная аутентификация уже настроена." |   alreadyRegistered: "Двухфакторная аутентификация уже настроена." | ||||||
|   registerTOTP: "Начните настраивать приложение-аутентификатор" |   registerTOTP: "Начните настраивать приложение-аутентификатор" | ||||||
|   passwordToTOTP: "Пожалуйста, введите свой пароль" |  | ||||||
|   step1: "Прежде всего, установите на устройство приложение для аутентификации, например, {a} или {b}." |   step1: "Прежде всего, установите на устройство приложение для аутентификации, например, {a} или {b}." | ||||||
|   step2: "Далее отсканируйте отображаемый QR-код при помощи приложения." |   step2: "Далее отсканируйте отображаемый QR-код при помощи приложения." | ||||||
|   step2Click: "Нажав на QR-код, вы можете зарегистрироваться с помощью приложения для аутентификации или брелка для ключей, установленного на вашем устройстве." |   step2Click: "Нажав на QR-код, вы можете зарегистрироваться с помощью приложения для аутентификации или брелка для ключей, установленного на вашем устройстве." | ||||||
| @@ -1616,7 +1618,6 @@ _2fa: | |||||||
|   securityKeyNotSupported: "Ваш браузер не поддерживает ключи безопасности." |   securityKeyNotSupported: "Ваш браузер не поддерживает ключи безопасности." | ||||||
|   registerTOTPBeforeKey: "Чтобы зарегистрировать ключ безопасности и пароль, сначала настройте приложение аутентификации." |   registerTOTPBeforeKey: "Чтобы зарегистрировать ключ безопасности и пароль, сначала настройте приложение аутентификации." | ||||||
|   securityKeyInfo: "Вы можете настроить вход с помощью аппаратного ключа безопасности, поддерживающего FIDO2, или отпечатка пальца или PIN-кода на устройстве." |   securityKeyInfo: "Вы можете настроить вход с помощью аппаратного ключа безопасности, поддерживающего FIDO2, или отпечатка пальца или PIN-кода на устройстве." | ||||||
|   chromePasskeyNotSupported: "В настоящее время Chrome не поддерживает пароль-ключи." |  | ||||||
|   registerSecurityKey: "Зарегистрируйте ключ безопасности ・Passkey" |   registerSecurityKey: "Зарегистрируйте ключ безопасности ・Passkey" | ||||||
|   securityKeyName: "Введите имя для ключа" |   securityKeyName: "Введите имя для ключа" | ||||||
|   tapSecurityKey: "Пожалуйста, следуйте инструкциям в вашем браузере, чтобы зарегистрировать свой ключ безопасности или пароль" |   tapSecurityKey: "Пожалуйста, следуйте инструкциям в вашем браузере, чтобы зарегистрировать свой ключ безопасности или пароль" | ||||||
| @@ -1950,3 +1951,6 @@ _webhookSettings: | |||||||
|   createWebhook: "Создать вебхук" |   createWebhook: "Создать вебхук" | ||||||
|   name: "Название" |   name: "Название" | ||||||
|   active: "Вкл." |   active: "Вкл." | ||||||
|  | _moderationLogTypes: | ||||||
|  |   suspend: "Заморозить" | ||||||
|  |   resetPassword: "Сброс пароля:" | ||||||
|   | |||||||
| @@ -337,7 +337,6 @@ invite: "Pozvať" | |||||||
| driveCapacityPerLocalAccount: "Kapacita disku pre používateľa" | driveCapacityPerLocalAccount: "Kapacita disku pre používateľa" | ||||||
| driveCapacityPerRemoteAccount: "Kapacita disku pre vzdialeného používateľa" | driveCapacityPerRemoteAccount: "Kapacita disku pre vzdialeného používateľa" | ||||||
| inMb: "V megabajtoch" | inMb: "V megabajtoch" | ||||||
| iconUrl: "Favicon URL" |  | ||||||
| bannerUrl: "URL obrázku bannera" | bannerUrl: "URL obrázku bannera" | ||||||
| backgroundImageUrl: "URL obrázku pozadia" | backgroundImageUrl: "URL obrázku pozadia" | ||||||
| basicInfo: "Základné informácie" | basicInfo: "Základné informácie" | ||||||
| @@ -920,6 +919,8 @@ color: "Farba" | |||||||
| horizontal: "Strana" | horizontal: "Strana" | ||||||
| youFollowing: "Sledované" | youFollowing: "Sledované" | ||||||
| icon: "Avatar" | icon: "Avatar" | ||||||
|  | replies: "Odpovedať" | ||||||
|  | renotes: "Preposlať" | ||||||
| _role: | _role: | ||||||
|   priority: "Priorita" |   priority: "Priorita" | ||||||
|   _priority: |   _priority: | ||||||
| @@ -977,6 +978,7 @@ _plugin: | |||||||
|   install: "Inštalova pluginy" |   install: "Inštalova pluginy" | ||||||
|   installWarn: "Prosím neinštalujte nedôveryhodné pluginy." |   installWarn: "Prosím neinštalujte nedôveryhodné pluginy." | ||||||
|   manage: "Spravovanie pluginov" |   manage: "Spravovanie pluginov" | ||||||
|  |   viewSource: "Ukázať zdroj" | ||||||
| _preferencesBackups: | _preferencesBackups: | ||||||
|   list: "Vytvorené zálohy" |   list: "Vytvorené zálohy" | ||||||
|   saveNew: "Uložiť novú" |   saveNew: "Uložiť novú" | ||||||
| @@ -1449,3 +1451,6 @@ _deck: | |||||||
| _webhookSettings: | _webhookSettings: | ||||||
|   name: "Názov" |   name: "Názov" | ||||||
|   active: "Zapnuté" |   active: "Zapnuté" | ||||||
|  | _moderationLogTypes: | ||||||
|  |   suspend: "Zmraziť" | ||||||
|  |   resetPassword: "Resetovať heslo" | ||||||
|   | |||||||
| @@ -337,7 +337,6 @@ registration: "Registrera" | |||||||
| enableRegistration: "Aktivera registrering av nya användare" | enableRegistration: "Aktivera registrering av nya användare" | ||||||
| invite: "Inbjudan" | invite: "Inbjudan" | ||||||
| inMb: "I megabyte" | inMb: "I megabyte" | ||||||
| iconUrl: "URL till profilbilden" |  | ||||||
| bannerUrl: "URL till banner-bilden" | bannerUrl: "URL till banner-bilden" | ||||||
| basicInfo: "Grundläggande info" | basicInfo: "Grundläggande info" | ||||||
| pinnedUsers: "Fästa användare" | pinnedUsers: "Fästa användare" | ||||||
| @@ -486,6 +485,8 @@ pleaseDonate: "Misskey är en gratis programvara som används på {host}. Donera | |||||||
| resetPasswordConfirm: "Återställ verkligen ditt lösenord?" | resetPasswordConfirm: "Återställ verkligen ditt lösenord?" | ||||||
| dataSaver: "Databesparing" | dataSaver: "Databesparing" | ||||||
| icon: "Profilbild" | icon: "Profilbild" | ||||||
|  | replies: "Svara" | ||||||
|  | renotes: "Omnotera" | ||||||
| _achievements: | _achievements: | ||||||
|   _types: |   _types: | ||||||
|     _open3windows: |     _open3windows: | ||||||
| @@ -509,7 +510,6 @@ _sfx: | |||||||
|   chat: "Chatt" |   chat: "Chatt" | ||||||
|   antenna: "Antenner" |   antenna: "Antenner" | ||||||
| _2fa: | _2fa: | ||||||
|   passwordToTOTP: "Skriv in ditt lösenord" |  | ||||||
|   renewTOTPCancel: "Nej tack" |   renewTOTPCancel: "Nej tack" | ||||||
| _antennaSources: | _antennaSources: | ||||||
|   all: "Alla noter" |   all: "Alla noter" | ||||||
| @@ -573,3 +573,6 @@ _deck: | |||||||
| _webhookSettings: | _webhookSettings: | ||||||
|   name: "Namn" |   name: "Namn" | ||||||
|   active: "Aktiverad" |   active: "Aktiverad" | ||||||
|  | _moderationLogTypes: | ||||||
|  |   suspend: "Suspendera" | ||||||
|  |   resetPassword: "Återställ Lösenord" | ||||||
|   | |||||||
| @@ -45,6 +45,7 @@ pin: "ปักหมุดไปยังโปรไฟล์" | |||||||
| unpin: "เลิกปักหมุดจากโปรไฟล์" | unpin: "เลิกปักหมุดจากโปรไฟล์" | ||||||
| copyContent: "คัดลอกเนื้อหา" | copyContent: "คัดลอกเนื้อหา" | ||||||
| copyLink: "คัดลอกลิงก์" | copyLink: "คัดลอกลิงก์" | ||||||
|  | copyLinkRenote: "คัดลอกลิงก์รีโน้ต" | ||||||
| delete: "ลบ" | delete: "ลบ" | ||||||
| deleteAndEdit: "ลบและแก้ไข" | deleteAndEdit: "ลบและแก้ไข" | ||||||
| deleteAndEditConfirm: "นายแน่ใจแล้วเหรอ? ว่าต้องการลบโน้ตนี้และแก้ไข คุณอาจจะสูญเสียการโต้ตอบ, โน้ต, และการตอบกลับทั้งหมดได้นะ" | deleteAndEditConfirm: "นายแน่ใจแล้วเหรอ? ว่าต้องการลบโน้ตนี้และแก้ไข คุณอาจจะสูญเสียการโต้ตอบ, โน้ต, และการตอบกลับทั้งหมดได้นะ" | ||||||
| @@ -355,7 +356,6 @@ invite: "เชิญชวน" | |||||||
| driveCapacityPerLocalAccount: "ความจุของไดรฟ์ต่อผู้ใช้ภายในเครื่อง" | driveCapacityPerLocalAccount: "ความจุของไดรฟ์ต่อผู้ใช้ภายในเครื่อง" | ||||||
| driveCapacityPerRemoteAccount: "ความจุของไดรฟ์ต่อผู้ใช้ระยะไกล" | driveCapacityPerRemoteAccount: "ความจุของไดรฟ์ต่อผู้ใช้ระยะไกล" | ||||||
| inMb: "เป็นเมกะไบต์" | inMb: "เป็นเมกะไบต์" | ||||||
| iconUrl: "ไอคอน URL" |  | ||||||
| bannerUrl: "URL รูปภาพแบนเนอร์" | bannerUrl: "URL รูปภาพแบนเนอร์" | ||||||
| backgroundImageUrl: "URL ภาพพื้นหลัง" | backgroundImageUrl: "URL ภาพพื้นหลัง" | ||||||
| basicInfo: "ข้อมูลเบื้องต้น" | basicInfo: "ข้อมูลเบื้องต้น" | ||||||
| @@ -411,10 +411,14 @@ aboutMisskey: "เกี่ยวกับ Misskey" | |||||||
| administrator: "ผู้ดูแลระบบ" | administrator: "ผู้ดูแลระบบ" | ||||||
| token: "โทเค็น" | token: "โทเค็น" | ||||||
| 2fa: "การยืนยันตัวตนแบบสองชั้น" | 2fa: "การยืนยันตัวตนแบบสองชั้น" | ||||||
|  | setupOf2fa: "ตั้งค่าการยืนยันตัวตนแบบสองชั้น" | ||||||
| totp: "แอป Authenticator" | totp: "แอป Authenticator" | ||||||
| totpDescription: "ใช้แอปยืนยันตัวตนเพื่อป้อนรหัสผ่านแบบใช้ครั้งเดียว" | totpDescription: "ใช้แอปยืนยันตัวตนเพื่อป้อนรหัสผ่านแบบใช้ครั้งเดียว" | ||||||
| moderator: "ผู้ควบคุม" | moderator: "ผู้ควบคุม" | ||||||
| moderation: "การกลั่นกรอง" | moderation: "การกลั่นกรอง" | ||||||
|  | moderationNote: "โน้ตการกลั่นกรอง" | ||||||
|  | addModerationNote: "เพิ่มโน้ตการกลั่นกรอง" | ||||||
|  | moderationLogs: "บันทึกการกลั่นกรอง" | ||||||
| nUsersMentioned: "กล่าวถึงโดยผู้ใช้ {n} รายนี้" | nUsersMentioned: "กล่าวถึงโดยผู้ใช้ {n} รายนี้" | ||||||
| securityKeyAndPasskey: "ความปลอดภัยและรหัสผ่าน" | securityKeyAndPasskey: "ความปลอดภัยและรหัสผ่าน" | ||||||
| securityKey: "กุญแจความปลอดภัย" | securityKey: "กุญแจความปลอดภัย" | ||||||
| @@ -654,6 +658,7 @@ behavior: "พฤติกรรม" | |||||||
| sample: "ตัวอย่าง" | sample: "ตัวอย่าง" | ||||||
| abuseReports: "รายงาน" | abuseReports: "รายงาน" | ||||||
| reportAbuse: "รายงาน" | reportAbuse: "รายงาน" | ||||||
|  | reportAbuseRenote: "รายงานรีโน้ต" | ||||||
| reportAbuseOf: "รายงาน {ชื่อ}" | reportAbuseOf: "รายงาน {ชื่อ}" | ||||||
| fillAbuseReportDescription: "กรุณากรอกรายละเอียดเกี่ยวกับรายงานนี้ หากเป็นเรื่องเกี่ยวกับโน้ตโดยเฉพาะ ได้โปรดระบุ URL" | fillAbuseReportDescription: "กรุณากรอกรายละเอียดเกี่ยวกับรายงานนี้ หากเป็นเรื่องเกี่ยวกับโน้ตโดยเฉพาะ ได้โปรดระบุ URL" | ||||||
| abuseReported: "เราได้ส่งรายงานของคุณไปแล้ว ขอบคุณมากๆนะ" | abuseReported: "เราได้ส่งรายงานของคุณไปแล้ว ขอบคุณมากๆนะ" | ||||||
| @@ -706,6 +711,7 @@ lockedAccountInfo: "เว้นแต่ว่าคุณจะต้องต | |||||||
| alwaysMarkSensitive: "ทำเครื่องหมายเป็น NSFW เป็นค่าเริ่มต้น" | alwaysMarkSensitive: "ทำเครื่องหมายเป็น NSFW เป็นค่าเริ่มต้น" | ||||||
| loadRawImages: "โหลดภาพต้นฉบับแทนการแสดงภาพขนาดย่อ" | loadRawImages: "โหลดภาพต้นฉบับแทนการแสดงภาพขนาดย่อ" | ||||||
| disableShowingAnimatedImages: "ไม่ต้องเล่นภาพเคลื่อนไหว" | disableShowingAnimatedImages: "ไม่ต้องเล่นภาพเคลื่อนไหว" | ||||||
|  | highlightSensitiveMedia: "ไฮไลท์สื่อที่ละเอียดอ่อน" | ||||||
| verificationEmailSent: "ส่งอีเมลยืนยันแล้วนะ ได้โปรดกรุณาไปที่ลิงก์ที่รวมไว้เพื่อทำการตรวจสอบให้เสร็จสิ้น" | verificationEmailSent: "ส่งอีเมลยืนยันแล้วนะ ได้โปรดกรุณาไปที่ลิงก์ที่รวมไว้เพื่อทำการตรวจสอบให้เสร็จสิ้น" | ||||||
| notSet: "ไม่ได้ตั้งค่า" | notSet: "ไม่ได้ตั้งค่า" | ||||||
| emailVerified: "อีเมลได้รับการยืนยันแล้ว" | emailVerified: "อีเมลได้รับการยืนยันแล้ว" | ||||||
| @@ -1020,7 +1026,7 @@ retryAllQueuesConfirmText: "สิ่งนี้จะเพิ่มการ | |||||||
| enableChartsForRemoteUser: "สร้างแผนภูมิข้อมูลผู้ใช้ระยะไกล" | enableChartsForRemoteUser: "สร้างแผนภูมิข้อมูลผู้ใช้ระยะไกล" | ||||||
| enableChartsForFederatedInstances: "สร้างแผนภูมิข้อมูลอินสแตนซ์ระยะไกล" | enableChartsForFederatedInstances: "สร้างแผนภูมิข้อมูลอินสแตนซ์ระยะไกล" | ||||||
| showClipButtonInNoteFooter: "เพิ่ม \"คลิป\" เพื่อบันทึกเมนูการทำงาน" | showClipButtonInNoteFooter: "เพิ่ม \"คลิป\" เพื่อบันทึกเมนูการทำงาน" | ||||||
| largeNoteReactions: "ขยายรีแอคชั่นการแสดงผล" | reactionsDisplaySize: "รีแอคชั่นแสดงผลขนาด" | ||||||
| noteIdOrUrl: "โน้ต ID หรือ URL" | noteIdOrUrl: "โน้ต ID หรือ URL" | ||||||
| video: "วีดีโอ" | video: "วีดีโอ" | ||||||
| videos: "วีดีโอ" | videos: "วีดีโอ" | ||||||
| @@ -1099,7 +1105,26 @@ iHaveReadXCarefullyAndAgree: "ฉันได้อ่านข้อควา | |||||||
| dialog: "ไดอะล็อก" | dialog: "ไดอะล็อก" | ||||||
| icon: "ไอคอน" | icon: "ไอคอน" | ||||||
| forYou: "สำหรับคุณ" | forYou: "สำหรับคุณ" | ||||||
|  | currentAnnouncements: "ประกาศในปัจจุบัน" | ||||||
|  | pastAnnouncements: "ประกาศที่ผ่านมา" | ||||||
|  | youHaveUnreadAnnouncements: "มีการประกาศที่ยังไม่ได้อ่าน" | ||||||
|  | replies: "ตอบกลับ" | ||||||
|  | renotes: "รีโน้ต" | ||||||
|  | loadReplies: "แสดงการตอบกลับ" | ||||||
|  | loadConversation: "แสดงบทสนทนา" | ||||||
|  | pinnedList: "รายการที่ปักหมุดไว้แล้ว" | ||||||
|  | keepScreenOn: "เปิดหน้าจอไว้" | ||||||
|  | notifyNotes: "แจ้งเตือนเกี่ยวกับโพสต์ใหม่" | ||||||
|  | unnotifyNotes: "หยุดการแจ้งเตือนเกี่ยวกับโน้ตใหม่" | ||||||
|  | authentication: "การตรวจสอบสิทธิ์" | ||||||
|  | dateAndTime: "เวลาประทับ" | ||||||
|  | showRenotes: "แสดงรีโน้ต" | ||||||
|  | edited: "แก้ไขแล้ว" | ||||||
|  | notificationRecieveConfig: "การตั้งค่าการแจ้งเตือน" | ||||||
|  | mutualFollow: "ติดตามซึ่งกันและกัน" | ||||||
|  | fileAttachedOnly: "เฉพาะโน้ตที่มีไฟล์เท่านั้น" | ||||||
| _announcement: | _announcement: | ||||||
|  |   forExistingUsers: "ผู้ใช้งานที่มีอยู่เท่านั้น" | ||||||
|   forExistingUsersDescription: "การประกาศนี้จะแสดงต่อผู้ใช้ที่มีอยู่ ณ จุดที่เผยแพร่นั้นๆถ้าหากเปิดใช้งาน ถ้าหากปิดใช้งานผู้ที่กำลังสมัครใหม่หลังจากโพสต์แล้วนั้นก็จะเห็นเช่นกัน" |   forExistingUsersDescription: "การประกาศนี้จะแสดงต่อผู้ใช้ที่มีอยู่ ณ จุดที่เผยแพร่นั้นๆถ้าหากเปิดใช้งาน ถ้าหากปิดใช้งานผู้ที่กำลังสมัครใหม่หลังจากโพสต์แล้วนั้นก็จะเห็นเช่นกัน" | ||||||
|   needConfirmationToReadDescription: "ข้อความแจ้งแยก ถ้าหากต้องการเพื่อยืนยันว่ากำลังทำเครื่องหมายประกาศนี้ว่าอ่านแล้วจะแสดงขึ้นถ้าหากเปิดใช้งาน การประกาศนั้นจะไม่รวมอยู่ในฟังก์ชั่นว่า \"ทำเครื่องหมายทั้งหมดว่าอ่านแล้ว\"" |   needConfirmationToReadDescription: "ข้อความแจ้งแยก ถ้าหากต้องการเพื่อยืนยันว่ากำลังทำเครื่องหมายประกาศนี้ว่าอ่านแล้วจะแสดงขึ้นถ้าหากเปิดใช้งาน การประกาศนั้นจะไม่รวมอยู่ในฟังก์ชั่นว่า \"ทำเครื่องหมายทั้งหมดว่าอ่านแล้ว\"" | ||||||
|   end: "ประกาศเก็บถาวร" |   end: "ประกาศเก็บถาวร" | ||||||
| @@ -1123,6 +1148,9 @@ _initialAccountSetting: | |||||||
|   laterAreYouSure: "ต้องการตั้งค่าโปรไฟล์ในภายหลังจริงๆอย่างงั้นหรอ?" |   laterAreYouSure: "ต้องการตั้งค่าโปรไฟล์ในภายหลังจริงๆอย่างงั้นหรอ?" | ||||||
| _serverRules: | _serverRules: | ||||||
|   description: "ชุดของกฎที่จะแสดงก่อนการลงทะเบียนเราขอแนะนำให้ตั้งค่าสรุปข้อกำหนดในการให้บริการ" |   description: "ชุดของกฎที่จะแสดงก่อนการลงทะเบียนเราขอแนะนำให้ตั้งค่าสรุปข้อกำหนดในการให้บริการ" | ||||||
|  | _serverSettings: | ||||||
|  |   iconUrl: "ไอคอน URL" | ||||||
|  |   shortName: "ชื่อย่อ" | ||||||
| _accountMigration: | _accountMigration: | ||||||
|   moveFrom: "ย้ายข้อมูลบัญชีอื่นไปยังอีกบัญชีนี้หนึ่ง" |   moveFrom: "ย้ายข้อมูลบัญชีอื่นไปยังอีกบัญชีนี้หนึ่ง" | ||||||
|   moveFromSub: "สร้างนามแฝงไปยังบัญชีอื่น" |   moveFromSub: "สร้างนามแฝงไปยังบัญชีอื่น" | ||||||
| @@ -1377,6 +1405,8 @@ _achievements: | |||||||
|       title: "Brain Diver" |       title: "Brain Diver" | ||||||
|       description: "โพสต์ลิงก์ไปยัง Brain Diver" |       description: "โพสต์ลิงก์ไปยัง Brain Diver" | ||||||
|       flavor: "Misskey-Misskey La-Tu-Ma" |       flavor: "Misskey-Misskey La-Tu-Ma" | ||||||
|  |     _smashTestNotificationButton: | ||||||
|  |       title: "ทดสอบโอเวอร์โฟลว์" | ||||||
| _role: | _role: | ||||||
|   new: "บทบาทใหม่" |   new: "บทบาทใหม่" | ||||||
|   edit: "แก้ไขบทบาท" |   edit: "แก้ไขบทบาท" | ||||||
| @@ -1415,6 +1445,7 @@ _role: | |||||||
|     gtlAvailable: "การดูไทม์ไลน์ทั่วโลก" |     gtlAvailable: "การดูไทม์ไลน์ทั่วโลก" | ||||||
|     ltlAvailable: "การดูไทม์ไลน์ในท้องถิ่น" |     ltlAvailable: "การดูไทม์ไลน์ในท้องถิ่น" | ||||||
|     canPublicNote: "สามารถส่งโน้ตสาธารณะ" |     canPublicNote: "สามารถส่งโน้ตสาธารณะ" | ||||||
|  |     canEditNote: "กำลังแก้ไขโน้ต" | ||||||
|     canInvite: "สร้างรหัสเชิญอินสแตนซ์" |     canInvite: "สร้างรหัสเชิญอินสแตนซ์" | ||||||
|     inviteLimit: "จำกัดการเชิญ" |     inviteLimit: "จำกัดการเชิญ" | ||||||
|     inviteLimitCycle: "จำกัดการเชิญไว้คูลดาวน์" |     inviteLimitCycle: "จำกัดการเชิญไว้คูลดาวน์" | ||||||
| @@ -1500,6 +1531,7 @@ _plugin: | |||||||
|   install: "ติดตั้งปลั๊กอิน" |   install: "ติดตั้งปลั๊กอิน" | ||||||
|   installWarn: "กรุณาอย่าติดตั้งปลั๊กอินที่ไม่น่าเชื่อถือนะคะ" |   installWarn: "กรุณาอย่าติดตั้งปลั๊กอินที่ไม่น่าเชื่อถือนะคะ" | ||||||
|   manage: "จัดการปลั๊กอิน" |   manage: "จัดการปลั๊กอิน" | ||||||
|  |   viewSource: "ดูต้นฉบับ" | ||||||
| _preferencesBackups: | _preferencesBackups: | ||||||
|   list: "สร้างการสำรองข้อมูล" |   list: "สร้างการสำรองข้อมูล" | ||||||
|   saveNew: "บันทึกใหม่" |   saveNew: "บันทึกใหม่" | ||||||
| @@ -1687,17 +1719,16 @@ _timelineTutorial: | |||||||
| _2fa: | _2fa: | ||||||
|   alreadyRegistered: "คุณได้ลงทะเบียนอุปกรณ์ยืนยันตัวตนแบบ 2 ชั้นแล้ว" |   alreadyRegistered: "คุณได้ลงทะเบียนอุปกรณ์ยืนยันตัวตนแบบ 2 ชั้นแล้ว" | ||||||
|   registerTOTP: "ลงทะเบียนแอพตัวตรวจสอบสิทธิ์" |   registerTOTP: "ลงทะเบียนแอพตัวตรวจสอบสิทธิ์" | ||||||
|   passwordToTOTP: "กรอกรหัสผ่าน" |  | ||||||
|   step1: "ขั้นตอนแรก ติดตั้งแอปยืนยันตัวตน (เช่น {a} หรือ {b}) บนอุปกรณ์ของคุณ" |   step1: "ขั้นตอนแรก ติดตั้งแอปยืนยันตัวตน (เช่น {a} หรือ {b}) บนอุปกรณ์ของคุณ" | ||||||
|   step2: "จากนั้นสแกนรหัส QR ที่แสดงบนหน้าจอนี้" |   step2: "จากนั้นสแกนรหัส QR ที่แสดงบนหน้าจอนี้" | ||||||
|   step2Click: "การคลิกที่รหัส QR นี้จะช่วยให้คุณนั้นสามารถลงทะเบียน 2FA กับคีย์ความปลอดภัยหรือแอปตรวจสอบความถูกต้องของโทรศัพท์ได้" |   step2Click: "การคลิกที่รหัส QR นี้จะช่วยให้คุณนั้นสามารถลงทะเบียน 2FA กับคีย์ความปลอดภัยหรือแอปตรวจสอบความถูกต้องของโทรศัพท์ได้" | ||||||
|   step3Title: "ป้อนรหัสยืนยัน" |   step3Title: "ป้อนรหัสยืนยัน" | ||||||
|   step3: "ป้อนโทเค็นที่แอปของคุณให้มาเพื่อเสร็จสิ้นการตั้งค่า" |   step3: "ป้อนโทเค็นที่แอปของคุณให้มาเพื่อเสร็จสิ้นการตั้งค่า" | ||||||
|  |   setupCompleted: "ตั้งค่าสำเร็จแล้ว" | ||||||
|   step4: "นับจากนี้เป็นต้นไปการพยายามเข้าสู่ระบบในอนาคตนั้น อาจจะต้องขอโทเค็นในการเข้าสู่ระบบดังกล่าว" |   step4: "นับจากนี้เป็นต้นไปการพยายามเข้าสู่ระบบในอนาคตนั้น อาจจะต้องขอโทเค็นในการเข้าสู่ระบบดังกล่าว" | ||||||
|   securityKeyNotSupported: "เบราว์เซอร์ของคุณไม่รองรับคีย์ความปลอดภัยนะ" |   securityKeyNotSupported: "เบราว์เซอร์ของคุณไม่รองรับคีย์ความปลอดภัยนะ" | ||||||
|   registerTOTPBeforeKey: "กรุณาตั้งค่าแอปยืนยันตัวตนเพื่อลงทะเบียนรหัสความปลอดภัยหรือรหัสผ่าน" |   registerTOTPBeforeKey: "กรุณาตั้งค่าแอปยืนยันตัวตนเพื่อลงทะเบียนรหัสความปลอดภัยหรือรหัสผ่าน" | ||||||
|   securityKeyInfo: "นอกจากนี้การตรวจสอบความถูกต้องด้วยลายนิ้วมือหรือ PIN แล้ว คุณยังสามารถตั้งค่าการตรวจสอบสิทธิ์ผ่านคีย์ความปลอดภัยของฮาร์ดแวร์ที่รองรับ FIDO2 เพื่อเพิ่มความปลอดภัยให้กับบัญชีของคุณ" |   securityKeyInfo: "นอกจากนี้การตรวจสอบความถูกต้องด้วยลายนิ้วมือหรือ PIN แล้ว คุณยังสามารถตั้งค่าการตรวจสอบสิทธิ์ผ่านคีย์ความปลอดภัยของฮาร์ดแวร์ที่รองรับ FIDO2 เพื่อเพิ่มความปลอดภัยให้กับบัญชีของคุณ" | ||||||
|   chromePasskeyNotSupported: "ขณะนี้ยังไม่รองรับรหัสผ่านของ Chrome" |  | ||||||
|   registerSecurityKey: "ลงทะเบียนรหัสความปลอดภัยหรือรหัสผ่าน" |   registerSecurityKey: "ลงทะเบียนรหัสความปลอดภัยหรือรหัสผ่าน" | ||||||
|   securityKeyName: "ป้อนชื่อคีย์" |   securityKeyName: "ป้อนชื่อคีย์" | ||||||
|   tapSecurityKey: "กรุณาทำตามเบราว์เซอร์ของคุณเพื่อลงทะเบียนรหัสความปลอดภัยหรือรหัสผ่าน" |   tapSecurityKey: "กรุณาทำตามเบราว์เซอร์ของคุณเพื่อลงทะเบียนรหัสความปลอดภัยหรือรหัสผ่าน" | ||||||
| @@ -1708,6 +1739,7 @@ _2fa: | |||||||
|   renewTOTPConfirm: "วิธีการแบบนี้จะทําให้รหัสยืนยันจากแอพก่อนหน้าของคุณหยุดทํางานเลยนะ" |   renewTOTPConfirm: "วิธีการแบบนี้จะทําให้รหัสยืนยันจากแอพก่อนหน้าของคุณหยุดทํางานเลยนะ" | ||||||
|   renewTOTPOk: "ตั้งค่าคอนฟิกใหม่" |   renewTOTPOk: "ตั้งค่าคอนฟิกใหม่" | ||||||
|   renewTOTPCancel: "ไม่เป็นไร" |   renewTOTPCancel: "ไม่เป็นไร" | ||||||
|  |   backupCodes: "รหัสสำรองข้อมูล" | ||||||
| _permissions: | _permissions: | ||||||
|   "read:account": "ดูข้อมูลบัญชีของคุณ" |   "read:account": "ดูข้อมูลบัญชีของคุณ" | ||||||
|   "write:account": "แก้ไขข้อมูลบัญชีของคุณ" |   "write:account": "แก้ไขข้อมูลบัญชีของคุณ" | ||||||
| @@ -1741,6 +1773,10 @@ _permissions: | |||||||
|   "write:gallery": "แก้ไขแกลเลอรี่ของคุณ" |   "write:gallery": "แก้ไขแกลเลอรี่ของคุณ" | ||||||
|   "read:gallery-likes": "ดูรายการโพสต์ในแกลเลอรีที่ชอบของคุณ" |   "read:gallery-likes": "ดูรายการโพสต์ในแกลเลอรีที่ชอบของคุณ" | ||||||
|   "write:gallery-likes": "แก้ไขรายการโพสต์ในแกลเลอรีที่ชอบของคุณ" |   "write:gallery-likes": "แก้ไขรายการโพสต์ในแกลเลอรีที่ชอบของคุณ" | ||||||
|  |   "read:flash": "วิว เพลย์" | ||||||
|  |   "write:flash": "แก้ไขเพลย์" | ||||||
|  |   "read:flash-likes": "ดูรายชื่อของไลค์ เพลย์" | ||||||
|  |   "write:flash-likes": "แก้ไขรายชื่อของไลค์ เพลย์" | ||||||
| _auth: | _auth: | ||||||
|   shareAccessTitle: "การให้สิทธิ์แอปพลิเคชัน" |   shareAccessTitle: "การให้สิทธิ์แอปพลิเคชัน" | ||||||
|   shareAccess: "คุณต้องการอนุญาตให้ \"{name}\" เข้าถึงบัญชีนี้เลยมั้ย?" |   shareAccess: "คุณต้องการอนุญาตให้ \"{name}\" เข้าถึงบัญชีนี้เลยมั้ย?" | ||||||
| @@ -1973,11 +2009,17 @@ _notification: | |||||||
|   youReceivedFollowRequest: "คุณมีคำขอติดตามใหม่น่ะ" |   youReceivedFollowRequest: "คุณมีคำขอติดตามใหม่น่ะ" | ||||||
|   yourFollowRequestAccepted: "คำขอติดตามของคุณได้รับการยอมรับแล้วน่ะ" |   yourFollowRequestAccepted: "คำขอติดตามของคุณได้รับการยอมรับแล้วน่ะ" | ||||||
|   pollEnded: "โพลสำรวจความคิดเห็นผลลัพธ์มีพร้อมใช้งาน" |   pollEnded: "โพลสำรวจความคิดเห็นผลลัพธ์มีพร้อมใช้งาน" | ||||||
|  |   newNote: "โพสต์ใหม่" | ||||||
|   unreadAntennaNote: "เสาอากาศ {name}" |   unreadAntennaNote: "เสาอากาศ {name}" | ||||||
|   emptyPushNotificationMessage: "การแจ้งเตือนแบบพุชได้รับการอัพเดทแล้ว" |   emptyPushNotificationMessage: "การแจ้งเตือนแบบพุชได้รับการอัพเดทแล้ว" | ||||||
|   achievementEarned: "รับความสำเร็จ" |   achievementEarned: "รับความสำเร็จ" | ||||||
|  |   testNotification: "ทดสอบการแจ้งเตือน" | ||||||
|  |   checkNotificationBehavior: "ตรวจสอบลักษณะที่ปรากฏการแจ้งเตือน" | ||||||
|  |   sendTestNotification: "ส่งทดสอบการแจ้งเตือน" | ||||||
|  |   notificationWillBeDisplayedLikeThis: "การแจ้งเตือนมีลักษณะแบบนี้" | ||||||
|   _types: |   _types: | ||||||
|     all: "ทั้งหมด" |     all: "ทั้งหมด" | ||||||
|  |     note: "โน้ตใหม่" | ||||||
|     follow: "กำลังติดตาม" |     follow: "กำลังติดตาม" | ||||||
|     mention: "กล่าวถึง" |     mention: "กล่าวถึง" | ||||||
|     reply: "ตอบกลับ" |     reply: "ตอบกลับ" | ||||||
| @@ -2011,6 +2053,8 @@ _deck: | |||||||
|   introduction2: "คลิกที่เครื่องหมาย + ทางขวาของหน้าจอเพื่อเพิ่มคอลัมน์ใหม่ทุกครั้งที่คุณต้องการ" |   introduction2: "คลิกที่เครื่องหมาย + ทางขวาของหน้าจอเพื่อเพิ่มคอลัมน์ใหม่ทุกครั้งที่คุณต้องการ" | ||||||
|   widgetsIntroduction: "กรุณาเลือก \"แก้ไขวิดเจ็ต\" ในเมนูคอลัมน์และเพิ่มวิดเจ็ต" |   widgetsIntroduction: "กรุณาเลือก \"แก้ไขวิดเจ็ต\" ในเมนูคอลัมน์และเพิ่มวิดเจ็ต" | ||||||
|   useSimpleUiForNonRootPages: "แสดง UI ของ Root Page อย่างง่าย " |   useSimpleUiForNonRootPages: "แสดง UI ของ Root Page อย่างง่าย " | ||||||
|  |   usedAsMinWidthWhenFlexible: "ความกว้างขั้นต่ำนั้นจะถูกใช้งานสำหรับสิ่งนี้เมื่อเปิดใช้งานตัวเลือก \"ปรับความกว้างอัตโนมัติ\" หากเลือกเปิดใช้งานแล้ว" | ||||||
|  |   flexible: "ปรับความกว้างอัตโนมัติ" | ||||||
|   _columns: |   _columns: | ||||||
|     main: "หลัก" |     main: "หลัก" | ||||||
|     widgets: "วิดเจ็ต" |     widgets: "วิดเจ็ต" | ||||||
| @@ -2045,3 +2089,24 @@ _webhookSettings: | |||||||
|     renote: "รีโน้ตแล้วเมื่อ" |     renote: "รีโน้ตแล้วเมื่อ" | ||||||
|     reaction: "เมื่อได้รับรีแอคชั่น" |     reaction: "เมื่อได้รับรีแอคชั่น" | ||||||
|     mention: "เมื่อกำลังถูกกล่าวถึง" |     mention: "เมื่อกำลังถูกกล่าวถึง" | ||||||
|  | _moderationLogTypes: | ||||||
|  |   createRole: "สร้างบทบาทแล้ว" | ||||||
|  |   deleteRole: "ลบบทบาทแล้ว" | ||||||
|  |   updateRole: "อัปเดตบทบาทแล้ว" | ||||||
|  |   assignRole: "ได้รับมอบหมายบทบาท" | ||||||
|  |   unassignRole: "ถอดออกจากบทบาทแล้ว" | ||||||
|  |   suspend: "ถูกระงับ" | ||||||
|  |   unsuspend: "เลิกถูกระงับ" | ||||||
|  |   addCustomEmoji: "เพิ่มอีโมจิที่กำหนดเองแล้ว" | ||||||
|  |   updateCustomEmoji: "อัปเดตอีโมจิที่กำหนดเองแล้ว" | ||||||
|  |   deleteCustomEmoji: "ลบอีโมจิที่กำหนดเองออกแล้ว" | ||||||
|  |   updateServerSettings: "อัปเดตการตั้งค่าเซิร์ฟเวอร์แล้ว" | ||||||
|  |   updateUserNote: "อัปเดตโน้ตการกลั่นกรองแล้ว" | ||||||
|  |   deleteDriveFile: "ลบไฟล์ออกแล้ว" | ||||||
|  |   deleteNote: "ลบโน้ตออกแล้ว" | ||||||
|  |   resetPassword: "รีเซ็ตรหัสผ่าน" | ||||||
|  |   resolveAbuseReport: "รายงานได้รับการแก้ไขแล้ว" | ||||||
|  |   createInvitation: "สร้างคำเชิญ" | ||||||
|  |   createAd: "สร้างโฆษณาแล้ว" | ||||||
|  |   deleteAd: "ลบโฆษณาออกแล้ว" | ||||||
|  |   updateAd: "อัปเดตโฆษณาแล้ว" | ||||||
|   | |||||||
| @@ -371,6 +371,8 @@ noRole: "Rol bulunamadı" | |||||||
| color: "Renk" | color: "Renk" | ||||||
| addMemo: "Kısa not ekle" | addMemo: "Kısa not ekle" | ||||||
| icon: "Avatar" | icon: "Avatar" | ||||||
|  | replies: "yanıt" | ||||||
|  | renotes: "vazgeçme" | ||||||
| _accountDelete: | _accountDelete: | ||||||
|   started: "Silme işlemi başlatıldı" |   started: "Silme işlemi başlatıldı" | ||||||
| _email: | _email: | ||||||
| @@ -446,3 +448,6 @@ _deck: | |||||||
|     tl: "Zaman çizelgesi" |     tl: "Zaman çizelgesi" | ||||||
|     list: "Listeler" |     list: "Listeler" | ||||||
|     mentions: "Bahsetmeler" |     mentions: "Bahsetmeler" | ||||||
|  | _moderationLogTypes: | ||||||
|  |   suspend: "askıya al" | ||||||
|  |   resetPassword: "Şifre sıfırlama" | ||||||
|   | |||||||
| @@ -338,7 +338,6 @@ invite: "Запросити" | |||||||
| driveCapacityPerLocalAccount: "Об'єм диска на одного локального користувача" | driveCapacityPerLocalAccount: "Об'єм диска на одного локального користувача" | ||||||
| driveCapacityPerRemoteAccount: "Об'єм диска на одного віддаленого користувача" | driveCapacityPerRemoteAccount: "Об'єм диска на одного віддаленого користувача" | ||||||
| inMb: "В мегабайтах" | inMb: "В мегабайтах" | ||||||
| iconUrl: "URL аватара" |  | ||||||
| bannerUrl: "URL банера" | bannerUrl: "URL банера" | ||||||
| backgroundImageUrl: "URL-адреса фонового зображення" | backgroundImageUrl: "URL-адреса фонового зображення" | ||||||
| basicInfo: "Основна інформація" | basicInfo: "Основна інформація" | ||||||
| @@ -906,6 +905,8 @@ letsLookAtTimeline: "Перегляд історії" | |||||||
| horizontal: "Збоку" | horizontal: "Збоку" | ||||||
| youFollowing: "Підписки" | youFollowing: "Підписки" | ||||||
| icon: "Аватар" | icon: "Аватар" | ||||||
|  | replies: "Відповісти" | ||||||
|  | renotes: "Поширити" | ||||||
| _achievements: | _achievements: | ||||||
|   earnedAt: "Відкрито" |   earnedAt: "Відкрито" | ||||||
|   _types: |   _types: | ||||||
| @@ -1179,6 +1180,7 @@ _plugin: | |||||||
|   install: "Встановити плагін" |   install: "Встановити плагін" | ||||||
|   installWarn: "Будь ласка, не встановлюйте плагінів, яким ви не довіряєте." |   installWarn: "Будь ласка, не встановлюйте плагінів, яким ви не довіряєте." | ||||||
|   manage: "Керування плагінами" |   manage: "Керування плагінами" | ||||||
|  |   viewSource: "Переглянути вихідний код" | ||||||
| _preferencesBackups: | _preferencesBackups: | ||||||
|   list: "Створені бекапи" |   list: "Створені бекапи" | ||||||
|   saveNew: "Зберегти як новий" |   saveNew: "Зберегти як новий" | ||||||
| @@ -1618,3 +1620,6 @@ _deck: | |||||||
| _webhookSettings: | _webhookSettings: | ||||||
|   name: "Ім'я" |   name: "Ім'я" | ||||||
|   active: "Увімкнено" |   active: "Увімкнено" | ||||||
|  | _moderationLogTypes: | ||||||
|  |   suspend: "Призупинити" | ||||||
|  |   resetPassword: "Скинути пароль" | ||||||
|   | |||||||
| @@ -355,7 +355,6 @@ invite: "Taklif qilish" | |||||||
| driveCapacityPerLocalAccount: "Har bir mahalliy foydalanuvchi uchun disk maydoni" | driveCapacityPerLocalAccount: "Har bir mahalliy foydalanuvchi uchun disk maydoni" | ||||||
| driveCapacityPerRemoteAccount: "Har bir masofaviy foydalanuvchi uchun disk maydoni" | driveCapacityPerRemoteAccount: "Har bir masofaviy foydalanuvchi uchun disk maydoni" | ||||||
| inMb: "Megabaytlarda" | inMb: "Megabaytlarda" | ||||||
| iconUrl: "Ikonkaning URL manzili (masalan: favicon)" |  | ||||||
| bannerUrl: "Banner URLi" | bannerUrl: "Banner URLi" | ||||||
| backgroundImageUrl: "Fon rasmi URL manzili" | backgroundImageUrl: "Fon rasmi URL manzili" | ||||||
| basicInfo: "Asosiy ma'lumot" | basicInfo: "Asosiy ma'lumot" | ||||||
| @@ -844,6 +843,8 @@ rolesAssignedToMe: "Mening rollarim" | |||||||
| resetPasswordConfirm: "Qayta parol o'rnatmoqchimisiz?" | resetPasswordConfirm: "Qayta parol o'rnatmoqchimisiz?" | ||||||
| sensitiveWords: "Ta'sirchan so'zlar" | sensitiveWords: "Ta'sirchan so'zlar" | ||||||
| icon: "Avatar" | icon: "Avatar" | ||||||
|  | replies: "Javob berish" | ||||||
|  | renotes: "Qayta qayd etish" | ||||||
| _achievements: | _achievements: | ||||||
|   _types: |   _types: | ||||||
|     _viewInstanceChart: |     _viewInstanceChart: | ||||||
| @@ -1083,3 +1084,6 @@ _webhookSettings: | |||||||
|   _events: |   _events: | ||||||
|     renote: "Qayta qayd qilinganda" |     renote: "Qayta qayd qilinganda" | ||||||
|     mention: "Eslanganda" |     mention: "Eslanganda" | ||||||
|  | _moderationLogTypes: | ||||||
|  |   suspend: "To'xtatish" | ||||||
|  |   resetPassword: "Parolni tiklash" | ||||||
|   | |||||||
| @@ -1,5 +1,5 @@ | |||||||
| --- | --- | ||||||
| _lang_: "Tiếng Việt" | _lang_: "Tiếng Nhật" | ||||||
| headlineMisskey: "Mạng xã hội liên hợp" | headlineMisskey: "Mạng xã hội liên hợp" | ||||||
| introMisskey: "Xin chào! Misskey là một nền tảng tiểu blog phi tập trung mã nguồn mở.\nViết \"tút\" để chia sẻ những suy nghĩ của bạn 📡\nBằng \"biểu cảm\", bạn có thể bày tỏ nhanh chóng cảm xúc của bạn với các tút 👍\nHãy khám phá một thế giới mới! 🚀" | introMisskey: "Xin chào! Misskey là một nền tảng tiểu blog phi tập trung mã nguồn mở.\nViết \"tút\" để chia sẻ những suy nghĩ của bạn 📡\nBằng \"biểu cảm\", bạn có thể bày tỏ nhanh chóng cảm xúc của bạn với các tút 👍\nHãy khám phá một thế giới mới! 🚀" | ||||||
| poweredByMisskeyDescription: "{name} là một trong những chủ máy của <b>Misskey</b> là nền tảng mã nguồn mở" | poweredByMisskeyDescription: "{name} là một trong những chủ máy của <b>Misskey</b> là nền tảng mã nguồn mở" | ||||||
| @@ -45,6 +45,7 @@ pin: "Ghim" | |||||||
| unpin: "Bỏ ghim" | unpin: "Bỏ ghim" | ||||||
| copyContent: "Chép nội dung" | copyContent: "Chép nội dung" | ||||||
| copyLink: "Chép liên kết" | copyLink: "Chép liên kết" | ||||||
|  | copyLinkRenote: "Sao chép liên kết ghi chú" | ||||||
| delete: "Xóa" | delete: "Xóa" | ||||||
| deleteAndEdit: "Sửa" | deleteAndEdit: "Sửa" | ||||||
| deleteAndEditConfirm: "Bạn có chắc muốn sửa tút này? Những biểu cảm, lượt trả lời và đăng lại sẽ bị mất." | deleteAndEditConfirm: "Bạn có chắc muốn sửa tút này? Những biểu cảm, lượt trả lời và đăng lại sẽ bị mất." | ||||||
| @@ -156,6 +157,7 @@ addEmoji: "Thêm emoji" | |||||||
| settingGuide: "Cài đặt đề xuất" | settingGuide: "Cài đặt đề xuất" | ||||||
| cacheRemoteFiles: "Tập tin cache từ xa" | cacheRemoteFiles: "Tập tin cache từ xa" | ||||||
| cacheRemoteFilesDescription: "Khi tùy chọn này bị tắt, các tập tin từ xa sẽ được tải trực tiếp từ máy chủ khác. Điều này sẽ giúp giảm dung lượng lưu trữ nhưng lại tăng lưu lượng truy cập, vì hình thu nhỏ sẽ không được tạo." | cacheRemoteFilesDescription: "Khi tùy chọn này bị tắt, các tập tin từ xa sẽ được tải trực tiếp từ máy chủ khác. Điều này sẽ giúp giảm dung lượng lưu trữ nhưng lại tăng lưu lượng truy cập, vì hình thu nhỏ sẽ không được tạo." | ||||||
|  | youCanCleanRemoteFilesCache: "Bạn có thể xoá bộ nhớ đệm bằng cách nhấn vào nút🗑️ở trong phần quản lý tệp." | ||||||
| cacheRemoteSensitiveFiles: "Lưu các tập tin nhạy cảm vào bộ nhớ tạm từ xa" | cacheRemoteSensitiveFiles: "Lưu các tập tin nhạy cảm vào bộ nhớ tạm từ xa" | ||||||
| cacheRemoteSensitiveFilesDescription: "Khi bạn tắt tính năng này, các tệp nhạy cảm sẽ được tải trực tiếp từ máy chủ và không được lưu vào bộ nhớ tạm" | cacheRemoteSensitiveFilesDescription: "Khi bạn tắt tính năng này, các tệp nhạy cảm sẽ được tải trực tiếp từ máy chủ và không được lưu vào bộ nhớ tạm" | ||||||
| flagAsBot: "Đánh dấu đây là tài khoản bot" | flagAsBot: "Đánh dấu đây là tài khoản bot" | ||||||
| @@ -354,7 +356,6 @@ invite: "Mời" | |||||||
| driveCapacityPerLocalAccount: "Dung lượng ổ đĩa tối đa cho mỗi người dùng" | driveCapacityPerLocalAccount: "Dung lượng ổ đĩa tối đa cho mỗi người dùng" | ||||||
| driveCapacityPerRemoteAccount: "Dung lượng ổ đĩa tối đa cho mỗi người dùng từ xa" | driveCapacityPerRemoteAccount: "Dung lượng ổ đĩa tối đa cho mỗi người dùng từ xa" | ||||||
| inMb: "Tính bằng MB" | inMb: "Tính bằng MB" | ||||||
| iconUrl: "URL Icon" |  | ||||||
| bannerUrl: "URL Ảnh bìa" | bannerUrl: "URL Ảnh bìa" | ||||||
| backgroundImageUrl: "URL Ảnh nền" | backgroundImageUrl: "URL Ảnh nền" | ||||||
| basicInfo: "Thông tin cơ bản" | basicInfo: "Thông tin cơ bản" | ||||||
| @@ -410,10 +411,13 @@ aboutMisskey: "Về Misskey" | |||||||
| administrator: "Quản trị viên" | administrator: "Quản trị viên" | ||||||
| token: "Token" | token: "Token" | ||||||
| 2fa: "Xác thực 2 yếu tố" | 2fa: "Xác thực 2 yếu tố" | ||||||
|  | setupOf2fa: "Thiết lập xác thực 2 yếu tố" | ||||||
| totp: "Ứng dụng xác thực" | totp: "Ứng dụng xác thực" | ||||||
| totpDescription: "Nhắn mã OTP bằng ứng dụng xác thực" | totpDescription: "Nhắn mã OTP bằng ứng dụng xác thực" | ||||||
| moderator: "Kiểm duyệt viên" | moderator: "Kiểm duyệt viên" | ||||||
| moderation: "Kiểm duyệt" | moderation: "Kiểm duyệt" | ||||||
|  | moderationNote: "Ghi chú kiểm duyệt" | ||||||
|  | addModerationNote: "Thêm ghi chú kiểm duyệt" | ||||||
| nUsersMentioned: "Dùng bởi {n} người" | nUsersMentioned: "Dùng bởi {n} người" | ||||||
| securityKeyAndPasskey: "Mã bảo mật・Passkey" | securityKeyAndPasskey: "Mã bảo mật・Passkey" | ||||||
| securityKey: "Khóa bảo mật" | securityKey: "Khóa bảo mật" | ||||||
| @@ -989,9 +993,95 @@ copyErrorInfo: "Sao chép thông tin lỗi" | |||||||
| joinThisServer: "Đăng ký trên chủ máy này" | joinThisServer: "Đăng ký trên chủ máy này" | ||||||
| exploreOtherServers: "Tìm chủ máy khác" | exploreOtherServers: "Tìm chủ máy khác" | ||||||
| letsLookAtTimeline: "Thử xem Timeline" | letsLookAtTimeline: "Thử xem Timeline" | ||||||
|  | emailNotSupported: "Máy chủ này không hỗ trợ gửi email" | ||||||
|  | postToTheChannel: "Đăng lên kênh" | ||||||
|  | cannotBeChangedLater: "Không thể thay đổi sau này." | ||||||
|  | rolesAssignedToMe: "Vai trò được giao cho tôi" | ||||||
|  | resetPasswordConfirm: "Bạn thực sự muốn đặt lại mật khẩu?" | ||||||
|  | sensitiveWords: "Các từ nhạy cảm" | ||||||
|  | license: "Giấy phép" | ||||||
|  | unfavoriteConfirm: "Bạn thực sự muốn xoá khỏi mục yêu thích?" | ||||||
|  | retryAllQueuesConfirmText: "Điều này sẽ tạm thời làm tăng mức độ tải của máy chủ." | ||||||
|  | enableChartsForRemoteUser: "Tạo biểu đồ người dùng từ xa" | ||||||
|  | video: "Video" | ||||||
|  | videos: "Các video" | ||||||
|  | dataSaver: "Tiết kiệm dung lượng" | ||||||
|  | accountMigration: "Chuyển tài khoản" | ||||||
|  | accountMoved: "Người dùng này đã chuyển sang một tài khoản mới:" | ||||||
|  | accountMovedShort: "Tài khoản này đã được chuyển" | ||||||
|  | operationForbidden: "Thao tác này không thể thực hiện" | ||||||
|  | forceShowAds: "Luôn hiện quảng cáo" | ||||||
|  | notificationDisplay: "Thông báo" | ||||||
|  | leftTop: "Phía trên bên tráí" | ||||||
|  | rightTop: "Phía trên bên phải" | ||||||
|  | leftBottom: "Phía dưới bên trái" | ||||||
|  | rightBottom: "Phía dưới bên phải" | ||||||
|  | stackAxis: "Hướng chồng" | ||||||
|  | vertical: "Dọc" | ||||||
| horizontal: "Thanh bên" | horizontal: "Thanh bên" | ||||||
|  | position: "Vị trí" | ||||||
|  | serverRules: "Luật của máy chủ" | ||||||
| youFollowing: "Đang theo dõi" | youFollowing: "Đang theo dõi" | ||||||
|  | later: "Để sau" | ||||||
|  | goToMisskey: "Tới Misskey" | ||||||
|  | installed: "Đã tải xuống" | ||||||
|  | branding: "Thương hiệu" | ||||||
|  | turnOffToImprovePerformance: "Tắt mục này có thể cải thiện hiệu năng." | ||||||
|  | expirationDate: "Ngày hết hạn" | ||||||
|  | noExpirationDate: "Vô thời hạn" | ||||||
|  | waitingForMailAuth: "Đang chờ xác nhận email" | ||||||
|  | unused: "Chưa được sử dụng" | ||||||
|  | used: "Đã được sử dụng" | ||||||
|  | expired: "Đã hết hạn" | ||||||
|  | doYouAgree: "Đồng ý?" | ||||||
|  | iHaveReadXCarefullyAndAgree: "Tôi đã đọc và đồng ý với \"x\"." | ||||||
|  | dialog: "Hộp thoại" | ||||||
| icon: "Ảnh đại diện" | icon: "Ảnh đại diện" | ||||||
|  | forYou: "Dành cho bạn" | ||||||
|  | currentAnnouncements: "Thông báo hiện tại" | ||||||
|  | pastAnnouncements: "Thông báo trước đó" | ||||||
|  | youHaveUnreadAnnouncements: "Có thông báo chưa đọc." | ||||||
|  | replies: "Trả lời" | ||||||
|  | renotes: "Đăng lại" | ||||||
|  | loadReplies: "Hiển thị các trả lời" | ||||||
|  | pinnedList: "Các mục đã được ghim" | ||||||
|  | keepScreenOn: "Giữ màn hình luôn bật" | ||||||
|  | verifiedLink: "Chúng tôi đã xác nhận bạn là chủ sở hữu của đường dẫn này" | ||||||
|  | _announcement: | ||||||
|  |   forExistingUsers: "Chỉ những người dùng đã tồn tại" | ||||||
|  |   forExistingUsersDescription: "Nếu được bật, thông báo này sẽ chỉ hiển thị với những người dùng đã tồn tại vào lúc thông báo được tạo. Nếu tắt đi, những tài khoản mới đăng ký sau khi thông báo được đăng lên cũng sẽ thấy nó." | ||||||
|  |   end: "Lưu trữ thông báo" | ||||||
|  |   tooManyActiveAnnouncementDescription: "Có quá nhiều thông báo sẽ làm trải nghiệm của người dùng tệ đi. Vui lòng lưu trữ những thông báo đã hết hiệu lực." | ||||||
|  |   readConfirmTitle: "Đánh dấu là đã đọc?" | ||||||
|  |   readConfirmText: "Điều này sẽ đánh dấu nội dung của \"{title}\" là đã đọc." | ||||||
|  | _initialAccountSetting: | ||||||
|  |   accountCreated: "Tài khoản của bạn đã được tạo thành công!" | ||||||
|  |   letsStartAccountSetup: "Để bắt đầu, hãy cùng thiết lập tài khoản nhé." | ||||||
|  |   letsFillYourProfile: "Đầu tiên, hãy thiết lập hồ sơ của bạn." | ||||||
|  |   profileSetting: "Thiết lập hồ sơ" | ||||||
|  |   privacySetting: "Cài đặt quyền riêng tư" | ||||||
|  |   theseSettingsCanEditLater: "Bạn vẫn có thể thay đổi những cài đặt này." | ||||||
|  |   youCanEditMoreSettingsInSettingsPageLater: "Còn rất nhiều những cài đặt khác bạn có thể thay đổi ở trang \"Cài đặt\". Hãy nhớ ghé thăm trong lần sau nhé." | ||||||
|  |   followUsers: "Thử theo dõi một vài người mà bạn có thể thích để xây dựng dòng thời gian của mình." | ||||||
|  |   pushNotificationDescription: "Bật thông báo đẩy sẽ cho phép bạn nhận thông báo từ {name} trực tiếp từ thiết bị của bạn." | ||||||
|  |   initialAccountSettingCompleted: "Thiết lập tài khoản thành công!" | ||||||
|  |   haveFun: "Hãy tận hưởng {name} nhé!" | ||||||
|  |   ifYouNeedLearnMore: "Nếu bạn muốn tìm hiểu thêm về cách sử dụng {name} (Misskey), hãy vào {link}." | ||||||
|  |   skipAreYouSure: "Bạn thực sự muốn bỏ qua mục thiết lập tài khoản?" | ||||||
|  |   laterAreYouSure: "Bạn thực sự muốn thiết lập tài khoản vào lúc khác?" | ||||||
|  | _serverSettings: | ||||||
|  |   iconUrl: "Biểu tượng URL" | ||||||
|  |   appIconResolutionMustBe: "Độ phân giải tối thiểu là {resolution}." | ||||||
|  |   manifestJsonOverride: "Ghi đè manifest.json" | ||||||
|  | _accountMigration: | ||||||
|  |   moveFrom: "Chuyển một tài khoản khác vào tài khoản này" | ||||||
|  |   moveFromLabel: "Tài khoản gốc #{n}" | ||||||
|  |   moveTo: "Chuyển tài khoản này vào một tài khoản khác" | ||||||
|  |   moveCannotBeUndone: "Việc chuyển tài khoản không thể huỷ." | ||||||
|  |   moveAccountDescription: "Điều này sẽ chuyển tài khoản này sang một tài khoản khác.\n ・Những người theo dõi sẽ tự động được chuyển sang tài khoản mới\n ・Tài khoản này sẽ tự bỏ theo dõi những người mà bạn đã theo dõi trước đây\n ・Bạn sẽ không thể đăng tút mới, v.v trên tài khoản này\n\nDù việc chuyển người theo dõi được diễn ra tự động, bạn vẫn phải tự chuẩn bị một vài bước để chuyển danh sách những người dùng bạn đang theo dõi. Để làm vậy, vui lòng thực hiện việc xuất dữ liệu những người dùng đã theo dõi mà sau này bạn sẽ dùng để nhập vào tài khoản mới ở menu Cài đặt. Hành động tương tự áp dụng với danh sách những người dùng bị chặn hoặc tắt tiếng.\n\n(Điều này áp dụng cho phiên bản Misskey v13.12.0 và sau này. Các phần mềm ActivityPub khác , ví dụ như Mastodon, sẽ có thể hoạt động khác đi.)" | ||||||
|  |   startMigration: "Chuyển" | ||||||
|  |   movedAndCannotBeUndone: "\nTài khoản này đã được chuyển đi.\nViệc di chuyển tài khoản không thể bị huỷ bỏ." | ||||||
|  |   movedTo: "Tài khoản mới:" | ||||||
| _achievements: | _achievements: | ||||||
|   earnedAt: "Ngày thu nhận" |   earnedAt: "Ngày thu nhận" | ||||||
|   _types: |   _types: | ||||||
| @@ -1030,6 +1120,8 @@ _achievements: | |||||||
|       title: "Hàng tinh đăng bài" |       title: "Hàng tinh đăng bài" | ||||||
|       description: "Đã đăng bài 50,000 lần rồi" |       description: "Đã đăng bài 50,000 lần rồi" | ||||||
|     _notes100000: |     _notes100000: | ||||||
|  |       title: "ALL YOUR NOTE ARE BELONG TO US" | ||||||
|  |       description: "Đăng 100,000 tút" | ||||||
|       flavor: "Liệu viết bài gì tầm này vậy? " |       flavor: "Liệu viết bài gì tầm này vậy? " | ||||||
|     _login3: |     _login3: | ||||||
|       title: "Sơ cấp I" |       title: "Sơ cấp I" | ||||||
| @@ -1061,6 +1153,15 @@ _achievements: | |||||||
|     _login400: |     _login400: | ||||||
|       title: "Khách hàng thường xuyên cấp III" |       title: "Khách hàng thường xuyên cấp III" | ||||||
|       description: "Tổng số ngày đăng nhập đạt 400 ngày" |       description: "Tổng số ngày đăng nhập đạt 400 ngày" | ||||||
|  |     _login1000: | ||||||
|  |       flavor: "Cảm ơn bạn đã sử dụng Misskey!" | ||||||
|  |     _noteFavorited1: | ||||||
|  |       title: "Nhà thiên văn học" | ||||||
|  |     _myNoteFavorited1: | ||||||
|  |       title: "Đi tìm những ngôi sao" | ||||||
|  |     _profileFilled: | ||||||
|  |       title: "Luôn sẵn sàng" | ||||||
|  |       description: "Thiết lập tài khoản của bạn" | ||||||
|     _markedAsCat: |     _markedAsCat: | ||||||
|       title: "Tôi là một con mèo" |       title: "Tôi là một con mèo" | ||||||
|       description: "Bật chế độ mèo" |       description: "Bật chế độ mèo" | ||||||
| @@ -1086,8 +1187,18 @@ _achievements: | |||||||
|     _followers10: |     _followers10: | ||||||
|       title: "FOLLOW ME!!" |       title: "FOLLOW ME!!" | ||||||
|       description: "Người theo dõi bạn vượt lên 10 người" |       description: "Người theo dõi bạn vượt lên 10 người" | ||||||
|  |     _followers50: | ||||||
|  |       title: "Từng chút một" | ||||||
|  |       description: "Đạt được 50 lượt theo dõi" | ||||||
|  |     _followers100: | ||||||
|  |       title: "Người nổi tiếng" | ||||||
|  |       description: "Đạt được 100 lượt theo dõi" | ||||||
|  |     _followers300: | ||||||
|  |       title: "Vui lòng xếp thành hàng nào" | ||||||
|  |       description: "Đạt được 300 lượt theo dõi" | ||||||
|     _followers500: |     _followers500: | ||||||
|       title: "Trạm phát sóng" |       title: "Trạm phát sóng" | ||||||
|  |       description: "Đạt được 500 lượt theo dõi" | ||||||
|     _followers1000: |     _followers1000: | ||||||
|       title: "Người có tầm ảnh hưởng" |       title: "Người có tầm ảnh hưởng" | ||||||
|       description: "Người theo dõi bạn vượt lên 1000 người" |       description: "Người theo dõi bạn vượt lên 1000 người" | ||||||
| @@ -1106,11 +1217,15 @@ _achievements: | |||||||
|       description: "Tìm thấy được những kho báu cất giấu" |       description: "Tìm thấy được những kho báu cất giấu" | ||||||
|     _client30min: |     _client30min: | ||||||
|       title: "Giải lao xỉu" |       title: "Giải lao xỉu" | ||||||
|  |       description: "Giữ Misskey mở trong ít nhất 30 phút" | ||||||
|  |     _client60min: | ||||||
|  |       description: "Giữ Misskey mở trong ít nhất 60 phút" | ||||||
|     _noteDeletedWithin1min: |     _noteDeletedWithin1min: | ||||||
|       title: "Xem như không có gì đâu nha" |       title: "Xem như không có gì đâu nha" | ||||||
|     _postedAtLateNight: |     _postedAtLateNight: | ||||||
|       title: "Loài ăn đêm" |       title: "Loài ăn đêm" | ||||||
|       description: "Đăng bài trong đêm khuya " |       description: "Đăng bài trong đêm khuya " | ||||||
|  |       flavor: "Đến giờ đi ngủ rồi." | ||||||
|     _postedAt0min0sec: |     _postedAt0min0sec: | ||||||
|       title: "Tín hiệu báo giờ" |       title: "Tín hiệu báo giờ" | ||||||
|       description: "Đăng bài vào 0 phút 0 giây" |       description: "Đăng bài vào 0 phút 0 giây" | ||||||
| @@ -1141,6 +1256,8 @@ _achievements: | |||||||
|     _setNameToSyuilo: |     _setNameToSyuilo: | ||||||
|       title: "Ngưỡng mộ với vị thần" |       title: "Ngưỡng mộ với vị thần" | ||||||
|       description: "Đạt tên là syuilo" |       description: "Đạt tên là syuilo" | ||||||
|  |     _passedSinceAccountCreated1: | ||||||
|  |       title: "Kỷ niệm một năm" | ||||||
|     _loggedInOnBirthday: |     _loggedInOnBirthday: | ||||||
|       title: "Sinh nhật vủi vẻ" |       title: "Sinh nhật vủi vẻ" | ||||||
|       description: "Đăng nhập vào ngày sinh" |       description: "Đăng nhập vào ngày sinh" | ||||||
| @@ -1226,6 +1343,7 @@ _plugin: | |||||||
|   install: "Cài đặt tiện ích" |   install: "Cài đặt tiện ích" | ||||||
|   installWarn: "Vui lòng không cài đặt những tiện ích đáng ngờ." |   installWarn: "Vui lòng không cài đặt những tiện ích đáng ngờ." | ||||||
|   manage: "Quản lý plugin" |   manage: "Quản lý plugin" | ||||||
|  |   viewSource: "Xem mã nguồn" | ||||||
| _preferencesBackups: | _preferencesBackups: | ||||||
|   list: "Tạo sao lưu" |   list: "Tạo sao lưu" | ||||||
|   saveNew: "Lưu bản sao lưu" |   saveNew: "Lưu bản sao lưu" | ||||||
| @@ -1400,7 +1518,6 @@ _timelineTutorial: | |||||||
| _2fa: | _2fa: | ||||||
|   alreadyRegistered: "Bạn đã đăng ký thiết bị xác minh 2 bước." |   alreadyRegistered: "Bạn đã đăng ký thiết bị xác minh 2 bước." | ||||||
|   registerTOTP: "Đăng ký ứng dụng xác thực" |   registerTOTP: "Đăng ký ứng dụng xác thực" | ||||||
|   passwordToTOTP: "Nhắn mật mã" |  | ||||||
|   step1: "Trước tiên, hãy cài đặt một ứng dụng xác minh (chẳng hạn như {a} hoặc {b}) trên thiết bị của bạn." |   step1: "Trước tiên, hãy cài đặt một ứng dụng xác minh (chẳng hạn như {a} hoặc {b}) trên thiết bị của bạn." | ||||||
|   step2: "Sau đó, quét mã QR hiển thị trên màn hình này." |   step2: "Sau đó, quét mã QR hiển thị trên màn hình này." | ||||||
|   step2Click: "Quét mã QR trên ứng dụng xác thực (Authy, Google authenticator, v.v.)" |   step2Click: "Quét mã QR trên ứng dụng xác thực (Authy, Google authenticator, v.v.)" | ||||||
| @@ -1410,7 +1527,6 @@ _2fa: | |||||||
|   securityKeyNotSupported: "Trình duyệt của bạn không hỗ trợ khóa bảo mật" |   securityKeyNotSupported: "Trình duyệt của bạn không hỗ trợ khóa bảo mật" | ||||||
|   registerTOTPBeforeKey: "Vui lòng thiết lập một ứng dụng xác thực để đăng ký khóa bảo mật hoặc mật khẩu." |   registerTOTPBeforeKey: "Vui lòng thiết lập một ứng dụng xác thực để đăng ký khóa bảo mật hoặc mật khẩu." | ||||||
|   securityKeyInfo: "Bên cạnh xác minh bằng vân tay hoặc mã PIN, bạn cũng có thể thiết lập xác minh thông qua khóa bảo mật phần cứng hỗ trợ FIDO2 để bảo mật hơn nữa cho tài khoản của mình." |   securityKeyInfo: "Bên cạnh xác minh bằng vân tay hoặc mã PIN, bạn cũng có thể thiết lập xác minh thông qua khóa bảo mật phần cứng hỗ trợ FIDO2 để bảo mật hơn nữa cho tài khoản của mình." | ||||||
|   chromePasskeyNotSupported: "Mật khẩu Chrome hiện không được hỗ trợ." |  | ||||||
|   registerSecurityKey: "Tạo khóa bảo mật hoặc mã bảo mật" |   registerSecurityKey: "Tạo khóa bảo mật hoặc mã bảo mật" | ||||||
|   securityKeyName: "Nhập tên khóa bảo mật" |   securityKeyName: "Nhập tên khóa bảo mật" | ||||||
|   tapSecurityKey: "Vui lòng làm theo hướng dẫn của trình duyệt để đăng ký mã bảo mật hoặc mã khóa" |   tapSecurityKey: "Vui lòng làm theo hướng dẫn của trình duyệt để đăng ký mã bảo mật hoặc mã khóa" | ||||||
| @@ -1744,3 +1860,6 @@ _webhookSettings: | |||||||
|   _events: |   _events: | ||||||
|     reaction: "Khi nhận được sự kiện" |     reaction: "Khi nhận được sự kiện" | ||||||
|     mention: "Khi có người nhắc tới bạn" |     mention: "Khi có người nhắc tới bạn" | ||||||
|  | _moderationLogTypes: | ||||||
|  |   suspend: "Vô hiệu hóa" | ||||||
|  |   resetPassword: "Đặt lại mật khẩu" | ||||||
|   | |||||||
| @@ -163,7 +163,7 @@ cacheRemoteSensitiveFilesDescription: "如果禁用这项设定,远程服务 | |||||||
| flagAsBot: "这是一个机器人账号" | flagAsBot: "这是一个机器人账号" | ||||||
| flagAsBotDescription: "如果此账户由程序控制,请启用此项。启用后,此标志可以帮助其他开发人员防止机器人之间产生无限互动的行为,并让 Misskey 的内部系统将此账户识别为机器人。" | flagAsBotDescription: "如果此账户由程序控制,请启用此项。启用后,此标志可以帮助其他开发人员防止机器人之间产生无限互动的行为,并让 Misskey 的内部系统将此账户识别为机器人。" | ||||||
| flagAsCat: "喵!!!!!!!!!!!!" | flagAsCat: "喵!!!!!!!!!!!!" | ||||||
| flagAsCatDescription: "如果您想表明此帐户是一只猫,请打开此标志。\n开启后,会在您的头像上出现猫耳朵,并将你的帖子中的「na」替换为「nya」,日文同理。" | flagAsCatDescription: "喵喵喵??" | ||||||
| flagShowTimelineReplies: "在时间线上显示帖子的回复" | flagShowTimelineReplies: "在时间线上显示帖子的回复" | ||||||
| flagShowTimelineRepliesDescription: "启用时,时间线除了显示用户的帖子外,还会显示其他用户对帖子的回复。" | flagShowTimelineRepliesDescription: "启用时,时间线除了显示用户的帖子外,还会显示其他用户对帖子的回复。" | ||||||
| autoAcceptFollowed: "自动允许来自我关注的用户对我的关注请求" | autoAcceptFollowed: "自动允许来自我关注的用户对我的关注请求" | ||||||
| @@ -356,7 +356,6 @@ invite: "邀请" | |||||||
| driveCapacityPerLocalAccount: "每个用户的网盘容量" | driveCapacityPerLocalAccount: "每个用户的网盘容量" | ||||||
| driveCapacityPerRemoteAccount: "每个远程用户的网盘容量" | driveCapacityPerRemoteAccount: "每个远程用户的网盘容量" | ||||||
| inMb: "以兆字节(MegaByte)为单位" | inMb: "以兆字节(MegaByte)为单位" | ||||||
| iconUrl: "图标 URL" |  | ||||||
| bannerUrl: "横幅 URL" | bannerUrl: "横幅 URL" | ||||||
| backgroundImageUrl: "背景图 URL" | backgroundImageUrl: "背景图 URL" | ||||||
| basicInfo: "基本信息" | basicInfo: "基本信息" | ||||||
| @@ -417,6 +416,9 @@ totp: "身份验证应用" | |||||||
| totpDescription: "使用认证应用输入一次性密码。" | totpDescription: "使用认证应用输入一次性密码。" | ||||||
| moderator: "监察员" | moderator: "监察员" | ||||||
| moderation: "管理" | moderation: "管理" | ||||||
|  | moderationNote: "管理笔记" | ||||||
|  | addModerationNote: "添加管理笔记" | ||||||
|  | moderationLogs: "管理日志" | ||||||
| nUsersMentioned: "{n} 被提到" | nUsersMentioned: "{n} 被提到" | ||||||
| securityKeyAndPasskey: "安全密钥或 Passkey" | securityKeyAndPasskey: "安全密钥或 Passkey" | ||||||
| securityKey: "安全密钥" | securityKey: "安全密钥" | ||||||
| @@ -709,6 +711,7 @@ lockedAccountInfo: "即使启用该功能,只要您不将帖子可见范围设 | |||||||
| alwaysMarkSensitive: "默认将媒体文件标记为敏感内容" | alwaysMarkSensitive: "默认将媒体文件标记为敏感内容" | ||||||
| loadRawImages: "添加附件图像的缩略图时使用原始图像质量" | loadRawImages: "添加附件图像的缩略图时使用原始图像质量" | ||||||
| disableShowingAnimatedImages: "不播放动画" | disableShowingAnimatedImages: "不播放动画" | ||||||
|  | highlightSensitiveMedia: "高亮显示敏感媒体" | ||||||
| verificationEmailSent: "已发送确认电子邮件。请访问电子邮件中的链接以完成设置。" | verificationEmailSent: "已发送确认电子邮件。请访问电子邮件中的链接以完成设置。" | ||||||
| notSet: "未设置" | notSet: "未设置" | ||||||
| emailVerified: "电子邮件地址已验证" | emailVerified: "电子邮件地址已验证" | ||||||
| @@ -1023,7 +1026,7 @@ retryAllQueuesConfirmText: "可能会使服务器负荷在一定时间内增加" | |||||||
| enableChartsForRemoteUser: "生成远程用户的图表" | enableChartsForRemoteUser: "生成远程用户的图表" | ||||||
| enableChartsForFederatedInstances: "生成远程服务器的图表" | enableChartsForFederatedInstances: "生成远程服务器的图表" | ||||||
| showClipButtonInNoteFooter: "在贴文下方显示便签按钮" | showClipButtonInNoteFooter: "在贴文下方显示便签按钮" | ||||||
| largeNoteReactions: "使用大图标来显示回应" | reactionsDisplaySize: "回应显示大小" | ||||||
| noteIdOrUrl: "帖子 ID 或 URL" | noteIdOrUrl: "帖子 ID 或 URL" | ||||||
| video: "视频" | video: "视频" | ||||||
| videos: "视频" | videos: "视频" | ||||||
| @@ -1105,6 +1108,24 @@ forYou: "您的" | |||||||
| currentAnnouncements: "现在的公告" | currentAnnouncements: "现在的公告" | ||||||
| pastAnnouncements: "过去的公告" | pastAnnouncements: "过去的公告" | ||||||
| youHaveUnreadAnnouncements: "您有未读的公告" | youHaveUnreadAnnouncements: "您有未读的公告" | ||||||
|  | useSecurityKey: "请根据浏览器或设备的提示,使用安全密钥或通行密钥。" | ||||||
|  | replies: "回复" | ||||||
|  | renotes: "转发" | ||||||
|  | loadReplies: "查看回复" | ||||||
|  | loadConversation: "查看对话" | ||||||
|  | pinnedList: "已置顶的列表" | ||||||
|  | keepScreenOn: "保持设备屏幕开启" | ||||||
|  | verifiedLink: "已验证的链接" | ||||||
|  | notifyNotes: "打开发帖通知" | ||||||
|  | unnotifyNotes: "关闭发帖通知" | ||||||
|  | authentication: "验证" | ||||||
|  | authenticationRequiredToContinue: "要继续,请先进行验证" | ||||||
|  | dateAndTime: "日期和时间" | ||||||
|  | showRenotes: "显示转帖" | ||||||
|  | edited: "已编辑" | ||||||
|  | notificationRecieveConfig: "通知接收设置" | ||||||
|  | mutualFollow: "互相关注" | ||||||
|  | fileAttachedOnly: "仅限媒体" | ||||||
| _announcement: | _announcement: | ||||||
|   forExistingUsers: "仅限现有用户" |   forExistingUsers: "仅限现有用户" | ||||||
|   forExistingUsersDescription: "若启用,该公告将仅对创建此公告时存在的用户可见。 如果禁用,则在创建此公告后注册的用户也可以看到该公告。" |   forExistingUsersDescription: "若启用,该公告将仅对创建此公告时存在的用户可见。 如果禁用,则在创建此公告后注册的用户也可以看到该公告。" | ||||||
| @@ -1131,6 +1152,15 @@ _initialAccountSetting: | |||||||
|   laterAreYouSure: "要稍后再进行初始设定吗?" |   laterAreYouSure: "要稍后再进行初始设定吗?" | ||||||
| _serverRules: | _serverRules: | ||||||
|   description: "在新用户注册前显示服务器的简单规则。推荐显示服务条款的主要内容。" |   description: "在新用户注册前显示服务器的简单规则。推荐显示服务条款的主要内容。" | ||||||
|  | _serverSettings: | ||||||
|  |   iconUrl: "图标 URL" | ||||||
|  |   appIconDescription: "指定当 {host} 显示为 app 时的图标。" | ||||||
|  |   appIconUsageExample: "例如:作为书签添加到 PWA 或手机主屏幕的时候" | ||||||
|  |   appIconStyleRecommendation: "因为有可能会被裁切为圆形或者圆角矩形,建议使用边缘带有留白背景的图标。" | ||||||
|  |   appIconResolutionMustBe: "分辨率必须为 {resolution}。" | ||||||
|  |   manifestJsonOverride: "覆盖 mainfest.json" | ||||||
|  |   shortName: "简称" | ||||||
|  |   shortNameDescription: "如果服务器的正式名称很长,可以用简称或者別名来替代。" | ||||||
| _accountMigration: | _accountMigration: | ||||||
|   moveFrom: "从别的账号迁移到此账户" |   moveFrom: "从别的账号迁移到此账户" | ||||||
|   moveFromSub: "为另一个账户建立别名" |   moveFromSub: "为另一个账户建立别名" | ||||||
| @@ -1385,6 +1415,9 @@ _achievements: | |||||||
|       title: "Brain Diver" |       title: "Brain Diver" | ||||||
|       description: "发布了包含 Brain Diver 链接的帖子" |       description: "发布了包含 Brain Diver 链接的帖子" | ||||||
|       flavor: "Misskey-Misskey La-Tu-Ma" |       flavor: "Misskey-Misskey La-Tu-Ma" | ||||||
|  |     _smashTestNotificationButton: | ||||||
|  |       title: "过度测试" | ||||||
|  |       description: "短时间内连续测试通知" | ||||||
| _role: | _role: | ||||||
|   new: "创建角色" |   new: "创建角色" | ||||||
|   edit: "编辑角色" |   edit: "编辑角色" | ||||||
| @@ -1423,6 +1456,7 @@ _role: | |||||||
|     gtlAvailable: "查看全局时间线" |     gtlAvailable: "查看全局时间线" | ||||||
|     ltlAvailable: "查看本地时间线" |     ltlAvailable: "查看本地时间线" | ||||||
|     canPublicNote: "允许公开发帖" |     canPublicNote: "允许公开发帖" | ||||||
|  |     canEditNote: "编辑帖子" | ||||||
|     canInvite: "发放服务器邀请码" |     canInvite: "发放服务器邀请码" | ||||||
|     inviteLimit: "可发行邀请码的数量" |     inviteLimit: "可发行邀请码的数量" | ||||||
|     inviteLimitCycle: "邀请码的发行间隔" |     inviteLimitCycle: "邀请码的发行间隔" | ||||||
| @@ -1508,6 +1542,7 @@ _plugin: | |||||||
|   install: "安装插件" |   install: "安装插件" | ||||||
|   installWarn: "请不要安装不可信的插件。" |   installWarn: "请不要安装不可信的插件。" | ||||||
|   manage: "管理插件..." |   manage: "管理插件..." | ||||||
|  |   viewSource: "查看源代码" | ||||||
| _preferencesBackups: | _preferencesBackups: | ||||||
|   list: "已创建的备份" |   list: "已创建的备份" | ||||||
|   saveNew: "另存为" |   saveNew: "另存为" | ||||||
| @@ -1695,7 +1730,6 @@ _timelineTutorial: | |||||||
| _2fa: | _2fa: | ||||||
|   alreadyRegistered: "此设备已被注册" |   alreadyRegistered: "此设备已被注册" | ||||||
|   registerTOTP: "开始设置认证应用" |   registerTOTP: "开始设置认证应用" | ||||||
|   passwordToTOTP: "请输入您的密码" |  | ||||||
|   step1: "首先,在您的设备上安装验证应用,例如 {a} 或 {b}。" |   step1: "首先,在您的设备上安装验证应用,例如 {a} 或 {b}。" | ||||||
|   step2: "然后,扫描屏幕上显示的二维码。" |   step2: "然后,扫描屏幕上显示的二维码。" | ||||||
|   step2Click: "通过点击二维码,您可以使用设备上安装的身份验证器应用程序或密钥环进行注册" |   step2Click: "通过点击二维码,您可以使用设备上安装的身份验证器应用程序或密钥环进行注册" | ||||||
| @@ -1707,7 +1741,6 @@ _2fa: | |||||||
|   securityKeyNotSupported: "您的浏览器不支持安全密钥。" |   securityKeyNotSupported: "您的浏览器不支持安全密钥。" | ||||||
|   registerTOTPBeforeKey: "要注册安全密钥或 Passkey,请先设置验证器应用程序。" |   registerTOTPBeforeKey: "要注册安全密钥或 Passkey,请先设置验证器应用程序。" | ||||||
|   securityKeyInfo: "注册兼容 WebAuthn 的密钥,例如支持 FIDO2 的硬件安全密钥、设备上的生物识别功能、PIN 码以及 Passkey 等。" |   securityKeyInfo: "注册兼容 WebAuthn 的密钥,例如支持 FIDO2 的硬件安全密钥、设备上的生物识别功能、PIN 码以及 Passkey 等。" | ||||||
|   chromePasskeyNotSupported: "目前不支持 Chrome 的 Passkey。" |  | ||||||
|   registerSecurityKey: "注册安全密钥或 Passkey" |   registerSecurityKey: "注册安全密钥或 Passkey" | ||||||
|   securityKeyName: "输入密钥名称" |   securityKeyName: "输入密钥名称" | ||||||
|   tapSecurityKey: "请按照浏览器说明操作来注册安全密钥或 Passkey。" |   tapSecurityKey: "请按照浏览器说明操作来注册安全密钥或 Passkey。" | ||||||
| @@ -1775,6 +1808,7 @@ _antennaSources: | |||||||
|   homeTimeline: "已关注用户的帖子" |   homeTimeline: "已关注用户的帖子" | ||||||
|   users: "来自指定用户的帖子" |   users: "来自指定用户的帖子" | ||||||
|   userList: "来自指定列表中的帖子" |   userList: "来自指定列表中的帖子" | ||||||
|  |   userBlacklist: "除掉已选择用户后所有的帖子" | ||||||
| _weekday: | _weekday: | ||||||
|   sunday: "星期日" |   sunday: "星期日" | ||||||
|   monday: "星期一" |   monday: "星期一" | ||||||
| @@ -1874,6 +1908,7 @@ _profile: | |||||||
|   metadataContent: "内容" |   metadataContent: "内容" | ||||||
|   changeAvatar: "修改头像" |   changeAvatar: "修改头像" | ||||||
|   changeBanner: "修改横幅" |   changeBanner: "修改横幅" | ||||||
|  |   verifiedLinkDescription: "如果将内容设置为 URL,当链接所指向的网页内包含自己的个人资料链接时,可以显示一个已验证图标。" | ||||||
| _exportOrImport: | _exportOrImport: | ||||||
|   allNotes: "所有帖子" |   allNotes: "所有帖子" | ||||||
|   favoritedNotes: "收藏的帖子" |   favoritedNotes: "收藏的帖子" | ||||||
| @@ -1992,11 +2027,17 @@ _notification: | |||||||
|   youReceivedFollowRequest: "您有新的关注请求" |   youReceivedFollowRequest: "您有新的关注请求" | ||||||
|   yourFollowRequestAccepted: "您的关注请求已通过" |   yourFollowRequestAccepted: "您的关注请求已通过" | ||||||
|   pollEnded: "问卷调查结果已生成。" |   pollEnded: "问卷调查结果已生成。" | ||||||
|  |   newNote: "新的帖子" | ||||||
|   unreadAntennaNote: "天线 {name}" |   unreadAntennaNote: "天线 {name}" | ||||||
|   emptyPushNotificationMessage: "推送通知已更新" |   emptyPushNotificationMessage: "推送通知已更新" | ||||||
|   achievementEarned: "获得成就" |   achievementEarned: "获得成就" | ||||||
|  |   testNotification: "测试通知" | ||||||
|  |   checkNotificationBehavior: "检查通知显示" | ||||||
|  |   sendTestNotification: "发送测试通知" | ||||||
|  |   notificationWillBeDisplayedLikeThis: "通知将会这样表示" | ||||||
|   _types: |   _types: | ||||||
|     all: "全部" |     all: "全部" | ||||||
|  |     note: "用户的新帖子" | ||||||
|     follow: "关注中" |     follow: "关注中" | ||||||
|     mention: "提及" |     mention: "提及" | ||||||
|     reply: "回复" |     reply: "回复" | ||||||
| @@ -2066,3 +2107,32 @@ _webhookSettings: | |||||||
|     renote: "被转发时" |     renote: "被转发时" | ||||||
|     reaction: "被回应时" |     reaction: "被回应时" | ||||||
|     mention: "被提及时" |     mention: "被提及时" | ||||||
|  | _moderationLogTypes: | ||||||
|  |   createRole: "创建角色" | ||||||
|  |   deleteRole: "删除角色" | ||||||
|  |   updateRole: "更新角色" | ||||||
|  |   assignRole: "分配角色" | ||||||
|  |   unassignRole: "取消分配角色" | ||||||
|  |   suspend: "冻结" | ||||||
|  |   unsuspend: "解除冻结" | ||||||
|  |   addCustomEmoji: "添加自定义表情符号" | ||||||
|  |   updateCustomEmoji: "更新自定义表情符号" | ||||||
|  |   deleteCustomEmoji: "删除自定义表情符号" | ||||||
|  |   updateServerSettings: "更新服务器设置" | ||||||
|  |   updateUserNote: "更新管理笔记" | ||||||
|  |   deleteDriveFile: "删除文件" | ||||||
|  |   deleteNote: "删除帖子" | ||||||
|  |   createGlobalAnnouncement: "创建全体通知" | ||||||
|  |   createUserAnnouncement: "创建用户通知" | ||||||
|  |   updateGlobalAnnouncement: "更新全体通知" | ||||||
|  |   updateUserAnnouncement: "更新用户通知" | ||||||
|  |   deleteGlobalAnnouncement: "删除全体通知" | ||||||
|  |   deleteUserAnnouncement: "删除用户通知" | ||||||
|  |   resetPassword: "重置密码" | ||||||
|  |   markSensitiveDriveFile: "标记网盘文件为敏感媒体" | ||||||
|  |   unmarkSensitiveDriveFile: "取消标记网盘文件为敏感媒体" | ||||||
|  |   resolveAbuseReport: "处理举报" | ||||||
|  |   createInvitation: "发行邀请码" | ||||||
|  |   createAd: "创建了广告" | ||||||
|  |   deleteAd: "删除了广告" | ||||||
|  |   updateAd: "更新了广告" | ||||||
|   | |||||||
| @@ -1,6 +1,6 @@ | |||||||
| --- | --- | ||||||
| _lang_: "繁體中文" | _lang_: "繁體中文(台灣)" | ||||||
| headlineMisskey: "貼文連繫網絡" | headlineMisskey: "貼文連繫網路" | ||||||
| introMisskey: "歡迎!Misskey 是一個開放原始碼且去中心化的社群網路服務。\n發布「貼文」向身邊的人分享您的想法!📡\n利用「反應」表達您對貼文的感覺!👍\n讓我們一起探索新的世界吧!🚀" | introMisskey: "歡迎!Misskey 是一個開放原始碼且去中心化的社群網路服務。\n發布「貼文」向身邊的人分享您的想法!📡\n利用「反應」表達您對貼文的感覺!👍\n讓我們一起探索新的世界吧!🚀" | ||||||
| poweredByMisskeyDescription: "{name}是開放原始碼平臺 <b>Misskey</b> 的伺服器之一。" | poweredByMisskeyDescription: "{name}是開放原始碼平臺 <b>Misskey</b> 的伺服器之一。" | ||||||
| monthAndDay: "{month} 月 {day} 日" | monthAndDay: "{month} 月 {day} 日" | ||||||
| @@ -45,6 +45,7 @@ pin: "置頂" | |||||||
| unpin: "取消置頂" | unpin: "取消置頂" | ||||||
| copyContent: "複製內容" | copyContent: "複製內容" | ||||||
| copyLink: "複製連結" | copyLink: "複製連結" | ||||||
|  | copyLinkRenote: "複製轉發的連結" | ||||||
| delete: "刪除" | delete: "刪除" | ||||||
| deleteAndEdit: "刪除並編輯" | deleteAndEdit: "刪除並編輯" | ||||||
| deleteAndEditConfirm: "要刪除並再次編輯嗎?此貼文的所有反應、轉發和回覆也將會消失。" | deleteAndEditConfirm: "要刪除並再次編輯嗎?此貼文的所有反應、轉發和回覆也將會消失。" | ||||||
| @@ -55,7 +56,7 @@ copyRSS: "複製RSS" | |||||||
| copyUsername: "複製使用者名稱" | copyUsername: "複製使用者名稱" | ||||||
| copyUserId: "複製使用者 ID" | copyUserId: "複製使用者 ID" | ||||||
| copyNoteId: "複製貼文 ID" | copyNoteId: "複製貼文 ID" | ||||||
| copyFileId: "複製檔案ID" | copyFileId: "複製檔案 ID" | ||||||
| copyFolderId: "複製資料夾ID" | copyFolderId: "複製資料夾ID" | ||||||
| copyProfileUrl: "複製個人資料網址" | copyProfileUrl: "複製個人資料網址" | ||||||
| searchUser: "搜尋使用者" | searchUser: "搜尋使用者" | ||||||
| @@ -74,9 +75,9 @@ import: "匯入" | |||||||
| export: "匯出" | export: "匯出" | ||||||
| files: "檔案" | files: "檔案" | ||||||
| download: "下載" | download: "下載" | ||||||
| driveFileDeleteConfirm: "確定要刪除檔案「{name}」嗎?使用此附件的貼文也會跟著消失。\n" | driveFileDeleteConfirm: "確定要刪除檔案「{name}」嗎?使用此檔案的貼文也會跟著被刪除。" | ||||||
| unfollowConfirm: "確定要取消追隨{name}嗎?" | unfollowConfirm: "確定要取消追隨{name}嗎?" | ||||||
| exportRequested: "已請求匯出。這可能會花一點時間。匯出的檔案將會被放到雲端裡。" | exportRequested: "已請求匯出。這可能會花一點時間。匯出的檔案將會被放到雲端硬碟裡。" | ||||||
| importRequested: "已請求匯入。這可能會花一點時間。" | importRequested: "已請求匯入。這可能會花一點時間。" | ||||||
| lists: "清單" | lists: "清單" | ||||||
| noLists: "你沒有任何清單" | noLists: "你沒有任何清單" | ||||||
| @@ -106,7 +107,7 @@ followRequestPending: "追隨許可待批准" | |||||||
| enterEmoji: "輸入表情符號" | enterEmoji: "輸入表情符號" | ||||||
| renote: "轉發" | renote: "轉發" | ||||||
| unrenote: "取消轉發" | unrenote: "取消轉發" | ||||||
| renoted: "轉發成功" | renoted: "轉發成功。" | ||||||
| cantRenote: "無法轉發此貼文。" | cantRenote: "無法轉發此貼文。" | ||||||
| cantReRenote: "無法轉發之前已經轉發過的內容。" | cantReRenote: "無法轉發之前已經轉發過的內容。" | ||||||
| quote: "引用" | quote: "引用" | ||||||
| @@ -137,8 +138,8 @@ suspend: "凍結" | |||||||
| unsuspend: "解除凍結" | unsuspend: "解除凍結" | ||||||
| blockConfirm: "確定要封鎖此使用者嗎?" | blockConfirm: "確定要封鎖此使用者嗎?" | ||||||
| unblockConfirm: "確定要解除封鎖此使用者嗎?" | unblockConfirm: "確定要解除封鎖此使用者嗎?" | ||||||
| suspendConfirm: "確定凍結此帳戶?" | suspendConfirm: "確定凍結此使用者?" | ||||||
| unsuspendConfirm: "確定解凍此帳戶?" | unsuspendConfirm: "確定解凍此使用者?" | ||||||
| selectList: "選擇清單" | selectList: "選擇清單" | ||||||
| editList: "編輯清單" | editList: "編輯清單" | ||||||
| selectChannel: "選擇頻道" | selectChannel: "選擇頻道" | ||||||
| @@ -151,20 +152,20 @@ customEmojis: "自訂表情符號" | |||||||
| emoji: "表情符號" | emoji: "表情符號" | ||||||
| emojis: "表情符號" | emojis: "表情符號" | ||||||
| emojiName: "表情符號名稱" | emojiName: "表情符號名稱" | ||||||
| emojiUrl: "表情符號URL" | emojiUrl: "表情符號 URL" | ||||||
| addEmoji: "新增表情符號" | addEmoji: "新增表情符號" | ||||||
| settingGuide: "推薦設定" | settingGuide: "推薦設定" | ||||||
| cacheRemoteFiles: "快取遠端檔案" | cacheRemoteFiles: "快取遠端檔案" | ||||||
| cacheRemoteFilesDescription: "禁用此設定會停止建立遠端檔案快取,從而節省伺服器儲存空間,但會因從遠端讀取資料而增加網路數據用量。" | cacheRemoteFilesDescription: "啟用此設定後,遠端檔案會被快取在本伺服器的儲存空間中。雖然顯示圖片會變快,但會消耗較多伺服器的儲存空間。至於要快取遠端使用者到什麼程度,是依照角色的雲端硬碟容量而定。當超過這個限制時,從較舊的檔案開始自快取中刪除並改為連結。關閉這個設定時,遠端檔案從一開始就維持連結的方式,但建議將 default.yml 的 proxyRemoteFiles 設為 true,以便產生圖片的縮圖並保護使用者的隱私,。" | ||||||
| youCanCleanRemoteFilesCache: "按檔案管理的🗑️按鈕,將快取全部刪除。" | youCanCleanRemoteFilesCache: "按檔案管理的🗑️按鈕,可將快取全部刪除。" | ||||||
| cacheRemoteSensitiveFiles: "快取遠端的敏感檔案" | cacheRemoteSensitiveFiles: "快取遠端的敏感檔案" | ||||||
| cacheRemoteSensitiveFilesDescription: "若停用這個設定,則不會快取遠端的敏感檔案,而是直接連結。" | cacheRemoteSensitiveFilesDescription: "若停用這個設定,則不會快取遠端的敏感檔案,而是直接連結。" | ||||||
| flagAsBot: "此使用者是機器人" | flagAsBot: "此使用者是機器人" | ||||||
| flagAsBotDescription: "標記本帳戶由程式控制,防止其他程式與本帳戶產生無限互動的行為。" | flagAsBotDescription: "如果本帳戶是由程式控制,請啟用此選項。啟用後,會作為標示幫助其他開發者防止機器人之間產生無限互動的行為,並會調整Misskey內部系統將本帳戶識別為機器人" | ||||||
| flagAsCat: "此帳戶是一隻貓,喵~~~!!!" | flagAsCat: "此帳戶是一隻貓,喵~~~!!!" | ||||||
| flagAsCatDescription: "如果想將本帳戶標示為一隻貓,請開啟此標示" | flagAsCatDescription: "如果想將本帳戶標示為一隻貓,請開啟此標示" | ||||||
| flagShowTimelineReplies: "在時間軸上顯示貼文的回覆" | flagShowTimelineReplies: "在時間軸上顯示貼文的回覆" | ||||||
| flagShowTimelineRepliesDescription: "啟用時,時間軸除了顯示使用者的貼文以外,還會顯示使用者對其他貼文的回覆。" | flagShowTimelineRepliesDescription: "啟用後,時間軸除了顯示使用者的貼文以外,還會顯示使用者對其他貼文的回覆。" | ||||||
| autoAcceptFollowed: "自動允許來自追隨中使用者的追隨請求" | autoAcceptFollowed: "自動允許來自追隨中使用者的追隨請求" | ||||||
| addAccount: "新增帳戶" | addAccount: "新增帳戶" | ||||||
| reloadAccountsList: "更新帳戶清單的資訊" | reloadAccountsList: "更新帳戶清單的資訊" | ||||||
| @@ -183,7 +184,7 @@ host: "主機" | |||||||
| selectUser: "選取使用者" | selectUser: "選取使用者" | ||||||
| recipient: "收件人" | recipient: "收件人" | ||||||
| annotation: "註解" | annotation: "註解" | ||||||
| federation: "聯邦宇宙" | federation: "站台聯邦" | ||||||
| instances: "伺服器" | instances: "伺服器" | ||||||
| registeredAt: "初次觀測" | registeredAt: "初次觀測" | ||||||
| latestRequestReceivedAt: "上次收到的請求" | latestRequestReceivedAt: "上次收到的請求" | ||||||
| @@ -320,7 +321,7 @@ copyUrl: "複製URL" | |||||||
| rename: "重新命名" | rename: "重新命名" | ||||||
| avatar: "大頭貼" | avatar: "大頭貼" | ||||||
| banner: "橫幅" | banner: "橫幅" | ||||||
| displayOfSensitiveMedia: "顯示敏感媒體" | displayOfSensitiveMedia: "敏感檔案的顯示" | ||||||
| whenServerDisconnected: "與伺服器的連接中斷時" | whenServerDisconnected: "與伺服器的連接中斷時" | ||||||
| disconnectedFromServer: "與伺服器中斷連線" | disconnectedFromServer: "與伺服器中斷連線" | ||||||
| reload: "重新整理" | reload: "重新整理" | ||||||
| @@ -355,7 +356,6 @@ invite: "邀請" | |||||||
| driveCapacityPerLocalAccount: "每個本地使用者的雲端硬碟容量" | driveCapacityPerLocalAccount: "每個本地使用者的雲端硬碟容量" | ||||||
| driveCapacityPerRemoteAccount: "每個非本地用戶的雲端空間大小" | driveCapacityPerRemoteAccount: "每個非本地用戶的雲端空間大小" | ||||||
| inMb: "以Mbps為單位" | inMb: "以Mbps為單位" | ||||||
| iconUrl: "圖標 URL(例如 favicon)" |  | ||||||
| bannerUrl: "橫幅圖片URL" | bannerUrl: "橫幅圖片URL" | ||||||
| backgroundImageUrl: "背景圖片的來源網址 " | backgroundImageUrl: "背景圖片的來源網址 " | ||||||
| basicInfo: "基本資訊" | basicInfo: "基本資訊" | ||||||
| @@ -416,12 +416,15 @@ totp: "驗證應用程式" | |||||||
| totpDescription: "以驗證應用程式輸入一次性密碼" | totpDescription: "以驗證應用程式輸入一次性密碼" | ||||||
| moderator: "審查員" | moderator: "審查員" | ||||||
| moderation: "審查" | moderation: "審查" | ||||||
| nUsersMentioned: "被提及到 {n} 次" | moderationNote: "管理筆記" | ||||||
|  | addModerationNote: "新增管理筆記" | ||||||
|  | moderationLogs: "管理日誌" | ||||||
|  | nUsersMentioned: "被 {n} 個人提及" | ||||||
| securityKeyAndPasskey: "安全金鑰、Passkey" | securityKeyAndPasskey: "安全金鑰、Passkey" | ||||||
| securityKey: "安全金鑰" | securityKey: "安全金鑰" | ||||||
| lastUsed: "上次使用" | lastUsed: "上次使用" | ||||||
| lastUsedAt: "上次使用:{t}" | lastUsedAt: "上次使用:{t}" | ||||||
| unregister: "註銷帳戶" | unregister: "註銷" | ||||||
| passwordLessLogin: "設置無密碼登入" | passwordLessLogin: "設置無密碼登入" | ||||||
| passwordLessLoginDescription: "不使用密碼,以安全金鑰或 Passkey 登入" | passwordLessLoginDescription: "不使用密碼,以安全金鑰或 Passkey 登入" | ||||||
| resetPassword: "重設密碼" | resetPassword: "重設密碼" | ||||||
| @@ -488,7 +491,7 @@ createAccount: "建立帳戶" | |||||||
| existingAccount: "現有帳戶" | existingAccount: "現有帳戶" | ||||||
| regenerate: "再次生成" | regenerate: "再次生成" | ||||||
| fontSize: "字體大小" | fontSize: "字體大小" | ||||||
| mediaListWithOneImageAppearance: "只有一張圖片時的媒體列表高度" | mediaListWithOneImageAppearance: "只有一張圖片時的檔案列表高度" | ||||||
| limitTo: "上限為 {x}" | limitTo: "上限為 {x}" | ||||||
| noFollowRequests: "沒有追隨您的請求" | noFollowRequests: "沒有追隨您的請求" | ||||||
| openImageInNewTab: "於新分頁中開啟圖片" | openImageInNewTab: "於新分頁中開啟圖片" | ||||||
| @@ -506,8 +509,8 @@ promote: "推廣" | |||||||
| numberOfDays: "有效天數" | numberOfDays: "有效天數" | ||||||
| hideThisNote: "隱藏此貼文" | hideThisNote: "隱藏此貼文" | ||||||
| showFeaturedNotesInTimeline: "在時間軸上顯示熱門推薦" | showFeaturedNotesInTimeline: "在時間軸上顯示熱門推薦" | ||||||
| objectStorage: "對象存儲" | objectStorage: "物件儲存" | ||||||
| useObjectStorage: "使用對象存儲" | useObjectStorage: "使用物件儲存" | ||||||
| objectStorageBaseUrl: "Base URL" | objectStorageBaseUrl: "Base URL" | ||||||
| objectStorageBaseUrlDesc: "用於引用的 URL。如果您使用的是 CDN 或反向代理,請指定其 URL,例如 S3(https://<bucket>.s3.amazonaws.com)、GCS(https://storage.googleapis.com/<bucket>)。" | objectStorageBaseUrlDesc: "用於引用的 URL。如果您使用的是 CDN 或反向代理,請指定其 URL,例如 S3(https://<bucket>.s3.amazonaws.com)、GCS(https://storage.googleapis.com/<bucket>)。" | ||||||
| objectStorageBucket: "儲存空間(Bucket)" | objectStorageBucket: "儲存空間(Bucket)" | ||||||
| @@ -544,7 +547,7 @@ recentUsed: "最近使用" | |||||||
| install: "安裝" | install: "安裝" | ||||||
| uninstall: "解除安裝" | uninstall: "解除安裝" | ||||||
| installedApps: "已授權的應用程式" | installedApps: "已授權的應用程式" | ||||||
| nothing: "無" | nothing: "查無項目" | ||||||
| installedDate: "安裝時間" | installedDate: "安裝時間" | ||||||
| lastUsedDate: "最後上線日期" | lastUsedDate: "最後上線日期" | ||||||
| state: "狀態" | state: "狀態" | ||||||
| @@ -570,7 +573,7 @@ tokenRevokedDescription: "登入權杖失效,請重新登入。" | |||||||
| accountDeleted: "帳戶已被刪除" | accountDeleted: "帳戶已被刪除" | ||||||
| accountDeletedDescription: "這個帳戶已被刪除。" | accountDeletedDescription: "這個帳戶已被刪除。" | ||||||
| menu: "選單" | menu: "選單" | ||||||
| divider: "分割線" | divider: "分隔線" | ||||||
| addItem: "新增項目" | addItem: "新增項目" | ||||||
| rearrange: "排序方式" | rearrange: "排序方式" | ||||||
| relays: "中繼" | relays: "中繼" | ||||||
| @@ -579,7 +582,7 @@ inboxUrl: "收件夾URL" | |||||||
| addedRelays: "已加入的中繼" | addedRelays: "已加入的中繼" | ||||||
| serviceworkerInfo: "您需要啟用推送通知。" | serviceworkerInfo: "您需要啟用推送通知。" | ||||||
| deletedNote: "已刪除的貼文" | deletedNote: "已刪除的貼文" | ||||||
| invisibleNote: "隱藏的貼文" | invisibleNote: "私密的貼文" | ||||||
| enableInfiniteScroll: "啟用自動滾動頁面模式" | enableInfiniteScroll: "啟用自動滾動頁面模式" | ||||||
| visibility: "可見性" | visibility: "可見性" | ||||||
| poll: "投票" | poll: "投票" | ||||||
| @@ -655,6 +658,7 @@ behavior: "行為" | |||||||
| sample: "範例" | sample: "範例" | ||||||
| abuseReports: "檢舉" | abuseReports: "檢舉" | ||||||
| reportAbuse: "檢舉" | reportAbuse: "檢舉" | ||||||
|  | reportAbuseRenote: "檢舉轉發貼文" | ||||||
| reportAbuseOf: "檢舉{name}" | reportAbuseOf: "檢舉{name}" | ||||||
| fillAbuseReportDescription: "請填寫檢舉的詳細理由。如有需要,請附上相關 URL。" | fillAbuseReportDescription: "請填寫檢舉的詳細理由。如有需要,請附上相關 URL。" | ||||||
| abuseReported: "檢舉完成。感謝您的報告。" | abuseReported: "檢舉完成。感謝您的報告。" | ||||||
| @@ -704,9 +708,10 @@ driveUsage: "雲端硬碟使用量" | |||||||
| noCrawle: "拒絕搜尋引擎索引" | noCrawle: "拒絕搜尋引擎索引" | ||||||
| noCrawleDescription: "要求網路搜尋引擎不要索引你的個人資料頁、貼文及頁面等。" | noCrawleDescription: "要求網路搜尋引擎不要索引你的個人資料頁、貼文及頁面等。" | ||||||
| lockedAccountInfo: "即使你通過了追隨者請求,除非你將貼文的可見性設定為 「追隨者」,否則任何人都能看見你的貼文。" | lockedAccountInfo: "即使你通過了追隨者請求,除非你將貼文的可見性設定為 「追隨者」,否則任何人都能看見你的貼文。" | ||||||
| alwaysMarkSensitive: "預設將多媒體標記為敏感內容" | alwaysMarkSensitive: "預設標記檔案為敏感內容" | ||||||
| loadRawImages: "以原始圖檔顯示附件圖檔的縮圖" | loadRawImages: "以原始圖檔顯示附件圖檔的縮圖" | ||||||
| disableShowingAnimatedImages: "不播放動態圖檔" | disableShowingAnimatedImages: "不播放動態圖檔" | ||||||
|  | highlightSensitiveMedia: "強調敏感標記" | ||||||
| verificationEmailSent: "已發送驗證電子郵件。請點擊進入電子郵件中的鏈接完成驗證。" | verificationEmailSent: "已發送驗證電子郵件。請點擊進入電子郵件中的鏈接完成驗證。" | ||||||
| notSet: "未設定" | notSet: "未設定" | ||||||
| emailVerified: "已成功驗證您的電郵" | emailVerified: "已成功驗證您的電郵" | ||||||
| @@ -923,7 +928,7 @@ type: "類型" | |||||||
| speed: "速度" | speed: "速度" | ||||||
| slow: "慢" | slow: "慢" | ||||||
| fast: "快" | fast: "快" | ||||||
| sensitiveMediaDetection: "敏感性媒體的檢測" | sensitiveMediaDetection: "敏感檔案的檢測" | ||||||
| localOnly: "僅限本地" | localOnly: "僅限本地" | ||||||
| remoteOnly: "僅限遠端" | remoteOnly: "僅限遠端" | ||||||
| failedToUpload: "上傳失敗" | failedToUpload: "上傳失敗" | ||||||
| @@ -932,7 +937,7 @@ cannotUploadBecauseNoFreeSpace: "由於雲端硬碟沒有可用空間,因此 | |||||||
| cannotUploadBecauseExceedsFileSizeLimit: "由於超過了檔案大小的限制,無法上傳。" | cannotUploadBecauseExceedsFileSizeLimit: "由於超過了檔案大小的限制,無法上傳。" | ||||||
| beta: "測試版" | beta: "測試版" | ||||||
| enableAutoSensitive: "自動 NSFW 判定" | enableAutoSensitive: "自動 NSFW 判定" | ||||||
| enableAutoSensitiveDescription: "如果可用,它將使用機器學習技術判斷多媒體內容是否需要標記 NSFW。即使關閉此功能,也可能會依實例規則而自動啟用。" | enableAutoSensitiveDescription: "如果可用,它將使用機器學習技術判斷檔案是否需要標記為敏感。即使關閉此功能,也可能會依實例規則而自動啟用。" | ||||||
| activeEmailValidationDescription: "積極驗證使用者的電郵地址,以判斷它是否可以通訊。關閉此選項代表只會檢查地址是否符合格式。" | activeEmailValidationDescription: "積極驗證使用者的電郵地址,以判斷它是否可以通訊。關閉此選項代表只會檢查地址是否符合格式。" | ||||||
| navbar: "導覽列" | navbar: "導覽列" | ||||||
| shuffle: "隨機" | shuffle: "隨機" | ||||||
| @@ -1021,7 +1026,7 @@ retryAllQueuesConfirmText: "伺服器的負荷可能會暫時增加。" | |||||||
| enableChartsForRemoteUser: "生成遠端使用者的圖表" | enableChartsForRemoteUser: "生成遠端使用者的圖表" | ||||||
| enableChartsForFederatedInstances: "生成遠端伺服器的圖表" | enableChartsForFederatedInstances: "生成遠端伺服器的圖表" | ||||||
| showClipButtonInNoteFooter: "新增摘錄至貼文" | showClipButtonInNoteFooter: "新增摘錄至貼文" | ||||||
| largeNoteReactions: "放大顯示貼文反應" | reactionsDisplaySize: "表情回應的顯示尺寸" | ||||||
| noteIdOrUrl: "貼文ID或URL" | noteIdOrUrl: "貼文ID或URL" | ||||||
| video: "影片" | video: "影片" | ||||||
| videos: "影片" | videos: "影片" | ||||||
| @@ -1103,6 +1108,22 @@ forYou: "給您" | |||||||
| currentAnnouncements: "最新公告" | currentAnnouncements: "最新公告" | ||||||
| pastAnnouncements: "歷史公告" | pastAnnouncements: "歷史公告" | ||||||
| youHaveUnreadAnnouncements: "有未讀的公告。" | youHaveUnreadAnnouncements: "有未讀的公告。" | ||||||
|  | useSecurityKey: "請按照瀏覽器或設備上的說明使用安全金鑰或 Passkey。" | ||||||
|  | replies: "回覆" | ||||||
|  | renotes: "轉發" | ||||||
|  | loadReplies: "閱覽回覆" | ||||||
|  | loadConversation: "閱覽對話" | ||||||
|  | pinnedList: "已置頂的清單" | ||||||
|  | keepScreenOn: "保持設備螢幕開啟" | ||||||
|  | verifiedLink: "已驗證連結" | ||||||
|  | notifyNotes: "開啟貼文通知" | ||||||
|  | unnotifyNotes: "關閉貼文通知" | ||||||
|  | authentication: "驗證" | ||||||
|  | authenticationRequiredToContinue: "請於繼續前完成驗證" | ||||||
|  | dateAndTime: "日期與時間" | ||||||
|  | showRenotes: "顯示轉發貼文" | ||||||
|  | edited: "已編輯" | ||||||
|  | mutualFollow: "互相追隨" | ||||||
| _announcement: | _announcement: | ||||||
|   forExistingUsers: "僅限既有的使用者" |   forExistingUsers: "僅限既有的使用者" | ||||||
|   forExistingUsersDescription: "啟用代表僅向現存使用者顯示;停用代表張貼後註冊的新使用者也會看到。" |   forExistingUsersDescription: "啟用代表僅向現存使用者顯示;停用代表張貼後註冊的新使用者也會看到。" | ||||||
| @@ -1129,6 +1150,15 @@ _initialAccountSetting: | |||||||
|   laterAreYouSure: "稍後再重新進行初始設定嗎?" |   laterAreYouSure: "稍後再重新進行初始設定嗎?" | ||||||
| _serverRules: | _serverRules: | ||||||
|   description: "設定在註冊頁面顯示的伺服器簡要規則。建議是服務條款的摘要。" |   description: "設定在註冊頁面顯示的伺服器簡要規則。建議是服務條款的摘要。" | ||||||
|  | _serverSettings: | ||||||
|  |   iconUrl: "圖示的 URL" | ||||||
|  |   appIconDescription: "指定顯示 {host} 為應用程式時的圖示。" | ||||||
|  |   appIconUsageExample: "例如:漸進式網路應用程式(PWA)、於手機桌面新增書籤" | ||||||
|  |   appIconStyleRecommendation: "因為可能會裁剪成圓形或圓角,所以建議用單色填滿邊框及背景。" | ||||||
|  |   appIconResolutionMustBe: "解析度必須為 {resolution}。" | ||||||
|  |   manifestJsonOverride: "覆寫 manifest.json" | ||||||
|  |   shortName: "簡稱" | ||||||
|  |   shortNameDescription: "如果伺服器的正式名稱很長,可用簡稱或通稱代替。" | ||||||
| _accountMigration: | _accountMigration: | ||||||
|   moveFrom: "從其他帳戶遷移到這個帳戶" |   moveFrom: "從其他帳戶遷移到這個帳戶" | ||||||
|   moveFromSub: "為另一個帳戶建立別名" |   moveFromSub: "為另一個帳戶建立別名" | ||||||
| @@ -1383,6 +1413,9 @@ _achievements: | |||||||
|       title: "Brain Driver" |       title: "Brain Driver" | ||||||
|       description: "發佈一篇含歌曲《Brain Driver》連結的貼文" |       description: "發佈一篇含歌曲《Brain Driver》連結的貼文" | ||||||
|       flavor: "Misskey-Misskey La-Tu-Ma" |       flavor: "Misskey-Misskey La-Tu-Ma" | ||||||
|  |     _smashTestNotificationButton: | ||||||
|  |       title: "過度測試" | ||||||
|  |       description: "極短時間內連續測試通知" | ||||||
| _role: | _role: | ||||||
|   new: "建立角色" |   new: "建立角色" | ||||||
|   edit: "編輯角色" |   edit: "編輯角色" | ||||||
| @@ -1421,6 +1454,7 @@ _role: | |||||||
|     gtlAvailable: "瀏覽全域時間軸" |     gtlAvailable: "瀏覽全域時間軸" | ||||||
|     ltlAvailable: "瀏覽本地時間軸" |     ltlAvailable: "瀏覽本地時間軸" | ||||||
|     canPublicNote: "允許公開貼文" |     canPublicNote: "允許公開貼文" | ||||||
|  |     canEditNote: "允許編輯貼文" | ||||||
|     canInvite: "發行實例邀請碼" |     canInvite: "發行實例邀請碼" | ||||||
|     inviteLimit: "可建立邀請碼的數量" |     inviteLimit: "可建立邀請碼的數量" | ||||||
|     inviteLimitCycle: "邀請碼的發放間隔" |     inviteLimitCycle: "邀請碼的發放間隔" | ||||||
| @@ -1455,7 +1489,7 @@ _role: | |||||||
|     or: "~或~" |     or: "~或~" | ||||||
|     not: "~否" |     not: "~否" | ||||||
| _sensitiveMediaDetection: | _sensitiveMediaDetection: | ||||||
|   description: "您可以使用機器學習自動檢測敏感媒體並將其用於審查。 伺服器的負荷會稍微增加。" |   description: "您可以使用機器學習自動檢測敏感檔案以便審查。這會稍微增加伺服器負荷。" | ||||||
|   sensitivity: "檢測敏感度" |   sensitivity: "檢測敏感度" | ||||||
|   sensitivityDescription: "敏感度低時,誤檢測(偽陽性)會減少。敏感度高時,漏檢(偽陰性)會減少。" |   sensitivityDescription: "敏感度低時,誤檢測(偽陽性)會減少。敏感度高時,漏檢(偽陰性)會減少。" | ||||||
|   setSensitiveFlagAutomatically: "設定 NSFW 標籤" |   setSensitiveFlagAutomatically: "設定 NSFW 標籤" | ||||||
| @@ -1506,6 +1540,7 @@ _plugin: | |||||||
|   install: "安裝外掛組件" |   install: "安裝外掛組件" | ||||||
|   installWarn: "請不要安裝來源不明的外掛。" |   installWarn: "請不要安裝來源不明的外掛。" | ||||||
|   manage: "管理外掛" |   manage: "管理外掛" | ||||||
|  |   viewSource: "檢視原始碼" | ||||||
| _preferencesBackups: | _preferencesBackups: | ||||||
|   list: "已備份的設定檔" |   list: "已備份的設定檔" | ||||||
|   saveNew: "另存新檔" |   saveNew: "另存新檔" | ||||||
| @@ -1540,13 +1575,13 @@ _aboutMisskey: | |||||||
|   morePatrons: "還有許許多多幫助我們的其他人,非常感謝你們。 🥰" |   morePatrons: "還有許許多多幫助我們的其他人,非常感謝你們。 🥰" | ||||||
|   patrons: "贊助者" |   patrons: "贊助者" | ||||||
| _displayOfSensitiveMedia: | _displayOfSensitiveMedia: | ||||||
|   respect: "隱藏被標記為敏感的多媒體內容" |   respect: "隱藏敏感檔案" | ||||||
|   ignore: "不隱藏被標記為敏感的多媒體內容" |   ignore: "顯示敏感檔案" | ||||||
|   force: "隱藏所有多媒體內容" |   force: "隱藏所有檔案" | ||||||
| _instanceTicker: | _instanceTicker: | ||||||
|   none: "隱藏" |   none: "隱藏" | ||||||
|   remote: "向遠端使用者顯示" |   remote: "只顯示遠端使用者" | ||||||
|   always: "總是顯示" |   always: "一律顯示" | ||||||
| _serverDisconnectedBehavior: | _serverDisconnectedBehavior: | ||||||
|   reload: "自動重載" |   reload: "自動重載" | ||||||
|   dialog: "彈出式警告" |   dialog: "彈出式警告" | ||||||
| @@ -1579,7 +1614,7 @@ _wordMute: | |||||||
|   mutedNotes: "已靜音的貼文" |   mutedNotes: "已靜音的貼文" | ||||||
| _instanceMute: | _instanceMute: | ||||||
|   instanceMuteDescription: "包括對被靜音實例上的使用者的回覆,被設定的實例上所有貼文及轉發都會被靜音。" |   instanceMuteDescription: "包括對被靜音實例上的使用者的回覆,被設定的實例上所有貼文及轉發都會被靜音。" | ||||||
|   instanceMuteDescription2: "換行以分隔" |   instanceMuteDescription2: "設定時以換行進行分隔" | ||||||
|   title: "將隱藏被設定的實例貼文。" |   title: "將隱藏被設定的實例貼文。" | ||||||
|   heading: "將實例靜音" |   heading: "將實例靜音" | ||||||
| _theme: | _theme: | ||||||
| @@ -1632,7 +1667,7 @@ _theme: | |||||||
|     mentionMe: "提到了我" |     mentionMe: "提到了我" | ||||||
|     renote: "轉發貼文" |     renote: "轉發貼文" | ||||||
|     modalBg: "對話框背景" |     modalBg: "對話框背景" | ||||||
|     divider: "分割線" |     divider: "分隔線" | ||||||
|     scrollbarHandle: "捲動條" |     scrollbarHandle: "捲動條" | ||||||
|     scrollbarHandleHover: "捲動條(懸浮)" |     scrollbarHandleHover: "捲動條(懸浮)" | ||||||
|     dateLabelFg: "日期標籤文字" |     dateLabelFg: "日期標籤文字" | ||||||
| @@ -1668,7 +1703,7 @@ _ago: | |||||||
|   future: "未來" |   future: "未來" | ||||||
|   justNow: "剛剛" |   justNow: "剛剛" | ||||||
|   secondsAgo: "{n} 秒前" |   secondsAgo: "{n} 秒前" | ||||||
|   minutesAgo: "{n}分鐘前 " |   minutesAgo: "{n} 分鐘前 " | ||||||
|   hoursAgo: "{n} 小時前" |   hoursAgo: "{n} 小時前" | ||||||
|   daysAgo: "{n} 天前" |   daysAgo: "{n} 天前" | ||||||
|   weeksAgo: "{n} 週前" |   weeksAgo: "{n} 週前" | ||||||
| @@ -1693,7 +1728,6 @@ _timelineTutorial: | |||||||
| _2fa: | _2fa: | ||||||
|   alreadyRegistered: "此裝置已被註冊過了" |   alreadyRegistered: "此裝置已被註冊過了" | ||||||
|   registerTOTP: "開始設定驗證應用程式" |   registerTOTP: "開始設定驗證應用程式" | ||||||
|   passwordToTOTP: "請輸入密碼" |  | ||||||
|   step1: "首先,在您的裝置上安裝驗證程式,例如 {a} 或 {b}。" |   step1: "首先,在您的裝置上安裝驗證程式,例如 {a} 或 {b}。" | ||||||
|   step2: "然後,掃描螢幕上的 QR 碼。" |   step2: "然後,掃描螢幕上的 QR 碼。" | ||||||
|   step2Click: "您可以點擊 QR 碼,以使用裝置上的驗證應用程式或金鑰環註冊。" |   step2Click: "您可以點擊 QR 碼,以使用裝置上的驗證應用程式或金鑰環註冊。" | ||||||
| @@ -1705,7 +1739,6 @@ _2fa: | |||||||
|   securityKeyNotSupported: "您的瀏覽器不支援安全金鑰。" |   securityKeyNotSupported: "您的瀏覽器不支援安全金鑰。" | ||||||
|   registerTOTPBeforeKey: "如要註冊安全金鑰或 Passkey,請先設定驗證應用程式。" |   registerTOTPBeforeKey: "如要註冊安全金鑰或 Passkey,請先設定驗證應用程式。" | ||||||
|   securityKeyInfo: "您可以設定使用支援 FIDO2 的硬體安全鎖、終端設備的指紋認證,或者 PIN 碼來登入。" |   securityKeyInfo: "您可以設定使用支援 FIDO2 的硬體安全鎖、終端設備的指紋認證,或者 PIN 碼來登入。" | ||||||
|   chromePasskeyNotSupported: "目前不支援 Chrome 的 Passkey。" |  | ||||||
|   registerSecurityKey: "註冊安全金鑰或 Passkey" |   registerSecurityKey: "註冊安全金鑰或 Passkey" | ||||||
|   securityKeyName: "輸入金鑰名稱" |   securityKeyName: "輸入金鑰名稱" | ||||||
|   tapSecurityKey: "按照瀏覽器的說明註冊安全金鑰或 Passkey。" |   tapSecurityKey: "按照瀏覽器的說明註冊安全金鑰或 Passkey。" | ||||||
| @@ -1773,6 +1806,7 @@ _antennaSources: | |||||||
|   homeTimeline: "來自已追隨使用者的貼文" |   homeTimeline: "來自已追隨使用者的貼文" | ||||||
|   users: "來自特定使用者的貼文" |   users: "來自特定使用者的貼文" | ||||||
|   userList: "來自特定清單中的貼文" |   userList: "來自特定清單中的貼文" | ||||||
|  |   userBlacklist: "除指定使用者外的所有貼文" | ||||||
| _weekday: | _weekday: | ||||||
|   sunday: "週日" |   sunday: "週日" | ||||||
|   monday: "週一" |   monday: "週一" | ||||||
| @@ -1872,6 +1906,7 @@ _profile: | |||||||
|   metadataContent: "内容" |   metadataContent: "内容" | ||||||
|   changeAvatar: "更換大頭貼" |   changeAvatar: "更換大頭貼" | ||||||
|   changeBanner: "變更橫幅圖像" |   changeBanner: "變更橫幅圖像" | ||||||
|  |   verifiedLinkDescription: "如果輸入包含您個人資料的網站 URL,欄位旁邊將出現驗證圖示。" | ||||||
| _exportOrImport: | _exportOrImport: | ||||||
|   allNotes: "所有貼文" |   allNotes: "所有貼文" | ||||||
|   favoritedNotes: "「我的最愛」貼文" |   favoritedNotes: "「我的最愛」貼文" | ||||||
| @@ -1990,11 +2025,17 @@ _notification: | |||||||
|   youReceivedFollowRequest: "您有新的追隨請求" |   youReceivedFollowRequest: "您有新的追隨請求" | ||||||
|   yourFollowRequestAccepted: "您的追隨請求已通過" |   yourFollowRequestAccepted: "您的追隨請求已通過" | ||||||
|   pollEnded: "問卷調查已產生結果" |   pollEnded: "問卷調查已產生結果" | ||||||
|  |   newNote: "新的貼文" | ||||||
|   unreadAntennaNote: "天線 {name}" |   unreadAntennaNote: "天線 {name}" | ||||||
|   emptyPushNotificationMessage: "推送通知已更新" |   emptyPushNotificationMessage: "推送通知已更新" | ||||||
|   achievementEarned: "獲得成就" |   achievementEarned: "獲得成就" | ||||||
|  |   testNotification: "通知測試" | ||||||
|  |   checkNotificationBehavior: "確認通知的顯示行為" | ||||||
|  |   sendTestNotification: "發送測試通知" | ||||||
|  |   notificationWillBeDisplayedLikeThis: "通知會以這樣的方式顯示" | ||||||
|   _types: |   _types: | ||||||
|     all: "全部 " |     all: "全部 " | ||||||
|  |     note: "使用者的最新貼文" | ||||||
|     follow: "追隨中" |     follow: "追隨中" | ||||||
|     mention: "提及" |     mention: "提及" | ||||||
|     reply: "回覆" |     reply: "回覆" | ||||||
| @@ -2064,3 +2105,34 @@ _webhookSettings: | |||||||
|     renote: "當被轉發時" |     renote: "當被轉發時" | ||||||
|     reaction: "當獲得反應時" |     reaction: "當獲得反應時" | ||||||
|     mention: "當被提到時" |     mention: "當被提到時" | ||||||
|  | _moderationLogTypes: | ||||||
|  |   createRole: "新增角色" | ||||||
|  |   deleteRole: "刪除角色 " | ||||||
|  |   updateRole: "更新角色設定" | ||||||
|  |   assignRole: "指派角色" | ||||||
|  |   unassignRole: "撤銷角色" | ||||||
|  |   suspend: "凍結" | ||||||
|  |   unsuspend: "解除凍結" | ||||||
|  |   addCustomEmoji: "新增自訂表情符號" | ||||||
|  |   updateCustomEmoji: "更新自訂表情符號" | ||||||
|  |   deleteCustomEmoji: "刪除自訂表情符號" | ||||||
|  |   updateServerSettings: "更新伺服器設定" | ||||||
|  |   updateUserNote: "更新管理筆記" | ||||||
|  |   deleteDriveFile: "刪除檔案" | ||||||
|  |   deleteNote: "刪除貼文" | ||||||
|  |   createGlobalAnnouncement: "建立全網通知" | ||||||
|  |   createUserAnnouncement: "建立使用者通知" | ||||||
|  |   updateGlobalAnnouncement: "更新全部的公告" | ||||||
|  |   updateUserAnnouncement: "更新使用者的公告" | ||||||
|  |   deleteGlobalAnnouncement: "刪除全部的公告" | ||||||
|  |   deleteUserAnnouncement: "刪除使用者的公告" | ||||||
|  |   resetPassword: "重設密碼" | ||||||
|  |   suspendRemoteInstance: "封鎖遠端伺服器" | ||||||
|  |   unsuspendRemoteInstance: "解除封鎖遠端伺服器" | ||||||
|  |   markSensitiveDriveFile: "標記為敏感檔案" | ||||||
|  |   unmarkSensitiveDriveFile: "撤銷標記為敏感檔案" | ||||||
|  |   resolveAbuseReport: "解決檢舉" | ||||||
|  |   createInvitation: "建立邀請碼" | ||||||
|  |   createAd: "建立廣告" | ||||||
|  |   deleteAd: "刪除廣告" | ||||||
|  |   updateAd: "更新廣告" | ||||||
|   | |||||||
							
								
								
									
										22
									
								
								package.json
									
									
									
									
									
								
							
							
						
						
									
										22
									
								
								package.json
									
									
									
									
									
								
							| @@ -1,12 +1,12 @@ | |||||||
| { | { | ||||||
| 	"name": "misskey", | 	"name": "misskey", | ||||||
| 	"version": "2023.9.0-beta.4", | 	"version": "2023.9.3", | ||||||
| 	"codename": "nasubi", | 	"codename": "nasubi", | ||||||
| 	"repository": { | 	"repository": { | ||||||
| 		"type": "git", | 		"type": "git", | ||||||
| 		"url": "https://github.com/misskey-dev/misskey.git" | 		"url": "https://github.com/misskey-dev/misskey.git" | ||||||
| 	}, | 	}, | ||||||
| 	"packageManager": "pnpm@8.7.4", | 	"packageManager": "pnpm@8.8.0", | ||||||
| 	"workspaces": [ | 	"workspaces": [ | ||||||
| 		"packages/frontend", | 		"packages/frontend", | ||||||
| 		"packages/backend", | 		"packages/backend", | ||||||
| @@ -18,8 +18,8 @@ | |||||||
| 		"build-assets": "node ./scripts/build-assets.mjs", | 		"build-assets": "node ./scripts/build-assets.mjs", | ||||||
| 		"build": "pnpm build-pre && pnpm -r build && pnpm build-assets", | 		"build": "pnpm build-pre && pnpm -r build && pnpm build-assets", | ||||||
| 		"build-storybook": "pnpm --filter frontend build-storybook", | 		"build-storybook": "pnpm --filter frontend build-storybook", | ||||||
| 		"start": "pnpm check:connect && cd packages/backend && node ./built/boot/index.js", | 		"start": "pnpm check:connect && cd packages/backend && node ./built/boot/entry.js", | ||||||
| 		"start:test": "cd packages/backend && cross-env NODE_ENV=test node ./built/boot/index.js", | 		"start:test": "cd packages/backend && cross-env NODE_ENV=test node ./built/boot/entry.js", | ||||||
| 		"init": "pnpm migrate", | 		"init": "pnpm migrate", | ||||||
| 		"migrate": "cd packages/backend && pnpm migrate", | 		"migrate": "cd packages/backend && pnpm migrate", | ||||||
| 		"check:connect": "cd packages/backend && pnpm check:connect", | 		"check:connect": "cd packages/backend && pnpm check:connect", | ||||||
| @@ -46,17 +46,17 @@ | |||||||
| 		"execa": "8.0.1", | 		"execa": "8.0.1", | ||||||
| 		"cssnano": "6.0.1", | 		"cssnano": "6.0.1", | ||||||
| 		"js-yaml": "4.1.0", | 		"js-yaml": "4.1.0", | ||||||
| 		"postcss": "8.4.27", | 		"postcss": "8.4.31", | ||||||
| 		"terser": "5.19.2", | 		"terser": "5.20.0", | ||||||
| 		"typescript": "5.2.2" | 		"typescript": "5.2.2" | ||||||
| 	}, | 	}, | ||||||
| 	"devDependencies": { | 	"devDependencies": { | ||||||
| 		"@typescript-eslint/eslint-plugin": "6.6.0", | 		"@typescript-eslint/eslint-plugin": "6.7.3", | ||||||
| 		"@typescript-eslint/parser": "6.6.0", | 		"@typescript-eslint/parser": "6.7.3", | ||||||
| 		"cross-env": "7.0.3", | 		"cross-env": "7.0.3", | ||||||
| 		"cypress": "13.1.0", | 		"cypress": "13.3.0", | ||||||
| 		"eslint": "8.48.0", | 		"eslint": "8.50.0", | ||||||
| 		"start-server-and-test": "2.0.0" | 		"start-server-and-test": "2.0.1" | ||||||
| 	}, | 	}, | ||||||
| 	"optionalDependencies": { | 	"optionalDependencies": { | ||||||
| 		"@tensorflow/tfjs-core": "4.4.0" | 		"@tensorflow/tfjs-core": "4.4.0" | ||||||
|   | |||||||
| @@ -11,7 +11,7 @@ | |||||||
| 			"decoratorMetadata": true | 			"decoratorMetadata": true | ||||||
| 		}, | 		}, | ||||||
| 		"experimental": { | 		"experimental": { | ||||||
| 			"keepImportAssertions": true | 			"keepImportAttributes": true | ||||||
| 		}, | 		}, | ||||||
| 		"baseUrl": "src", | 		"baseUrl": "src", | ||||||
| 		"paths": { | 		"paths": { | ||||||
|   | |||||||
| @@ -216,4 +216,6 @@ module.exports = { | |||||||
| 	maxWorkers: 1, // Make it use worker (that can be killed and restarted) | 	maxWorkers: 1, // Make it use worker (that can be killed and restarted) | ||||||
| 	logHeapUsage: true, // To debug when out-of-memory happens on CI | 	logHeapUsage: true, // To debug when out-of-memory happens on CI | ||||||
| 	workerIdleMemoryLimit: '1GiB', // Limit the worker to 1GB (GitHub Workflows dies at 2GB) | 	workerIdleMemoryLimit: '1GiB', // Limit the worker to 1GB (GitHub Workflows dies at 2GB) | ||||||
|  |  | ||||||
|  | 	maxConcurrency: 32, | ||||||
| }; | }; | ||||||
|   | |||||||
| @@ -0,0 +1,10 @@ | |||||||
|  | export class UserBlacklistAnntena1689325027964 { | ||||||
|  |     name = 'UserBlacklistAnntena1689325027964' | ||||||
|  |  | ||||||
|  |     async up(queryRunner) { | ||||||
|  |         await queryRunner.query(`ALTER TYPE "antenna_src_enum" ADD VALUE 'users_blacklist' AFTER 'list'`); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     async down(queryRunner) { | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										49
									
								
								packages/backend/migration/1691959191872-passkey-support.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										49
									
								
								packages/backend/migration/1691959191872-passkey-support.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,49 @@ | |||||||
|  | /* | ||||||
|  |  * SPDX-FileCopyrightText: syuilo and other misskey contributors | ||||||
|  |  * SPDX-License-Identifier: AGPL-3.0-only | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | export class PasskeySupport1691959191872 { | ||||||
|  | 	name = 'PasskeySupport1691959191872' | ||||||
|  |  | ||||||
|  | 	async up(queryRunner) { | ||||||
|  | 			await queryRunner.query(`ALTER TABLE "user_security_key" ADD "counter" bigint NOT NULL DEFAULT '0'`); | ||||||
|  | 			await queryRunner.query(`COMMENT ON COLUMN "user_security_key"."counter" IS 'The number of times the UserSecurityKey was validated.'`); | ||||||
|  | 			await queryRunner.query(`ALTER TABLE "user_security_key" ADD "credentialDeviceType" character varying(32)`); | ||||||
|  | 			await queryRunner.query(`COMMENT ON COLUMN "user_security_key"."credentialDeviceType" IS 'The type of Backup Eligibility in authenticator data'`); | ||||||
|  | 			await queryRunner.query(`ALTER TABLE "user_security_key" ADD "credentialBackedUp" boolean`); | ||||||
|  | 			await queryRunner.query(`COMMENT ON COLUMN "user_security_key"."credentialBackedUp" IS 'Whether or not the credential has been backed up'`); | ||||||
|  | 			await queryRunner.query(`ALTER TABLE "user_security_key" ADD "transports" character varying(32) array`); | ||||||
|  | 			await queryRunner.query(`COMMENT ON COLUMN "user_security_key"."transports" IS 'The type of the credential returned by the browser'`); | ||||||
|  | 			await queryRunner.query(`COMMENT ON COLUMN "user_security_key"."publicKey" IS 'The public key of the UserSecurityKey, hex-encoded.'`); | ||||||
|  | 			await queryRunner.query(`COMMENT ON COLUMN "user_security_key"."lastUsed" IS 'Timestamp of the last time the UserSecurityKey was used.'`); | ||||||
|  | 			await queryRunner.query(`ALTER TABLE "user_security_key" ALTER COLUMN "lastUsed" SET DEFAULT now()`); | ||||||
|  | 			await queryRunner.query(`UPDATE "user_security_key" SET "id" = REPLACE(REPLACE(REPLACE(REPLACE(ENCODE(DECODE("id", 'hex'), 'base64'), E'\\n', ''), '+', '-'), '/', '_'), '=', ''), "publicKey" = REPLACE(REPLACE(REPLACE(REPLACE(ENCODE(DECODE("publicKey", 'hex'), 'base64'), E'\\n', ''), '+', '-'), '/', '_'), '=', '')`); | ||||||
|  | 			await queryRunner.query(`ALTER TABLE "attestation_challenge" DROP CONSTRAINT "FK_f1a461a618fa1755692d0e0d592"`); | ||||||
|  | 			await queryRunner.query(`DROP INDEX "IDX_47efb914aed1f72dd39a306c7b"`); | ||||||
|  | 			await queryRunner.query(`DROP INDEX "IDX_f1a461a618fa1755692d0e0d59"`); | ||||||
|  | 			await queryRunner.query(`DROP TABLE "attestation_challenge"`); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	async down(queryRunner) { | ||||||
|  | 			await queryRunner.query(`CREATE TABLE "attestation_challenge" ("id" character varying(32) NOT NULL, "userId" character varying(32) NOT NULL, "challenge" character varying(64) NOT NULL, "createdAt" TIMESTAMP WITH TIME ZONE NOT NULL, "registrationChallenge" boolean NOT NULL DEFAULT false, CONSTRAINT "PK_d0ba6786e093f1bcb497572a6b5" PRIMARY KEY ("id", "userId"))`); | ||||||
|  | 			await queryRunner.query(`CREATE INDEX "IDX_f1a461a618fa1755692d0e0d59" ON "attestation_challenge" ("userId") `); | ||||||
|  | 			await queryRunner.query(`CREATE INDEX "IDX_47efb914aed1f72dd39a306c7b" ON "attestation_challenge" ("challenge") `); | ||||||
|  | 			await queryRunner.query(`ALTER TABLE "attestation_challenge" ADD CONSTRAINT "FK_f1a461a618fa1755692d0e0d592" FOREIGN KEY ("userId") REFERENCES "user"("id") ON DELETE CASCADE ON UPDATE NO ACTION`); | ||||||
|  | 			await queryRunner.query(`COMMENT ON COLUMN "attestation_challenge"."challenge" IS 'Hex-encoded sha256 hash of the challenge.'`); | ||||||
|  | 			await queryRunner.query(`COMMENT ON COLUMN "attestation_challenge"."createdAt" IS 'The date challenge was created for expiry purposes.'`); | ||||||
|  | 			await queryRunner.query(`COMMENT ON COLUMN "attestation_challenge"."registrationChallenge" IS 'Indicates that the challenge is only for registration purposes if true to prevent the challenge for being used as authentication.'`); | ||||||
|  | 			await queryRunner.query(`UPDATE "user_security_key" SET "id" = ENCODE(DECODE(REPLACE(REPLACE("id" || CASE WHEN LENGTH("id") % 4 = 2 THEN '==' WHEN LENGTH("id") % 4 = 3 THEN '=' ELSE '' END, '-', '+'), '_', '/'), 'base64'), 'hex'), "publicKey" = ENCODE(DECODE(REPLACE(REPLACE("publicKey" || CASE WHEN LENGTH("publicKey") % 4 = 2 THEN '==' WHEN LENGTH("publicKey") % 4 = 3 THEN '=' ELSE '' END, '-', '+'), '_', '/'), 'base64'), 'hex')`); | ||||||
|  | 			await queryRunner.query(`ALTER TABLE "user_security_key" ALTER COLUMN "lastUsed" DROP DEFAULT`); | ||||||
|  | 			await queryRunner.query(`COMMENT ON COLUMN "user_security_key"."lastUsed" IS 'The date of the last time the UserSecurityKey was successfully validated.'`); | ||||||
|  | 			await queryRunner.query(`COMMENT ON COLUMN "user_security_key"."publicKey" IS 'Variable-length public key used to verify attestations (hex-encoded).'`); | ||||||
|  | 			await queryRunner.query(`COMMENT ON COLUMN "user_security_key"."transports" IS 'The type of the credential returned by the browser'`); | ||||||
|  | 			await queryRunner.query(`ALTER TABLE "user_security_key" DROP COLUMN "transports"`); | ||||||
|  | 			await queryRunner.query(`COMMENT ON COLUMN "user_security_key"."credentialBackedUp" IS 'Whether or not the credential has been backed up'`); | ||||||
|  | 			await queryRunner.query(`ALTER TABLE "user_security_key" DROP COLUMN "credentialBackedUp"`); | ||||||
|  | 			await queryRunner.query(`COMMENT ON COLUMN "user_security_key"."credentialDeviceType" IS 'The type of Backup Eligibility in authenticator data'`); | ||||||
|  | 			await queryRunner.query(`ALTER TABLE "user_security_key" DROP COLUMN "credentialDeviceType"`); | ||||||
|  | 			await queryRunner.query(`COMMENT ON COLUMN "user_security_key"."counter" IS 'The number of times the UserSecurityKey was validated.'`); | ||||||
|  | 			await queryRunner.query(`ALTER TABLE "user_security_key" DROP COLUMN "counter"`); | ||||||
|  | 	} | ||||||
|  | } | ||||||
| @@ -0,0 +1,20 @@ | |||||||
|  | /* | ||||||
|  |  * SPDX-FileCopyrightText: syuilo and other misskey contributors | ||||||
|  |  * SPDX-License-Identifier: AGPL-3.0-only | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | export class ServerIconsAndManifest1694850832075 { | ||||||
|  |     name = 'ServerIconsAndManifest1694850832075' | ||||||
|  |  | ||||||
|  |     async up(queryRunner) { | ||||||
|  |         await queryRunner.query(`ALTER TABLE "meta" ADD "app192IconUrl" character varying(1024)`); | ||||||
|  |         await queryRunner.query(`ALTER TABLE "meta" ADD "app512IconUrl" character varying(1024)`); | ||||||
|  |         await queryRunner.query(`ALTER TABLE "meta" ADD "manifestJsonOverride" character varying(8192) NOT NULL DEFAULT '{}'`); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     async down(queryRunner) { | ||||||
|  |         await queryRunner.query(`ALTER TABLE "meta" DROP COLUMN "manifestJsonOverride"`); | ||||||
|  |         await queryRunner.query(`ALTER TABLE "meta" DROP COLUMN "app512IconUrl"`); | ||||||
|  |         await queryRunner.query(`ALTER TABLE "meta" DROP COLUMN "app192IconUrl"`); | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										16
									
								
								packages/backend/migration/1694915420864-clipped-count.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								packages/backend/migration/1694915420864-clipped-count.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,16 @@ | |||||||
|  | /* | ||||||
|  |  * SPDX-FileCopyrightText: syuilo and other misskey contributors | ||||||
|  |  * SPDX-License-Identifier: AGPL-3.0-only | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | export class ClippedCount1694915420864 { | ||||||
|  |     name = 'ClippedCount1694915420864' | ||||||
|  |  | ||||||
|  |     async up(queryRunner) { | ||||||
|  |         await queryRunner.query(`ALTER TABLE "note" ADD "clippedCount" smallint NOT NULL DEFAULT '0'`); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     async down(queryRunner) { | ||||||
|  |         await queryRunner.query(`ALTER TABLE "note" DROP COLUMN "clippedCount"`); | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										11
									
								
								packages/backend/migration/1695260774117-verified-links.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								packages/backend/migration/1695260774117-verified-links.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,11 @@ | |||||||
|  | export class VerifiedLinks1695260774117 { | ||||||
|  |     name = 'VerifiedLinks1695260774117' | ||||||
|  |  | ||||||
|  |     async up(queryRunner) { | ||||||
|  |         await queryRunner.query(`ALTER TABLE "user_profile" ADD "verifiedLinks" character varying array NOT NULL DEFAULT '{}'`); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     async down(queryRunner) { | ||||||
|  |         await queryRunner.query(`ALTER TABLE "user_profile" DROP COLUMN "verifiedLinks"`); | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										13
									
								
								packages/backend/migration/1695288787870-following-notify.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								packages/backend/migration/1695288787870-following-notify.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,13 @@ | |||||||
|  | export class FollowingNotify1695288787870 { | ||||||
|  |     name = 'FollowingNotify1695288787870' | ||||||
|  |  | ||||||
|  |     async up(queryRunner) { | ||||||
|  |         await queryRunner.query(`ALTER TABLE "following" ADD "notify" character varying(32)`); | ||||||
|  |         await queryRunner.query(`CREATE INDEX "IDX_5108098457488634a4768e1d12" ON "following" ("notify") `); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     async down(queryRunner) { | ||||||
|  |         await queryRunner.query(`DROP INDEX "public"."IDX_5108098457488634a4768e1d12"`); | ||||||
|  |         await queryRunner.query(`ALTER TABLE "following" DROP COLUMN "notify"`); | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										11
									
								
								packages/backend/migration/1695440131671-short-name.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								packages/backend/migration/1695440131671-short-name.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,11 @@ | |||||||
|  | export class ShortName1695440131671 { | ||||||
|  |     name = 'ShortName1695440131671' | ||||||
|  |  | ||||||
|  |     async up(queryRunner) { | ||||||
|  |         await queryRunner.query(`ALTER TABLE "meta" ADD "shortName" character varying(64)`); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     async down(queryRunner) { | ||||||
|  |         await queryRunner.query(`ALTER TABLE "meta" DROP COLUMN "shortName"`); | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,21 @@ | |||||||
|  | export class MutingNotificationTypes1695605508898 { | ||||||
|  |     name = 'MutingNotificationTypes1695605508898' | ||||||
|  |  | ||||||
|  |     async up(queryRunner) { | ||||||
|  |         await queryRunner.query(`ALTER TYPE "public"."user_profile_mutingnotificationtypes_enum" RENAME TO "user_profile_mutingnotificationtypes_enum_old"`); | ||||||
|  |         await queryRunner.query(`CREATE TYPE "public"."user_profile_mutingnotificationtypes_enum" AS ENUM('note', 'follow', 'mention', 'reply', 'renote', 'quote', 'reaction', 'pollEnded', 'receiveFollowRequest', 'followRequestAccepted', 'achievementEarned', 'app', 'test', 'pollVote', 'groupInvited')`); | ||||||
|  |         await queryRunner.query(`ALTER TABLE "user_profile" ALTER COLUMN "mutingNotificationTypes" DROP DEFAULT`); | ||||||
|  |         await queryRunner.query(`ALTER TABLE "user_profile" ALTER COLUMN "mutingNotificationTypes" TYPE "public"."user_profile_mutingnotificationtypes_enum"[] USING "mutingNotificationTypes"::"text"::"public"."user_profile_mutingnotificationtypes_enum"[]`); | ||||||
|  |         await queryRunner.query(`ALTER TABLE "user_profile" ALTER COLUMN "mutingNotificationTypes" SET DEFAULT '{}'`); | ||||||
|  |         await queryRunner.query(`DROP TYPE "public"."user_profile_mutingnotificationtypes_enum_old"`); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     async down(queryRunner) { | ||||||
|  |         await queryRunner.query(`CREATE TYPE "public"."user_profile_mutingnotificationtypes_enum_old" AS ENUM('follow', 'mention', 'reply', 'renote', 'quote', 'reaction', 'pollVote', 'pollEnded', 'receiveFollowRequest', 'followRequestAccepted', 'groupInvited', 'achievementEarned', 'app')`); | ||||||
|  |         await queryRunner.query(`ALTER TABLE "user_profile" ALTER COLUMN "mutingNotificationTypes" DROP DEFAULT`); | ||||||
|  |         await queryRunner.query(`ALTER TABLE "user_profile" ALTER COLUMN "mutingNotificationTypes" TYPE "public"."user_profile_mutingnotificationtypes_enum_old"[] USING "mutingNotificationTypes"::"text"::"public"."user_profile_mutingnotificationtypes_enum_old"[]`); | ||||||
|  |         await queryRunner.query(`ALTER TABLE "user_profile" ALTER COLUMN "mutingNotificationTypes" SET DEFAULT '{}'`); | ||||||
|  |         await queryRunner.query(`DROP TYPE "public"."user_profile_mutingnotificationtypes_enum"`); | ||||||
|  |         await queryRunner.query(`ALTER TYPE "public"."user_profile_mutingnotificationtypes_enum_old" RENAME TO "user_profile_mutingnotificationtypes_enum"`); | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										11
									
								
								packages/backend/migration/1695901659683-note-updated-at.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								packages/backend/migration/1695901659683-note-updated-at.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,11 @@ | |||||||
|  | export class NoteUpdatedAt1695901659683 { | ||||||
|  |     name = 'NoteUpdatedAt1695901659683' | ||||||
|  |  | ||||||
|  |     async up(queryRunner) { | ||||||
|  |         await queryRunner.query(`ALTER TABLE "note" ADD "updatedAt" TIMESTAMP WITH TIME ZONE`); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     async down(queryRunner) { | ||||||
|  |         await queryRunner.query(`ALTER TABLE "note" DROP COLUMN "updatedAt"`); | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,18 @@ | |||||||
|  | /* | ||||||
|  |  * SPDX-FileCopyrightText: syuilo and other misskey contributors | ||||||
|  |  * SPDX-License-Identifier: AGPL-3.0-only | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | export class NotificationRecieveConfig1695944637565 { | ||||||
|  |     name = 'NotificationRecieveConfig1695944637565' | ||||||
|  |  | ||||||
|  |     async up(queryRunner) { | ||||||
|  |         await queryRunner.query(`ALTER TABLE "user_profile" DROP COLUMN "mutingNotificationTypes"`); | ||||||
|  |         await queryRunner.query(`ALTER TABLE "user_profile" ADD "notificationRecieveConfig" jsonb NOT NULL DEFAULT '{}'`); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     async down(queryRunner) { | ||||||
|  |         await queryRunner.query(`ALTER TABLE "user_profile" DROP COLUMN "notificationRecieveConfig"`); | ||||||
|  |         await queryRunner.query(`ALTER TABLE "user_profile" ADD "mutingNotificationTypes" "public"."user_profile_notificationrecieveconfig_enum" array NOT NULL DEFAULT '{}'`); | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										20
									
								
								packages/backend/migration/1696222183852-withReplies.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								packages/backend/migration/1696222183852-withReplies.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,20 @@ | |||||||
|  | /* | ||||||
|  |  * SPDX-FileCopyrightText: syuilo and other misskey contributors | ||||||
|  |  * SPDX-License-Identifier: AGPL-3.0-only | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | export class WithReplies1696222183852 { | ||||||
|  |     name = 'WithReplies1696222183852' | ||||||
|  |  | ||||||
|  |     async up(queryRunner) { | ||||||
|  |         await queryRunner.query(`ALTER TABLE "following" ADD "withReplies" boolean NOT NULL DEFAULT false`); | ||||||
|  |         await queryRunner.query(`ALTER TABLE "user_list_joining" ADD "withReplies" boolean NOT NULL DEFAULT false`); | ||||||
|  |         await queryRunner.query(`CREATE INDEX "IDX_d74d8ab5efa7e3bb82825c0fa2" ON "following" ("followeeId", "followerHost") `); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     async down(queryRunner) { | ||||||
|  |         await queryRunner.query(`DROP INDEX "public"."IDX_d74d8ab5efa7e3bb82825c0fa2"`); | ||||||
|  |         await queryRunner.query(`ALTER TABLE "user_list_joining" DROP COLUMN "withReplies"`); | ||||||
|  |         await queryRunner.query(`ALTER TABLE "following" DROP COLUMN "withReplies"`); | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,11 @@ | |||||||
|  | export class UserListMembership1696323464251 { | ||||||
|  |     name = 'UserListMembership1696323464251' | ||||||
|  |  | ||||||
|  |     async up(queryRunner) { | ||||||
|  |         await queryRunner.query(`ALTER TABLE "user_list_joining" RENAME TO "user_list_membership"`); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     async down(queryRunner) { | ||||||
|  | 			await queryRunner.query(`ALTER TABLE "user_list_membership" RENAME TO "user_list_joining"`); | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										17
									
								
								packages/backend/migration/1696331570827-hibernation.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								packages/backend/migration/1696331570827-hibernation.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,17 @@ | |||||||
|  | export class Hibernation1696331570827 { | ||||||
|  |     name = 'Hibernation1696331570827' | ||||||
|  |  | ||||||
|  |     async up(queryRunner) { | ||||||
|  | 				await queryRunner.query(`DROP INDEX "public"."IDX_d74d8ab5efa7e3bb82825c0fa2"`); | ||||||
|  |         await queryRunner.query(`ALTER TABLE "user" ADD "isHibernated" boolean NOT NULL DEFAULT false`); | ||||||
|  |         await queryRunner.query(`ALTER TABLE "following" ADD "isFollowerHibernated" boolean NOT NULL DEFAULT false`); | ||||||
|  |         await queryRunner.query(`CREATE INDEX "IDX_ce62b50d882d4e9dee10ad0d2f" ON "following" ("followeeId", "followerHost", "isFollowerHibernated") `); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     async down(queryRunner) { | ||||||
|  |         await queryRunner.query(`DROP INDEX "public"."IDX_ce62b50d882d4e9dee10ad0d2f"`); | ||||||
|  |         await queryRunner.query(`ALTER TABLE "following" DROP COLUMN "isFollowerHibernated"`); | ||||||
|  |         await queryRunner.query(`ALTER TABLE "user" DROP COLUMN "isHibernated"`); | ||||||
|  |         await queryRunner.query(`CREATE INDEX "IDX_d74d8ab5efa7e3bb82825c0fa2" ON "following" ("followeeId", "followerHost") `); | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -39,7 +39,7 @@ | |||||||
| 		"@swc/core-win32-x64-msvc": "1.3.56", | 		"@swc/core-win32-x64-msvc": "1.3.56", | ||||||
| 		"@tensorflow/tfjs": "4.4.0", | 		"@tensorflow/tfjs": "4.4.0", | ||||||
| 		"@tensorflow/tfjs-node": "4.4.0", | 		"@tensorflow/tfjs-node": "4.4.0", | ||||||
| 		"bufferutil": "^4.0.7", | 		"bufferutil": "4.0.7", | ||||||
| 		"slacc-android-arm-eabi": "0.0.10", | 		"slacc-android-arm-eabi": "0.0.10", | ||||||
| 		"slacc-android-arm64": "0.0.10", | 		"slacc-android-arm64": "0.0.10", | ||||||
| 		"slacc-darwin-arm64": "0.0.10", | 		"slacc-darwin-arm64": "0.0.10", | ||||||
| @@ -53,31 +53,32 @@ | |||||||
| 		"slacc-linux-x64-musl": "0.0.10", | 		"slacc-linux-x64-musl": "0.0.10", | ||||||
| 		"slacc-win32-arm64-msvc": "0.0.10", | 		"slacc-win32-arm64-msvc": "0.0.10", | ||||||
| 		"slacc-win32-x64-msvc": "0.0.10", | 		"slacc-win32-x64-msvc": "0.0.10", | ||||||
| 		"utf-8-validate": "^6.0.3" | 		"utf-8-validate": "6.0.3" | ||||||
| 	}, | 	}, | ||||||
| 	"dependencies": { | 	"dependencies": { | ||||||
| 		"@aws-sdk/client-s3": "3.400.0", | 		"@aws-sdk/client-s3": "3.412.0", | ||||||
| 		"@aws-sdk/lib-storage": "3.400.0", | 		"@aws-sdk/lib-storage": "3.412.0", | ||||||
| 		"@aws-sdk/node-http-handler": "3.374.0", | 		"@smithy/node-http-handler": "2.1.5", | ||||||
| 		"@bull-board/api": "5.8.1", | 		"@bull-board/api": "5.8.4", | ||||||
| 		"@bull-board/fastify": "5.8.1", | 		"@bull-board/fastify": "5.8.4", | ||||||
| 		"@bull-board/ui": "5.8.1", | 		"@bull-board/ui": "5.8.4", | ||||||
| 		"@discordapp/twemoji": "14.1.2", | 		"@discordapp/twemoji": "14.1.2", | ||||||
| 		"@fastify/accepts": "4.2.0", | 		"@fastify/accepts": "4.2.0", | ||||||
| 		"@fastify/cookie": "9.0.4", | 		"@fastify/cookie": "9.1.0", | ||||||
| 		"@fastify/cors": "8.3.0", | 		"@fastify/cors": "8.4.0", | ||||||
| 		"@fastify/express": "2.3.0", | 		"@fastify/express": "2.3.0", | ||||||
| 		"@fastify/http-proxy": "9.2.1", | 		"@fastify/http-proxy": "9.2.1", | ||||||
| 		"@fastify/multipart": "7.7.3", | 		"@fastify/multipart": "8.0.0", | ||||||
| 		"@fastify/static": "6.11.0", | 		"@fastify/static": "6.11.2", | ||||||
| 		"@fastify/view": "8.0.0", | 		"@fastify/view": "8.2.0", | ||||||
| 		"@nestjs/common": "10.2.4", | 		"@nestjs/common": "10.2.6", | ||||||
| 		"@nestjs/core": "10.2.4", | 		"@nestjs/core": "10.2.6", | ||||||
| 		"@nestjs/testing": "10.2.4", | 		"@nestjs/testing": "10.2.6", | ||||||
| 		"@peertube/http-signature": "1.7.0", | 		"@peertube/http-signature": "1.7.0", | ||||||
|  | 		"@simplewebauthn/server": "8.2.0", | ||||||
| 		"@sinonjs/fake-timers": "11.1.0", | 		"@sinonjs/fake-timers": "11.1.0", | ||||||
| 		"@swc/cli": "0.1.62", | 		"@swc/cli": "0.1.62", | ||||||
| 		"@swc/core": "1.3.82", | 		"@swc/core": "1.3.90", | ||||||
| 		"accepts": "1.3.8", | 		"accepts": "1.3.8", | ||||||
| 		"ajv": "8.12.0", | 		"ajv": "8.12.0", | ||||||
| 		"archiver": "6.0.1", | 		"archiver": "6.0.1", | ||||||
| @@ -85,7 +86,7 @@ | |||||||
| 		"bcryptjs": "2.4.3", | 		"bcryptjs": "2.4.3", | ||||||
| 		"blurhash": "2.0.5", | 		"blurhash": "2.0.5", | ||||||
| 		"body-parser": "1.20.2", | 		"body-parser": "1.20.2", | ||||||
| 		"bullmq": "4.8.0", | 		"bullmq": "4.11.4", | ||||||
| 		"cacheable-lookup": "7.0.0", | 		"cacheable-lookup": "7.0.0", | ||||||
| 		"cbor": "9.0.1", | 		"cbor": "9.0.1", | ||||||
| 		"chalk": "5.3.0", | 		"chalk": "5.3.0", | ||||||
| @@ -96,7 +97,7 @@ | |||||||
| 		"content-disposition": "0.5.4", | 		"content-disposition": "0.5.4", | ||||||
| 		"date-fns": "2.30.0", | 		"date-fns": "2.30.0", | ||||||
| 		"deep-email-validator": "0.1.21", | 		"deep-email-validator": "0.1.21", | ||||||
| 		"fastify": "4.22.2", | 		"fastify": "4.23.2", | ||||||
| 		"feed": "4.2.2", | 		"feed": "4.2.2", | ||||||
| 		"file-type": "18.5.0", | 		"file-type": "18.5.0", | ||||||
| 		"fluent-ffmpeg": "2.1.2", | 		"fluent-ffmpeg": "2.1.2", | ||||||
| @@ -112,18 +113,18 @@ | |||||||
| 		"js-yaml": "4.1.0", | 		"js-yaml": "4.1.0", | ||||||
| 		"jsdom": "22.1.0", | 		"jsdom": "22.1.0", | ||||||
| 		"json5": "2.2.3", | 		"json5": "2.2.3", | ||||||
| 		"jsonld": "8.2.1", | 		"jsonld": "8.3.1", | ||||||
| 		"jsrsasign": "10.8.6", | 		"jsrsasign": "10.8.6", | ||||||
| 		"meilisearch": "0.34.1", | 		"meilisearch": "0.35.0", | ||||||
| 		"mfm-js": "0.23.3", | 		"mfm-js": "0.23.3", | ||||||
| 		"microformats-parser": "1.4.1", | 		"microformats-parser": "1.5.2", | ||||||
| 		"mime-types": "2.1.35", | 		"mime-types": "2.1.35", | ||||||
| 		"misskey-js": "workspace:*", | 		"misskey-js": "workspace:*", | ||||||
| 		"ms": "3.0.0-canary.1", | 		"ms": "3.0.0-canary.1", | ||||||
| 		"nanoid": "4.0.2", | 		"nanoid": "5.0.1", | ||||||
| 		"nested-property": "4.0.0", | 		"nested-property": "4.0.0", | ||||||
| 		"node-fetch": "3.3.2", | 		"node-fetch": "3.3.2", | ||||||
| 		"nodemailer": "6.9.4", | 		"nodemailer": "6.9.5", | ||||||
| 		"nsfwjs": "2.4.2", | 		"nsfwjs": "2.4.2", | ||||||
| 		"oauth": "0.10.0", | 		"oauth": "0.10.0", | ||||||
| 		"oauth2orize": "1.11.1", | 		"oauth2orize": "1.11.1", | ||||||
| @@ -148,77 +149,78 @@ | |||||||
| 		"rss-parser": "3.13.0", | 		"rss-parser": "3.13.0", | ||||||
| 		"rxjs": "7.8.1", | 		"rxjs": "7.8.1", | ||||||
| 		"sanitize-html": "2.11.0", | 		"sanitize-html": "2.11.0", | ||||||
| 		"sharp": "0.32.5", | 		"sharp": "0.32.6", | ||||||
| 		"sharp-read-bmp": "github:misskey-dev/sharp-read-bmp", | 		"sharp-read-bmp": "github:misskey-dev/sharp-read-bmp", | ||||||
| 		"slacc": "0.0.10", | 		"slacc": "0.0.10", | ||||||
| 		"strict-event-emitter-types": "2.0.0", | 		"strict-event-emitter-types": "2.0.0", | ||||||
| 		"stringz": "2.1.0", | 		"stringz": "2.1.0", | ||||||
| 		"summaly": "github:misskey-dev/summaly", | 		"summaly": "github:misskey-dev/summaly", | ||||||
| 		"systeminformation": "5.21.4", | 		"systeminformation": "5.21.9", | ||||||
| 		"tinycolor2": "1.6.0", | 		"tinycolor2": "1.6.0", | ||||||
| 		"tmp": "0.2.1", | 		"tmp": "0.2.1", | ||||||
| 		"tsc-alias": "1.8.7", | 		"tsc-alias": "1.8.8", | ||||||
| 		"tsconfig-paths": "4.2.0", | 		"tsconfig-paths": "4.2.0", | ||||||
| 		"twemoji-parser": "14.0.0", | 		"twemoji-parser": "14.0.0", | ||||||
| 		"typeorm": "0.3.17", | 		"typeorm": "0.3.17", | ||||||
| 		"typescript": "5.2.2", | 		"typescript": "5.2.2", | ||||||
| 		"ulid": "2.3.0", | 		"ulid": "2.3.0", | ||||||
| 		"vary": "1.1.2", | 		"vary": "1.1.2", | ||||||
| 		"web-push": "3.6.5", | 		"web-push": "3.6.6", | ||||||
| 		"ws": "8.13.0", | 		"ws": "8.14.2", | ||||||
| 		"xev": "3.0.2" | 		"xev": "3.0.2" | ||||||
| 	}, | 	}, | ||||||
| 	"devDependencies": { | 	"devDependencies": { | ||||||
| 		"@jest/globals": "29.6.4", | 		"@jest/globals": "29.7.0", | ||||||
|  | 		"@simplewebauthn/typescript-types": "8.0.0", | ||||||
| 		"@swc/jest": "0.2.29", | 		"@swc/jest": "0.2.29", | ||||||
| 		"@types/accepts": "1.3.5", | 		"@types/accepts": "1.3.5", | ||||||
| 		"@types/archiver": "5.3.2", | 		"@types/archiver": "5.3.3", | ||||||
| 		"@types/bcryptjs": "2.4.3", | 		"@types/bcryptjs": "2.4.4", | ||||||
| 		"@types/body-parser": "1.19.2", | 		"@types/body-parser": "1.19.3", | ||||||
| 		"@types/cbor": "6.0.0", | 		"@types/cbor": "6.0.0", | ||||||
| 		"@types/color-convert": "2.0.1", | 		"@types/color-convert": "2.0.1", | ||||||
| 		"@types/content-disposition": "0.5.6", | 		"@types/content-disposition": "0.5.6", | ||||||
| 		"@types/fluent-ffmpeg": "2.1.21", | 		"@types/fluent-ffmpeg": "2.1.22", | ||||||
| 		"@types/http-link-header": "1.0.3", | 		"@types/http-link-header": "1.0.3", | ||||||
| 		"@types/jest": "29.5.4", | 		"@types/jest": "29.5.5", | ||||||
| 		"@types/js-yaml": "4.0.5", | 		"@types/js-yaml": "4.0.6", | ||||||
| 		"@types/jsdom": "21.1.2", | 		"@types/jsdom": "21.1.3", | ||||||
| 		"@types/jsonld": "1.5.9", | 		"@types/jsonld": "1.5.10", | ||||||
| 		"@types/jsrsasign": "10.5.8", | 		"@types/jsrsasign": "10.5.9", | ||||||
| 		"@types/mime-types": "2.1.1", | 		"@types/mime-types": "2.1.2", | ||||||
| 		"@types/ms": "0.7.31", | 		"@types/ms": "0.7.32", | ||||||
| 		"@types/node": "20.5.9", | 		"@types/node": "20.7.1", | ||||||
| 		"@types/node-fetch": "3.0.3", | 		"@types/node-fetch": "3.0.3", | ||||||
| 		"@types/nodemailer": "6.4.9", | 		"@types/nodemailer": "6.4.11", | ||||||
| 		"@types/oauth": "0.9.2", | 		"@types/oauth": "0.9.2", | ||||||
| 		"@types/oauth2orize": "1.11.1", | 		"@types/oauth2orize": "1.11.1", | ||||||
| 		"@types/oauth2orize-pkce": "0.1.0", | 		"@types/oauth2orize-pkce": "0.1.0", | ||||||
| 		"@types/pg": "8.10.2", | 		"@types/pg": "8.10.3", | ||||||
| 		"@types/pug": "2.0.6", | 		"@types/pug": "2.0.7", | ||||||
| 		"@types/punycode": "2.1.0", | 		"@types/punycode": "2.1.0", | ||||||
| 		"@types/qrcode": "1.5.2", | 		"@types/qrcode": "1.5.2", | ||||||
| 		"@types/random-seed": "0.3.3", | 		"@types/random-seed": "0.3.3", | ||||||
| 		"@types/ratelimiter": "3.4.4", | 		"@types/ratelimiter": "3.4.4", | ||||||
| 		"@types/rename": "1.0.4", | 		"@types/rename": "1.0.5", | ||||||
| 		"@types/sanitize-html": "2.9.0", | 		"@types/sanitize-html": "2.9.1", | ||||||
| 		"@types/semver": "7.5.1", | 		"@types/semver": "7.5.3", | ||||||
| 		"@types/sharp": "0.32.0", | 		"@types/sharp": "0.32.0", | ||||||
| 		"@types/simple-oauth2": "5.0.4", | 		"@types/simple-oauth2": "5.0.5", | ||||||
| 		"@types/sinonjs__fake-timers": "8.1.2", | 		"@types/sinonjs__fake-timers": "8.1.3", | ||||||
| 		"@types/tinycolor2": "1.4.3", | 		"@types/tinycolor2": "1.4.4", | ||||||
| 		"@types/tmp": "0.2.3", | 		"@types/tmp": "0.2.4", | ||||||
| 		"@types/vary": "1.1.0", | 		"@types/vary": "1.1.1", | ||||||
| 		"@types/web-push": "3.6.0", | 		"@types/web-push": "3.6.1", | ||||||
| 		"@types/ws": "8.5.5", | 		"@types/ws": "8.5.6", | ||||||
| 		"@typescript-eslint/eslint-plugin": "6.6.0", | 		"@typescript-eslint/eslint-plugin": "6.7.3", | ||||||
| 		"@typescript-eslint/parser": "6.6.0", | 		"@typescript-eslint/parser": "6.7.3", | ||||||
| 		"aws-sdk-client-mock": "3.0.0", | 		"aws-sdk-client-mock": "3.0.0", | ||||||
| 		"cross-env": "7.0.3", | 		"cross-env": "7.0.3", | ||||||
| 		"eslint": "8.48.0", | 		"eslint": "8.50.0", | ||||||
| 		"eslint-plugin-import": "2.28.1", | 		"eslint-plugin-import": "2.28.1", | ||||||
| 		"execa": "8.0.1", | 		"execa": "8.0.1", | ||||||
| 		"jest": "29.6.4", | 		"jest": "29.7.0", | ||||||
| 		"jest-mock": "29.6.3", | 		"jest-mock": "29.7.0", | ||||||
| 		"simple-oauth2": "5.0.0" | 		"simple-oauth2": "5.0.0" | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|   | |||||||
| @@ -70,11 +70,19 @@ const $redisForSub: Provider = { | |||||||
| 	inject: [DI.config], | 	inject: [DI.config], | ||||||
| }; | }; | ||||||
|  |  | ||||||
|  | const $redisForTimelines: Provider = { | ||||||
|  | 	provide: DI.redisForTimelines, | ||||||
|  | 	useFactory: (config: Config) => { | ||||||
|  | 		return new Redis.Redis(config.redisForTimelines); | ||||||
|  | 	}, | ||||||
|  | 	inject: [DI.config], | ||||||
|  | }; | ||||||
|  |  | ||||||
| @Global() | @Global() | ||||||
| @Module({ | @Module({ | ||||||
| 	imports: [RepositoryModule], | 	imports: [RepositoryModule], | ||||||
| 	providers: [$config, $db, $meilisearch, $redis, $redisForPub, $redisForSub], | 	providers: [$config, $db, $meilisearch, $redis, $redisForPub, $redisForSub, $redisForTimelines], | ||||||
| 	exports: [$config, $db, $meilisearch, $redis, $redisForPub, $redisForSub, RepositoryModule], | 	exports: [$config, $db, $meilisearch, $redis, $redisForPub, $redisForSub, $redisForTimelines, RepositoryModule], | ||||||
| }) | }) | ||||||
| export class GlobalModule implements OnApplicationShutdown { | export class GlobalModule implements OnApplicationShutdown { | ||||||
| 	constructor( | 	constructor( | ||||||
| @@ -82,6 +90,7 @@ export class GlobalModule implements OnApplicationShutdown { | |||||||
| 		@Inject(DI.redis) private redisClient: Redis.Redis, | 		@Inject(DI.redis) private redisClient: Redis.Redis, | ||||||
| 		@Inject(DI.redisForPub) private redisForPub: Redis.Redis, | 		@Inject(DI.redisForPub) private redisForPub: Redis.Redis, | ||||||
| 		@Inject(DI.redisForSub) private redisForSub: Redis.Redis, | 		@Inject(DI.redisForSub) private redisForSub: Redis.Redis, | ||||||
|  | 		@Inject(DI.redisForTimelines) private redisForTimelines: Redis.Redis, | ||||||
| 	) {} | 	) {} | ||||||
|  |  | ||||||
| 	public async dispose(): Promise<void> { | 	public async dispose(): Promise<void> { | ||||||
| @@ -98,6 +107,7 @@ export class GlobalModule implements OnApplicationShutdown { | |||||||
| 			this.redisClient.disconnect(), | 			this.redisClient.disconnect(), | ||||||
| 			this.redisForPub.disconnect(), | 			this.redisForPub.disconnect(), | ||||||
| 			this.redisForSub.disconnect(), | 			this.redisForSub.disconnect(), | ||||||
|  | 			this.redisForTimelines.disconnect(), | ||||||
| 		]); | 		]); | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|   | |||||||
| @@ -8,7 +8,6 @@ import { ChartManagementService } from '@/core/chart/ChartManagementService.js'; | |||||||
| import { QueueProcessorService } from '@/queue/QueueProcessorService.js'; | import { QueueProcessorService } from '@/queue/QueueProcessorService.js'; | ||||||
| import { NestLogger } from '@/NestLogger.js'; | import { NestLogger } from '@/NestLogger.js'; | ||||||
| import { QueueProcessorModule } from '@/queue/QueueProcessorModule.js'; | import { QueueProcessorModule } from '@/queue/QueueProcessorModule.js'; | ||||||
| import { JanitorService } from '@/daemons/JanitorService.js'; |  | ||||||
| import { QueueStatsService } from '@/daemons/QueueStatsService.js'; | import { QueueStatsService } from '@/daemons/QueueStatsService.js'; | ||||||
| import { ServerStatsService } from '@/daemons/ServerStatsService.js'; | import { ServerStatsService } from '@/daemons/ServerStatsService.js'; | ||||||
| import { ServerService } from '@/server/ServerService.js'; | import { ServerService } from '@/server/ServerService.js'; | ||||||
| @@ -25,7 +24,6 @@ export async function server() { | |||||||
|  |  | ||||||
| 	if (process.env.NODE_ENV !== 'test') { | 	if (process.env.NODE_ENV !== 'test') { | ||||||
| 		app.get(ChartManagementService).start(); | 		app.get(ChartManagementService).start(); | ||||||
| 		app.get(JanitorService).start(); |  | ||||||
| 		app.get(QueueStatsService).start(); | 		app.get(QueueStatsService).start(); | ||||||
| 		app.get(ServerStatsService).start(); | 		app.get(ServerStatsService).start(); | ||||||
| 	} | 	} | ||||||
|   | |||||||
| @@ -63,26 +63,40 @@ export async function masterMain() { | |||||||
| 		showNodejsVersion(); | 		showNodejsVersion(); | ||||||
| 		config = loadConfigBoot(); | 		config = loadConfigBoot(); | ||||||
| 		//await connectDb(); | 		//await connectDb(); | ||||||
|  | 		if (config.pidFile) fs.writeFileSync(config.pidFile, process.pid.toString()); | ||||||
| 	} catch (e) { | 	} catch (e) { | ||||||
| 		bootLogger.error('Fatal error occurred during initialization', null, true); | 		bootLogger.error('Fatal error occurred during initialization', null, true); | ||||||
| 		process.exit(1); | 		process.exit(1); | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if (envOption.onlyServer) { |  | ||||||
| 		await server(); |  | ||||||
| 	} else if (envOption.onlyQueue) { |  | ||||||
| 		await jobQueue(); |  | ||||||
| 	} else { |  | ||||||
| 		await server(); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	bootLogger.succ('Misskey initialized'); | 	bootLogger.succ('Misskey initialized'); | ||||||
|  |  | ||||||
| 	if (!envOption.disableClustering) { | 	if (envOption.disableClustering) { | ||||||
|  | 		if (envOption.onlyServer) { | ||||||
|  | 			await server(); | ||||||
|  | 		} else if (envOption.onlyQueue) { | ||||||
|  | 			await jobQueue(); | ||||||
|  | 		} else { | ||||||
|  | 			await server(); | ||||||
|  | 			await jobQueue(); | ||||||
|  | 		} | ||||||
|  | 	} else { | ||||||
|  | 		if (envOption.onlyServer) { | ||||||
|  | 			// nop | ||||||
|  | 		} else if (envOption.onlyQueue) { | ||||||
|  | 			// nop | ||||||
|  | 		} else { | ||||||
|  | 			await server(); | ||||||
|  | 		} | ||||||
|  |  | ||||||
| 		await spawnWorkers(config.clusterLimit); | 		await spawnWorkers(config.clusterLimit); | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	bootLogger.succ(config.socket ? `Now listening on socket ${config.socket} on ${config.url}` : `Now listening on port ${config.port} on ${config.url}`, null, true); | 	if (envOption.onlyQueue) { | ||||||
|  | 		bootLogger.succ('Queue started', null, true); | ||||||
|  | 	} else { | ||||||
|  | 		bootLogger.succ(config.socket ? `Now listening on socket ${config.socket} on ${config.url}` : `Now listening on port ${config.port} on ${config.url}`, null, true); | ||||||
|  | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
| function showEnvironment(): void { | function showEnvironment(): void { | ||||||
|   | |||||||
| @@ -47,6 +47,7 @@ type Source = { | |||||||
| 	redis: RedisOptionsSource; | 	redis: RedisOptionsSource; | ||||||
| 	redisForPubsub?: RedisOptionsSource; | 	redisForPubsub?: RedisOptionsSource; | ||||||
| 	redisForJobQueue?: RedisOptionsSource; | 	redisForJobQueue?: RedisOptionsSource; | ||||||
|  | 	redisForTimelines?: RedisOptionsSource; | ||||||
| 	meilisearch?: { | 	meilisearch?: { | ||||||
| 		host: string; | 		host: string; | ||||||
| 		port: string; | 		port: string; | ||||||
| @@ -89,6 +90,7 @@ type Source = { | |||||||
| 	perChannelMaxNoteCacheCount?: number; | 	perChannelMaxNoteCacheCount?: number; | ||||||
| 	perUserNotificationsMaxCount?: number; | 	perUserNotificationsMaxCount?: number; | ||||||
| 	deactivateAntennaThreshold?: number; | 	deactivateAntennaThreshold?: number; | ||||||
|  | 	pidFile: string; | ||||||
| }; | }; | ||||||
|  |  | ||||||
| export type Config = { | export type Config = { | ||||||
| @@ -160,9 +162,11 @@ export type Config = { | |||||||
| 	redis: RedisOptions & RedisOptionsSource; | 	redis: RedisOptions & RedisOptionsSource; | ||||||
| 	redisForPubsub: RedisOptions & RedisOptionsSource; | 	redisForPubsub: RedisOptions & RedisOptionsSource; | ||||||
| 	redisForJobQueue: RedisOptions & RedisOptionsSource; | 	redisForJobQueue: RedisOptions & RedisOptionsSource; | ||||||
|  | 	redisForTimelines: RedisOptions & RedisOptionsSource; | ||||||
| 	perChannelMaxNoteCacheCount: number; | 	perChannelMaxNoteCacheCount: number; | ||||||
| 	perUserNotificationsMaxCount: number; | 	perUserNotificationsMaxCount: number; | ||||||
| 	deactivateAntennaThreshold: number; | 	deactivateAntennaThreshold: number; | ||||||
|  | 	pidFile: string; | ||||||
| }; | }; | ||||||
|  |  | ||||||
| const _filename = fileURLToPath(import.meta.url); | const _filename = fileURLToPath(import.meta.url); | ||||||
| @@ -225,6 +229,7 @@ export function loadConfig(): Config { | |||||||
| 		redis, | 		redis, | ||||||
| 		redisForPubsub: config.redisForPubsub ? convertRedisOptions(config.redisForPubsub, host) : redis, | 		redisForPubsub: config.redisForPubsub ? convertRedisOptions(config.redisForPubsub, host) : redis, | ||||||
| 		redisForJobQueue: config.redisForJobQueue ? convertRedisOptions(config.redisForJobQueue, host) : redis, | 		redisForJobQueue: config.redisForJobQueue ? convertRedisOptions(config.redisForJobQueue, host) : redis, | ||||||
|  | 		redisForTimelines: config.redisForTimelines ? convertRedisOptions(config.redisForTimelines, host) : redis, | ||||||
| 		id: config.id, | 		id: config.id, | ||||||
| 		proxy: config.proxy, | 		proxy: config.proxy, | ||||||
| 		proxySmtp: config.proxySmtp, | 		proxySmtp: config.proxySmtp, | ||||||
| @@ -255,6 +260,7 @@ export function loadConfig(): Config { | |||||||
| 		perChannelMaxNoteCacheCount: config.perChannelMaxNoteCacheCount ?? 1000, | 		perChannelMaxNoteCacheCount: config.perChannelMaxNoteCacheCount ?? 1000, | ||||||
| 		perUserNotificationsMaxCount: config.perUserNotificationsMaxCount ?? 300, | 		perUserNotificationsMaxCount: config.perUserNotificationsMaxCount ?? 300, | ||||||
| 		deactivateAntennaThreshold: config.deactivateAntennaThreshold ?? (1000 * 60 * 60 * 24 * 7), | 		deactivateAntennaThreshold: config.deactivateAntennaThreshold ?? (1000 * 60 * 60 * 24 * 7), | ||||||
|  | 		pidFile: config.pidFile, | ||||||
| 	}; | 	}; | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -8,8 +8,8 @@ import { IsNull, In, MoreThan, Not } from 'typeorm'; | |||||||
|  |  | ||||||
| import { bindThis } from '@/decorators.js'; | import { bindThis } from '@/decorators.js'; | ||||||
| import { DI } from '@/di-symbols.js'; | import { DI } from '@/di-symbols.js'; | ||||||
| import type { MiLocalUser, MiRemoteUser, MiUser } from '@/models/entities/User.js'; | import type { MiLocalUser, MiRemoteUser, MiUser } from '@/models/User.js'; | ||||||
| import type { BlockingsRepository, FollowingsRepository, InstancesRepository, MutingsRepository, UserListJoiningsRepository, UsersRepository } from '@/models/index.js'; | import type { BlockingsRepository, FollowingsRepository, InstancesRepository, MutingsRepository, UserListMembershipsRepository, UsersRepository } from '@/models/_.js'; | ||||||
| import type { RelationshipJobData, ThinUser } from '@/queue/types.js'; | import type { RelationshipJobData, ThinUser } from '@/queue/types.js'; | ||||||
|  |  | ||||||
| import { IdService } from '@/core/IdService.js'; | import { IdService } from '@/core/IdService.js'; | ||||||
| @@ -42,8 +42,8 @@ export class AccountMoveService { | |||||||
| 		@Inject(DI.mutingsRepository) | 		@Inject(DI.mutingsRepository) | ||||||
| 		private mutingsRepository: MutingsRepository, | 		private mutingsRepository: MutingsRepository, | ||||||
|  |  | ||||||
| 		@Inject(DI.userListJoiningsRepository) | 		@Inject(DI.userListMembershipsRepository) | ||||||
| 		private userListJoiningsRepository: UserListJoiningsRepository, | 		private userListMembershipsRepository: UserListMembershipsRepository, | ||||||
|  |  | ||||||
| 		@Inject(DI.instancesRepository) | 		@Inject(DI.instancesRepository) | ||||||
| 		private instancesRepository: InstancesRepository, | 		private instancesRepository: InstancesRepository, | ||||||
| @@ -215,40 +215,40 @@ export class AccountMoveService { | |||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public async updateLists(src: ThinUser, dst: MiUser): Promise<void> { | 	public async updateLists(src: ThinUser, dst: MiUser): Promise<void> { | ||||||
| 		// Return if there is no list to be updated. | 		// Return if there is no list to be updated. | ||||||
| 		const oldJoinings = await this.userListJoiningsRepository.find({ | 		const oldMemberships = await this.userListMembershipsRepository.find({ | ||||||
| 			where: { | 			where: { | ||||||
| 				userId: src.id, | 				userId: src.id, | ||||||
| 			}, | 			}, | ||||||
| 		}); | 		}); | ||||||
| 		if (oldJoinings.length === 0) return; | 		if (oldMemberships.length === 0) return; | ||||||
|  |  | ||||||
| 		const existingUserListIds = await this.userListJoiningsRepository.find({ | 		const existingUserListIds = await this.userListMembershipsRepository.find({ | ||||||
| 			where: { | 			where: { | ||||||
| 				userId: dst.id, | 				userId: dst.id, | ||||||
| 			}, | 			}, | ||||||
| 		}).then(joinings => joinings.map(joining => joining.userListId)); | 		}).then(memberships => memberships.map(membership => membership.userListId)); | ||||||
|  |  | ||||||
| 		const newJoinings: Map<string, { createdAt: Date; userId: string; userListId: string; }> = new Map(); | 		const newMemberships: Map<string, { createdAt: Date; userId: string; userListId: string; }> = new Map(); | ||||||
|  |  | ||||||
| 		// 重複しないようにIDを生成 | 		// 重複しないようにIDを生成 | ||||||
| 		const genId = (): string => { | 		const genId = (): string => { | ||||||
| 			let id: string; | 			let id: string; | ||||||
| 			do { | 			do { | ||||||
| 				id = this.idService.genId(); | 				id = this.idService.genId(); | ||||||
| 			} while (newJoinings.has(id)); | 			} while (newMemberships.has(id)); | ||||||
| 			return id; | 			return id; | ||||||
| 		}; | 		}; | ||||||
| 		for (const joining of oldJoinings) { | 		for (const membership of oldMemberships) { | ||||||
| 			if (existingUserListIds.includes(joining.userListId)) continue; // skip if dst exists in this user's list | 			if (existingUserListIds.includes(membership.userListId)) continue; // skip if dst exists in this user's list | ||||||
| 			newJoinings.set(genId(), { | 			newMemberships.set(genId(), { | ||||||
| 				createdAt: new Date(), | 				createdAt: new Date(), | ||||||
| 				userId: dst.id, | 				userId: dst.id, | ||||||
| 				userListId: joining.userListId, | 				userListId: membership.userListId, | ||||||
| 			}); | 			}); | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		const arrayToInsert = Array.from(newJoinings.entries()).map(entry => ({ ...entry[1], id: entry[0] })); | 		const arrayToInsert = Array.from(newMemberships.entries()).map(entry => ({ ...entry[1], id: entry[0] })); | ||||||
| 		await this.userListJoiningsRepository.insert(arrayToInsert); | 		await this.userListMembershipsRepository.insert(arrayToInsert); | ||||||
|  |  | ||||||
| 		// Have the proxy account follow the new account in the same way as UserListService.push | 		// Have the proxy account follow the new account in the same way as UserListService.push | ||||||
| 		if (this.userEntityService.isRemoteUser(dst)) { | 		if (this.userEntityService.isRemoteUser(dst)) { | ||||||
|   | |||||||
| @@ -5,8 +5,8 @@ | |||||||
|  |  | ||||||
| import { Inject, Injectable } from '@nestjs/common'; | import { Inject, Injectable } from '@nestjs/common'; | ||||||
| import { DI } from '@/di-symbols.js'; | import { DI } from '@/di-symbols.js'; | ||||||
| import type { UsersRepository } from '@/models/index.js'; | import type { UsersRepository } from '@/models/_.js'; | ||||||
| import type { MiUser } from '@/models/entities/User.js'; | import type { MiUser } from '@/models/User.js'; | ||||||
| import { ApRendererService } from '@/core/activitypub/ApRendererService.js'; | import { ApRendererService } from '@/core/activitypub/ApRendererService.js'; | ||||||
| import { RelayService } from '@/core/RelayService.js'; | import { RelayService } from '@/core/RelayService.js'; | ||||||
| import { ApDeliverManagerService } from '@/core/activitypub/ApDeliverManagerService.js'; | import { ApDeliverManagerService } from '@/core/activitypub/ApDeliverManagerService.js'; | ||||||
|   | |||||||
| @@ -4,8 +4,8 @@ | |||||||
|  */ |  */ | ||||||
|  |  | ||||||
| import { Inject, Injectable } from '@nestjs/common'; | import { Inject, Injectable } from '@nestjs/common'; | ||||||
| import type { UserProfilesRepository } from '@/models/index.js'; | import type { UserProfilesRepository } from '@/models/_.js'; | ||||||
| import type { MiUser } from '@/models/entities/User.js'; | import type { MiUser } from '@/models/User.js'; | ||||||
| import { DI } from '@/di-symbols.js'; | import { DI } from '@/di-symbols.js'; | ||||||
| import { bindThis } from '@/decorators.js'; | import { bindThis } from '@/decorators.js'; | ||||||
| import { NotificationService } from '@/core/NotificationService.js'; | import { NotificationService } from '@/core/NotificationService.js'; | ||||||
| @@ -85,6 +85,7 @@ export const ACHIEVEMENT_TYPES = [ | |||||||
| 	'setNameToSyuilo', | 	'setNameToSyuilo', | ||||||
| 	'cookieClicked', | 	'cookieClicked', | ||||||
| 	'brainDiver', | 	'brainDiver', | ||||||
|  | 	'smashTestNotificationButton', | ||||||
| ] as const; | ] as const; | ||||||
|  |  | ||||||
| @Injectable() | @Injectable() | ||||||
|   | |||||||
| @@ -6,12 +6,13 @@ | |||||||
| import { Inject, Injectable } from '@nestjs/common'; | import { Inject, Injectable } from '@nestjs/common'; | ||||||
| import { Brackets } from 'typeorm'; | import { Brackets } from 'typeorm'; | ||||||
| import { DI } from '@/di-symbols.js'; | import { DI } from '@/di-symbols.js'; | ||||||
| import type { MiUser } from '@/models/entities/User.js'; | import type { MiUser } from '@/models/User.js'; | ||||||
| import type { AnnouncementReadsRepository, AnnouncementsRepository, MiAnnouncement, MiAnnouncementRead } from '@/models/index.js'; | import type { AnnouncementReadsRepository, AnnouncementsRepository, MiAnnouncement, MiAnnouncementRead, UsersRepository } from '@/models/_.js'; | ||||||
| import { bindThis } from '@/decorators.js'; | import { bindThis } from '@/decorators.js'; | ||||||
| import { Packed } from '@/misc/json-schema.js'; | import { Packed } from '@/misc/json-schema.js'; | ||||||
| import { IdService } from '@/core/IdService.js'; | import { IdService } from '@/core/IdService.js'; | ||||||
| import { GlobalEventService } from '@/core/GlobalEventService.js'; | import { GlobalEventService } from '@/core/GlobalEventService.js'; | ||||||
|  | import { ModerationLogService } from '@/core/ModerationLogService.js'; | ||||||
|  |  | ||||||
| @Injectable() | @Injectable() | ||||||
| export class AnnouncementService { | export class AnnouncementService { | ||||||
| @@ -22,8 +23,12 @@ export class AnnouncementService { | |||||||
| 		@Inject(DI.announcementReadsRepository) | 		@Inject(DI.announcementReadsRepository) | ||||||
| 		private announcementReadsRepository: AnnouncementReadsRepository, | 		private announcementReadsRepository: AnnouncementReadsRepository, | ||||||
|  |  | ||||||
|  | 		@Inject(DI.usersRepository) | ||||||
|  | 		private usersRepository: UsersRepository, | ||||||
|  |  | ||||||
| 		private idService: IdService, | 		private idService: IdService, | ||||||
| 		private globalEventService: GlobalEventService, | 		private globalEventService: GlobalEventService, | ||||||
|  | 		private moderationLogService: ModerationLogService, | ||||||
| 	) { | 	) { | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| @@ -58,7 +63,7 @@ export class AnnouncementService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public async create(values: Partial<MiAnnouncement>): Promise<{ raw: MiAnnouncement; packed: Packed<'Announcement'> }> { | 	public async create(values: Partial<MiAnnouncement>, moderator?: MiUser): Promise<{ raw: MiAnnouncement; packed: Packed<'Announcement'> }> { | ||||||
| 		const announcement = await this.announcementsRepository.insert({ | 		const announcement = await this.announcementsRepository.insert({ | ||||||
| 			id: this.idService.genId(), | 			id: this.idService.genId(), | ||||||
| 			createdAt: new Date(), | 			createdAt: new Date(), | ||||||
| @@ -79,10 +84,28 @@ export class AnnouncementService { | |||||||
| 			this.globalEventService.publishMainStream(values.userId, 'announcementCreated', { | 			this.globalEventService.publishMainStream(values.userId, 'announcementCreated', { | ||||||
| 				announcement: packed, | 				announcement: packed, | ||||||
| 			}); | 			}); | ||||||
|  |  | ||||||
|  | 			if (moderator) { | ||||||
|  | 				const user = await this.usersRepository.findOneByOrFail({ id: values.userId }); | ||||||
|  | 				this.moderationLogService.log(moderator, 'createUserAnnouncement', { | ||||||
|  | 					announcementId: announcement.id, | ||||||
|  | 					announcement: announcement, | ||||||
|  | 					userId: values.userId, | ||||||
|  | 					userUsername: user.username, | ||||||
|  | 					userHost: user.host, | ||||||
|  | 				}); | ||||||
|  | 			} | ||||||
| 		} else { | 		} else { | ||||||
| 			this.globalEventService.publishBroadcastStream('announcementCreated', { | 			this.globalEventService.publishBroadcastStream('announcementCreated', { | ||||||
| 				announcement: packed, | 				announcement: packed, | ||||||
| 			}); | 			}); | ||||||
|  |  | ||||||
|  | 			if (moderator) { | ||||||
|  | 				this.moderationLogService.log(moderator, 'createGlobalAnnouncement', { | ||||||
|  | 					announcementId: announcement.id, | ||||||
|  | 					announcement: announcement, | ||||||
|  | 				}); | ||||||
|  | 			} | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		return { | 		return { | ||||||
| @@ -91,6 +114,63 @@ export class AnnouncementService { | |||||||
| 		}; | 		}; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	@bindThis | ||||||
|  | 	public async update(announcement: MiAnnouncement, values: Partial<MiAnnouncement>, moderator?: MiUser): Promise<void> { | ||||||
|  | 		await this.announcementsRepository.update(announcement.id, { | ||||||
|  | 			updatedAt: new Date(), | ||||||
|  | 			title: values.title, | ||||||
|  | 			text: values.text, | ||||||
|  | 			/* eslint-disable-next-line @typescript-eslint/prefer-nullish-coalescing -- 空の文字列の場合、nullを渡すようにするため */ | ||||||
|  | 			imageUrl: values.imageUrl || null, | ||||||
|  | 			display: values.display, | ||||||
|  | 			icon: values.icon, | ||||||
|  | 			forExistingUsers: values.forExistingUsers, | ||||||
|  | 			needConfirmationToRead: values.needConfirmationToRead, | ||||||
|  | 			isActive: values.isActive, | ||||||
|  | 		}); | ||||||
|  |  | ||||||
|  | 		const after = await this.announcementsRepository.findOneByOrFail({ id: announcement.id }); | ||||||
|  |  | ||||||
|  | 		if (moderator) { | ||||||
|  | 			if (announcement.userId) { | ||||||
|  | 				const user = await this.usersRepository.findOneByOrFail({ id: announcement.userId }); | ||||||
|  | 				this.moderationLogService.log(moderator, 'updateUserAnnouncement', { | ||||||
|  | 					announcementId: announcement.id, | ||||||
|  | 					before: announcement, | ||||||
|  | 					after: after, | ||||||
|  | 					userId: announcement.userId, | ||||||
|  | 					userUsername: user.username, | ||||||
|  | 					userHost: user.host, | ||||||
|  | 				}); | ||||||
|  | 			} else { | ||||||
|  | 				this.moderationLogService.log(moderator, 'updateGlobalAnnouncement', { | ||||||
|  | 					announcementId: announcement.id, | ||||||
|  | 					before: announcement, | ||||||
|  | 					after: after, | ||||||
|  | 				}); | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	@bindThis | ||||||
|  | 	public async delete(announcement: MiAnnouncement, moderator?: MiUser): Promise<void> { | ||||||
|  | 		await this.announcementsRepository.delete(announcement.id); | ||||||
|  |  | ||||||
|  | 		if (moderator) { | ||||||
|  | 			if (announcement.userId) { | ||||||
|  | 				this.moderationLogService.log(moderator, 'deleteUserAnnouncement', { | ||||||
|  | 					announcementId: announcement.id, | ||||||
|  | 					announcement: announcement, | ||||||
|  | 				}); | ||||||
|  | 			} else { | ||||||
|  | 				this.moderationLogService.log(moderator, 'deleteGlobalAnnouncement', { | ||||||
|  | 					announcementId: announcement.id, | ||||||
|  | 					announcement: announcement, | ||||||
|  | 				}); | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public async read(user: MiUser, announcementId: MiAnnouncement['id']): Promise<void> { | 	public async read(user: MiUser, announcementId: MiAnnouncement['id']): Promise<void> { | ||||||
| 		try { | 		try { | ||||||
|   | |||||||
| @@ -5,17 +5,17 @@ | |||||||
|  |  | ||||||
| import { Inject, Injectable } from '@nestjs/common'; | import { Inject, Injectable } from '@nestjs/common'; | ||||||
| import * as Redis from 'ioredis'; | import * as Redis from 'ioredis'; | ||||||
| import type { MiAntenna } from '@/models/entities/Antenna.js'; | import type { MiAntenna } from '@/models/Antenna.js'; | ||||||
| import type { MiNote } from '@/models/entities/Note.js'; | import type { MiNote } from '@/models/Note.js'; | ||||||
| import type { MiUser } from '@/models/entities/User.js'; | import type { MiUser } from '@/models/User.js'; | ||||||
| import { GlobalEventService } from '@/core/GlobalEventService.js'; | import { GlobalEventService } from '@/core/GlobalEventService.js'; | ||||||
| import * as Acct from '@/misc/acct.js'; | import * as Acct from '@/misc/acct.js'; | ||||||
| import type { Packed } from '@/misc/json-schema.js'; | import type { Packed } from '@/misc/json-schema.js'; | ||||||
| import { DI } from '@/di-symbols.js'; | import { DI } from '@/di-symbols.js'; | ||||||
| import type { AntennasRepository, UserListJoiningsRepository } from '@/models/index.js'; | import type { AntennasRepository, UserListMembershipsRepository } from '@/models/_.js'; | ||||||
| import { UtilityService } from '@/core/UtilityService.js'; | import { UtilityService } from '@/core/UtilityService.js'; | ||||||
| import { bindThis } from '@/decorators.js'; | import { bindThis } from '@/decorators.js'; | ||||||
| import { StreamMessages } from '@/server/api/stream/types.js'; | import type { GlobalEvents } from '@/core/GlobalEventService.js'; | ||||||
| import type { OnApplicationShutdown } from '@nestjs/common'; | import type { OnApplicationShutdown } from '@nestjs/common'; | ||||||
|  |  | ||||||
| @Injectable() | @Injectable() | ||||||
| @@ -24,8 +24,8 @@ export class AntennaService implements OnApplicationShutdown { | |||||||
| 	private antennas: MiAntenna[]; | 	private antennas: MiAntenna[]; | ||||||
|  |  | ||||||
| 	constructor( | 	constructor( | ||||||
| 		@Inject(DI.redis) | 		@Inject(DI.redisForTimelines) | ||||||
| 		private redisClient: Redis.Redis, | 		private redisForTimelines: Redis.Redis, | ||||||
|  |  | ||||||
| 		@Inject(DI.redisForSub) | 		@Inject(DI.redisForSub) | ||||||
| 		private redisForSub: Redis.Redis, | 		private redisForSub: Redis.Redis, | ||||||
| @@ -33,8 +33,8 @@ export class AntennaService implements OnApplicationShutdown { | |||||||
| 		@Inject(DI.antennasRepository) | 		@Inject(DI.antennasRepository) | ||||||
| 		private antennasRepository: AntennasRepository, | 		private antennasRepository: AntennasRepository, | ||||||
|  |  | ||||||
| 		@Inject(DI.userListJoiningsRepository) | 		@Inject(DI.userListMembershipsRepository) | ||||||
| 		private userListJoiningsRepository: UserListJoiningsRepository, | 		private userListMembershipsRepository: UserListMembershipsRepository, | ||||||
|  |  | ||||||
| 		private utilityService: UtilityService, | 		private utilityService: UtilityService, | ||||||
| 		private globalEventService: GlobalEventService, | 		private globalEventService: GlobalEventService, | ||||||
| @@ -50,7 +50,7 @@ export class AntennaService implements OnApplicationShutdown { | |||||||
| 		const obj = JSON.parse(data); | 		const obj = JSON.parse(data); | ||||||
|  |  | ||||||
| 		if (obj.channel === 'internal') { | 		if (obj.channel === 'internal') { | ||||||
| 			const { type, body } = obj.message as StreamMessages['internal']['payload']; | 			const { type, body } = obj.message as GlobalEvents['internal']['payload']; | ||||||
| 			switch (type) { | 			switch (type) { | ||||||
| 				case 'antennaCreated': | 				case 'antennaCreated': | ||||||
| 					this.antennas.push({ | 					this.antennas.push({ | ||||||
| @@ -81,7 +81,7 @@ export class AntennaService implements OnApplicationShutdown { | |||||||
| 		const antennasWithMatchResult = await Promise.all(antennas.map(antenna => this.checkHitAntenna(antenna, note, noteUser).then(hit => [antenna, hit] as const))); | 		const antennasWithMatchResult = await Promise.all(antennas.map(antenna => this.checkHitAntenna(antenna, note, noteUser).then(hit => [antenna, hit] as const))); | ||||||
| 		const matchedAntennas = antennasWithMatchResult.filter(([, hit]) => hit).map(([antenna]) => antenna); | 		const matchedAntennas = antennasWithMatchResult.filter(([, hit]) => hit).map(([antenna]) => antenna); | ||||||
|  |  | ||||||
| 		const redisPipeline = this.redisClient.pipeline(); | 		const redisPipeline = this.redisForTimelines.pipeline(); | ||||||
|  |  | ||||||
| 		for (const antenna of matchedAntennas) { | 		for (const antenna of matchedAntennas) { | ||||||
| 			redisPipeline.xadd( | 			redisPipeline.xadd( | ||||||
| @@ -108,7 +108,7 @@ export class AntennaService implements OnApplicationShutdown { | |||||||
| 		if (antenna.src === 'home') { | 		if (antenna.src === 'home') { | ||||||
| 			// TODO | 			// TODO | ||||||
| 		} else if (antenna.src === 'list') { | 		} else if (antenna.src === 'list') { | ||||||
| 			const listUsers = (await this.userListJoiningsRepository.findBy({ | 			const listUsers = (await this.userListMembershipsRepository.findBy({ | ||||||
| 				userListId: antenna.userListId!, | 				userListId: antenna.userListId!, | ||||||
| 			})).map(x => x.userId); | 			})).map(x => x.userId); | ||||||
|  |  | ||||||
| @@ -119,6 +119,12 @@ export class AntennaService implements OnApplicationShutdown { | |||||||
| 				return this.utilityService.getFullApAccount(username, host).toLowerCase(); | 				return this.utilityService.getFullApAccount(username, host).toLowerCase(); | ||||||
| 			}); | 			}); | ||||||
| 			if (!accts.includes(this.utilityService.getFullApAccount(noteUser.username, noteUser.host).toLowerCase())) return false; | 			if (!accts.includes(this.utilityService.getFullApAccount(noteUser.username, noteUser.host).toLowerCase())) return false; | ||||||
|  | 		} else if (antenna.src === 'users_blacklist') { | ||||||
|  | 			const accts = antenna.users.map(x => { | ||||||
|  | 				const { username, host } = Acct.parse(x); | ||||||
|  | 				return this.utilityService.getFullApAccount(username, host).toLowerCase(); | ||||||
|  | 			}); | ||||||
|  | 			if (accts.includes(this.utilityService.getFullApAccount(noteUser.username, noteUser.host).toLowerCase())) return false; | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		const keywords = antenna.keywords | 		const keywords = antenna.keywords | ||||||
|   | |||||||
| @@ -5,13 +5,13 @@ | |||||||
|  |  | ||||||
| import { Inject, Injectable } from '@nestjs/common'; | import { Inject, Injectable } from '@nestjs/common'; | ||||||
| import * as Redis from 'ioredis'; | import * as Redis from 'ioredis'; | ||||||
| import type { BlockingsRepository, ChannelFollowingsRepository, FollowingsRepository, MutingsRepository, RenoteMutingsRepository, MiUserProfile, UserProfilesRepository, UsersRepository } from '@/models/index.js'; | import type { BlockingsRepository, ChannelFollowingsRepository, FollowingsRepository, MutingsRepository, RenoteMutingsRepository, MiUserProfile, UserProfilesRepository, UsersRepository, MiFollowing } from '@/models/_.js'; | ||||||
| import { MemoryKVCache, RedisKVCache } from '@/misc/cache.js'; | import { MemoryKVCache, RedisKVCache } from '@/misc/cache.js'; | ||||||
| import type { MiLocalUser, MiUser } from '@/models/entities/User.js'; | import type { MiLocalUser, MiUser } from '@/models/User.js'; | ||||||
| import { DI } from '@/di-symbols.js'; | import { DI } from '@/di-symbols.js'; | ||||||
| import { UserEntityService } from '@/core/entities/UserEntityService.js'; | import { UserEntityService } from '@/core/entities/UserEntityService.js'; | ||||||
| import { bindThis } from '@/decorators.js'; | import { bindThis } from '@/decorators.js'; | ||||||
| import { StreamMessages } from '@/server/api/stream/types.js'; | import type { GlobalEvents } from '@/core/GlobalEventService.js'; | ||||||
| import type { OnApplicationShutdown } from '@nestjs/common'; | import type { OnApplicationShutdown } from '@nestjs/common'; | ||||||
|  |  | ||||||
| @Injectable() | @Injectable() | ||||||
| @@ -25,7 +25,7 @@ export class CacheService implements OnApplicationShutdown { | |||||||
| 	public userBlockingCache: RedisKVCache<Set<string>>; | 	public userBlockingCache: RedisKVCache<Set<string>>; | ||||||
| 	public userBlockedCache: RedisKVCache<Set<string>>; // NOTE: 「被」Blockキャッシュ | 	public userBlockedCache: RedisKVCache<Set<string>>; // NOTE: 「被」Blockキャッシュ | ||||||
| 	public renoteMutingsCache: RedisKVCache<Set<string>>; | 	public renoteMutingsCache: RedisKVCache<Set<string>>; | ||||||
| 	public userFollowingsCache: RedisKVCache<Set<string>>; | 	public userFollowingsCache: RedisKVCache<Record<string, Pick<MiFollowing, 'withReplies'> | undefined>>; | ||||||
| 	public userFollowingChannelsCache: RedisKVCache<Set<string>>; | 	public userFollowingChannelsCache: RedisKVCache<Set<string>>; | ||||||
|  |  | ||||||
| 	constructor( | 	constructor( | ||||||
| @@ -136,12 +136,18 @@ export class CacheService implements OnApplicationShutdown { | |||||||
| 			fromRedisConverter: (value) => new Set(JSON.parse(value)), | 			fromRedisConverter: (value) => new Set(JSON.parse(value)), | ||||||
| 		}); | 		}); | ||||||
|  |  | ||||||
| 		this.userFollowingsCache = new RedisKVCache<Set<string>>(this.redisClient, 'userFollowings', { | 		this.userFollowingsCache = new RedisKVCache<Record<string, Pick<MiFollowing, 'withReplies'> | undefined>>(this.redisClient, 'userFollowings', { | ||||||
| 			lifetime: 1000 * 60 * 30, // 30m | 			lifetime: 1000 * 60 * 30, // 30m | ||||||
| 			memoryCacheLifetime: 1000 * 60, // 1m | 			memoryCacheLifetime: 1000 * 60, // 1m | ||||||
| 			fetcher: (key) => this.followingsRepository.find({ where: { followerId: key }, select: ['followeeId'] }).then(xs => new Set(xs.map(x => x.followeeId))), | 			fetcher: (key) => this.followingsRepository.find({ where: { followerId: key }, select: ['followeeId', 'withReplies'] }).then(xs => { | ||||||
| 			toRedisConverter: (value) => JSON.stringify(Array.from(value)), | 				const obj: Record<string, Pick<MiFollowing, 'withReplies'> | undefined> = {}; | ||||||
| 			fromRedisConverter: (value) => new Set(JSON.parse(value)), | 				for (const x of xs) { | ||||||
|  | 					obj[x.followeeId] = { withReplies: x.withReplies }; | ||||||
|  | 				} | ||||||
|  | 				return obj; | ||||||
|  | 			}), | ||||||
|  | 			toRedisConverter: (value) => JSON.stringify(value), | ||||||
|  | 			fromRedisConverter: (value) => JSON.parse(value), | ||||||
| 		}); | 		}); | ||||||
|  |  | ||||||
| 		this.userFollowingChannelsCache = new RedisKVCache<Set<string>>(this.redisClient, 'userFollowingChannels', { | 		this.userFollowingChannelsCache = new RedisKVCache<Set<string>>(this.redisClient, 'userFollowingChannels', { | ||||||
| @@ -160,7 +166,7 @@ export class CacheService implements OnApplicationShutdown { | |||||||
| 		const obj = JSON.parse(data); | 		const obj = JSON.parse(data); | ||||||
|  |  | ||||||
| 		if (obj.channel === 'internal') { | 		if (obj.channel === 'internal') { | ||||||
| 			const { type, body } = obj.message as StreamMessages['internal']['payload']; | 			const { type, body } = obj.message as GlobalEvents['internal']['payload']; | ||||||
| 			switch (type) { | 			switch (type) { | ||||||
| 				case 'userChangeSuspendedState': | 				case 'userChangeSuspendedState': | ||||||
| 				case 'remoteUserUpdated': { | 				case 'remoteUserUpdated': { | ||||||
| @@ -188,6 +194,7 @@ export class CacheService implements OnApplicationShutdown { | |||||||
| 					if (follower) follower.followingCount++; | 					if (follower) follower.followingCount++; | ||||||
| 					const followee = this.userByIdCache.get(body.followeeId); | 					const followee = this.userByIdCache.get(body.followeeId); | ||||||
| 					if (followee) followee.followersCount++; | 					if (followee) followee.followersCount++; | ||||||
|  | 					this.userFollowingsCache.delete(body.followerId); | ||||||
| 					break; | 					break; | ||||||
| 				} | 				} | ||||||
| 				default: | 				default: | ||||||
|   | |||||||
							
								
								
									
										159
									
								
								packages/backend/src/core/ClipService.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										159
									
								
								packages/backend/src/core/ClipService.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,159 @@ | |||||||
|  | /* | ||||||
|  |  * SPDX-FileCopyrightText: syuilo and other misskey contributors | ||||||
|  |  * SPDX-License-Identifier: AGPL-3.0-only | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | import { Inject, Injectable } from '@nestjs/common'; | ||||||
|  | import { QueryFailedError } from 'typeorm'; | ||||||
|  | import { DI } from '@/di-symbols.js'; | ||||||
|  | import type { ClipsRepository, MiNote, MiClip, ClipNotesRepository, NotesRepository } from '@/models/_.js'; | ||||||
|  | import { bindThis } from '@/decorators.js'; | ||||||
|  | import { isDuplicateKeyValueError } from '@/misc/is-duplicate-key-value-error.js'; | ||||||
|  | import { RoleService } from '@/core/RoleService.js'; | ||||||
|  | import { IdService } from '@/core/IdService.js'; | ||||||
|  | import type { MiLocalUser } from '@/models/User.js'; | ||||||
|  |  | ||||||
|  | @Injectable() | ||||||
|  | export class ClipService { | ||||||
|  | 	public static NoSuchNoteError = class extends Error {}; | ||||||
|  | 	public static NoSuchClipError = class extends Error {}; | ||||||
|  | 	public static AlreadyAddedError = class extends Error {}; | ||||||
|  | 	public static TooManyClipNotesError = class extends Error {}; | ||||||
|  | 	public static TooManyClipsError = class extends Error {}; | ||||||
|  |  | ||||||
|  | 	constructor( | ||||||
|  | 		@Inject(DI.clipsRepository) | ||||||
|  | 		private clipsRepository: ClipsRepository, | ||||||
|  |  | ||||||
|  | 		@Inject(DI.clipNotesRepository) | ||||||
|  | 		private clipNotesRepository: ClipNotesRepository, | ||||||
|  |  | ||||||
|  | 		@Inject(DI.notesRepository) | ||||||
|  | 		private notesRepository: NotesRepository, | ||||||
|  |  | ||||||
|  | 		private roleService: RoleService, | ||||||
|  | 		private idService: IdService, | ||||||
|  | 	) { | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	@bindThis | ||||||
|  | 	public async create(me: MiLocalUser, name: string, isPublic: boolean, description: string | null): Promise<MiClip> { | ||||||
|  | 		const currentCount = await this.clipsRepository.countBy({ | ||||||
|  | 			userId: me.id, | ||||||
|  | 		}); | ||||||
|  | 		if (currentCount > (await this.roleService.getUserPolicies(me.id)).clipLimit) { | ||||||
|  | 			throw new ClipService.TooManyClipsError(); | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		const clip = await this.clipsRepository.insert({ | ||||||
|  | 			id: this.idService.genId(), | ||||||
|  | 			createdAt: new Date(), | ||||||
|  | 			userId: me.id, | ||||||
|  | 			name: name, | ||||||
|  | 			isPublic: isPublic, | ||||||
|  | 			description: description, | ||||||
|  | 		}).then(x => this.clipsRepository.findOneByOrFail(x.identifiers[0])); | ||||||
|  |  | ||||||
|  | 		return clip; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	@bindThis | ||||||
|  | 	public async update(me: MiLocalUser, clipId: MiClip['id'], name: string | undefined, isPublic: boolean | undefined, description: string | null | undefined): Promise<void> { | ||||||
|  | 		const clip = await this.clipsRepository.findOneBy({ | ||||||
|  | 			id: clipId, | ||||||
|  | 			userId: me.id, | ||||||
|  | 		}); | ||||||
|  |  | ||||||
|  | 		if (clip == null) { | ||||||
|  | 			throw new ClipService.NoSuchClipError(); | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		await this.clipsRepository.update(clip.id, { | ||||||
|  | 			name: name, | ||||||
|  | 			description: description, | ||||||
|  | 			isPublic: isPublic, | ||||||
|  | 		}); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	@bindThis | ||||||
|  | 	public async delete(me: MiLocalUser, clipId: MiClip['id']): Promise<void> { | ||||||
|  | 		const clip = await this.clipsRepository.findOneBy({ | ||||||
|  | 			id: clipId, | ||||||
|  | 			userId: me.id, | ||||||
|  | 		}); | ||||||
|  |  | ||||||
|  | 		if (clip == null) { | ||||||
|  | 			throw new ClipService.NoSuchClipError(); | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		await this.clipsRepository.delete(clip.id); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	@bindThis | ||||||
|  | 	public async addNote(me: MiLocalUser, clipId: MiClip['id'], noteId: MiNote['id']): Promise<void> { | ||||||
|  | 		const clip = await this.clipsRepository.findOneBy({ | ||||||
|  | 			id: clipId, | ||||||
|  | 			userId: me.id, | ||||||
|  | 		}); | ||||||
|  |  | ||||||
|  | 		if (clip == null) { | ||||||
|  | 			throw new ClipService.NoSuchClipError(); | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		const currentCount = await this.clipNotesRepository.countBy({ | ||||||
|  | 			clipId: clip.id, | ||||||
|  | 		}); | ||||||
|  | 		if (currentCount > (await this.roleService.getUserPolicies(me.id)).noteEachClipsLimit) { | ||||||
|  | 			throw new ClipService.TooManyClipNotesError(); | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		try { | ||||||
|  | 			await this.clipNotesRepository.insert({ | ||||||
|  | 				id: this.idService.genId(), | ||||||
|  | 				noteId: noteId, | ||||||
|  | 				clipId: clip.id, | ||||||
|  | 			}); | ||||||
|  | 		} catch (e: unknown) { | ||||||
|  | 			if (e instanceof QueryFailedError) { | ||||||
|  | 				if (isDuplicateKeyValueError(e)) { | ||||||
|  | 					throw new ClipService.AlreadyAddedError(); | ||||||
|  | 				} else if (e.driverError.detail.includes('is not present in table "note".')) { | ||||||
|  | 					throw new ClipService.NoSuchNoteError(); | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			throw e; | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		this.clipsRepository.update(clip.id, { | ||||||
|  | 			lastClippedAt: new Date(), | ||||||
|  | 		}); | ||||||
|  |  | ||||||
|  | 		this.notesRepository.increment({ id: noteId }, 'clippedCount', 1); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	@bindThis | ||||||
|  | 	public async removeNote(me: MiLocalUser, clipId: MiClip['id'], noteId: MiNote['id']): Promise<void> { | ||||||
|  | 		const clip = await this.clipsRepository.findOneBy({ | ||||||
|  | 			id: clipId, | ||||||
|  | 			userId: me.id, | ||||||
|  | 		}); | ||||||
|  |  | ||||||
|  | 		if (clip == null) { | ||||||
|  | 			throw new ClipService.NoSuchClipError(); | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		const note = await this.notesRepository.findOneBy({ id: noteId }); | ||||||
|  |  | ||||||
|  | 		if (note == null) { | ||||||
|  | 			throw new ClipService.NoSuchNoteError(); | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		await this.clipNotesRepository.delete({ | ||||||
|  | 			noteId: noteId, | ||||||
|  | 			clipId: clip.id, | ||||||
|  | 		}); | ||||||
|  |  | ||||||
|  | 		this.notesRepository.decrement({ id: noteId }, 'clippedCount', 1); | ||||||
|  | 	} | ||||||
|  | } | ||||||
| @@ -43,20 +43,23 @@ import { RelayService } from './RelayService.js'; | |||||||
| import { RoleService } from './RoleService.js'; | import { RoleService } from './RoleService.js'; | ||||||
| import { S3Service } from './S3Service.js'; | import { S3Service } from './S3Service.js'; | ||||||
| import { SignupService } from './SignupService.js'; | import { SignupService } from './SignupService.js'; | ||||||
| import { TwoFactorAuthenticationService } from './TwoFactorAuthenticationService.js'; | import { WebAuthnService } from './WebAuthnService.js'; | ||||||
| import { UserBlockingService } from './UserBlockingService.js'; | import { UserBlockingService } from './UserBlockingService.js'; | ||||||
| import { CacheService } from './CacheService.js'; | import { CacheService } from './CacheService.js'; | ||||||
|  | import { UserService } from './UserService.js'; | ||||||
| import { UserFollowingService } from './UserFollowingService.js'; | import { UserFollowingService } from './UserFollowingService.js'; | ||||||
| import { UserKeypairService } from './UserKeypairService.js'; | import { UserKeypairService } from './UserKeypairService.js'; | ||||||
| import { UserListService } from './UserListService.js'; | import { UserListService } from './UserListService.js'; | ||||||
| import { UserMutingService } from './UserMutingService.js'; | import { UserMutingService } from './UserMutingService.js'; | ||||||
| import { UserSuspendService } from './UserSuspendService.js'; | import { UserSuspendService } from './UserSuspendService.js'; | ||||||
|  | import { UserAuthService } from './UserAuthService.js'; | ||||||
| import { VideoProcessingService } from './VideoProcessingService.js'; | import { VideoProcessingService } from './VideoProcessingService.js'; | ||||||
| import { WebhookService } from './WebhookService.js'; | import { WebhookService } from './WebhookService.js'; | ||||||
| import { ProxyAccountService } from './ProxyAccountService.js'; | import { ProxyAccountService } from './ProxyAccountService.js'; | ||||||
| import { UtilityService } from './UtilityService.js'; | import { UtilityService } from './UtilityService.js'; | ||||||
| import { FileInfoService } from './FileInfoService.js'; | import { FileInfoService } from './FileInfoService.js'; | ||||||
| import { SearchService } from './SearchService.js'; | import { SearchService } from './SearchService.js'; | ||||||
|  | import { ClipService } from './ClipService.js'; | ||||||
| import { ChartLoggerService } from './chart/ChartLoggerService.js'; | import { ChartLoggerService } from './chart/ChartLoggerService.js'; | ||||||
| import FederationChart from './chart/charts/federation.js'; | import FederationChart from './chart/charts/federation.js'; | ||||||
| import NotesChart from './chart/charts/notes.js'; | import NotesChart from './chart/charts/notes.js'; | ||||||
| @@ -168,19 +171,22 @@ const $RelayService: Provider = { provide: 'RelayService', useExisting: RelaySer | |||||||
| const $RoleService: Provider = { provide: 'RoleService', useExisting: RoleService }; | const $RoleService: Provider = { provide: 'RoleService', useExisting: RoleService }; | ||||||
| const $S3Service: Provider = { provide: 'S3Service', useExisting: S3Service }; | const $S3Service: Provider = { provide: 'S3Service', useExisting: S3Service }; | ||||||
| const $SignupService: Provider = { provide: 'SignupService', useExisting: SignupService }; | const $SignupService: Provider = { provide: 'SignupService', useExisting: SignupService }; | ||||||
| const $TwoFactorAuthenticationService: Provider = { provide: 'TwoFactorAuthenticationService', useExisting: TwoFactorAuthenticationService }; | const $WebAuthnService: Provider = { provide: 'WebAuthnService', useExisting: WebAuthnService }; | ||||||
| const $UserBlockingService: Provider = { provide: 'UserBlockingService', useExisting: UserBlockingService }; | const $UserBlockingService: Provider = { provide: 'UserBlockingService', useExisting: UserBlockingService }; | ||||||
| const $CacheService: Provider = { provide: 'CacheService', useExisting: CacheService }; | const $CacheService: Provider = { provide: 'CacheService', useExisting: CacheService }; | ||||||
|  | const $UserService: Provider = { provide: 'UserService', useExisting: UserService }; | ||||||
| const $UserFollowingService: Provider = { provide: 'UserFollowingService', useExisting: UserFollowingService }; | const $UserFollowingService: Provider = { provide: 'UserFollowingService', useExisting: UserFollowingService }; | ||||||
| const $UserKeypairService: Provider = { provide: 'UserKeypairService', useExisting: UserKeypairService }; | const $UserKeypairService: Provider = { provide: 'UserKeypairService', useExisting: UserKeypairService }; | ||||||
| const $UserListService: Provider = { provide: 'UserListService', useExisting: UserListService }; | const $UserListService: Provider = { provide: 'UserListService', useExisting: UserListService }; | ||||||
| const $UserMutingService: Provider = { provide: 'UserMutingService', useExisting: UserMutingService }; | const $UserMutingService: Provider = { provide: 'UserMutingService', useExisting: UserMutingService }; | ||||||
| const $UserSuspendService: Provider = { provide: 'UserSuspendService', useExisting: UserSuspendService }; | const $UserSuspendService: Provider = { provide: 'UserSuspendService', useExisting: UserSuspendService }; | ||||||
|  | const $UserAuthService: Provider = { provide: 'UserAuthService', useExisting: UserAuthService }; | ||||||
| const $VideoProcessingService: Provider = { provide: 'VideoProcessingService', useExisting: VideoProcessingService }; | const $VideoProcessingService: Provider = { provide: 'VideoProcessingService', useExisting: VideoProcessingService }; | ||||||
| const $WebhookService: Provider = { provide: 'WebhookService', useExisting: WebhookService }; | const $WebhookService: Provider = { provide: 'WebhookService', useExisting: WebhookService }; | ||||||
| const $UtilityService: Provider = { provide: 'UtilityService', useExisting: UtilityService }; | const $UtilityService: Provider = { provide: 'UtilityService', useExisting: UtilityService }; | ||||||
| const $FileInfoService: Provider = { provide: 'FileInfoService', useExisting: FileInfoService }; | const $FileInfoService: Provider = { provide: 'FileInfoService', useExisting: FileInfoService }; | ||||||
| const $SearchService: Provider = { provide: 'SearchService', useExisting: SearchService }; | const $SearchService: Provider = { provide: 'SearchService', useExisting: SearchService }; | ||||||
|  | const $ClipService: Provider = { provide: 'ClipService', useExisting: ClipService }; | ||||||
|  |  | ||||||
| const $ChartLoggerService: Provider = { provide: 'ChartLoggerService', useExisting: ChartLoggerService }; | const $ChartLoggerService: Provider = { provide: 'ChartLoggerService', useExisting: ChartLoggerService }; | ||||||
| const $FederationChart: Provider = { provide: 'FederationChart', useExisting: FederationChart }; | const $FederationChart: Provider = { provide: 'FederationChart', useExisting: FederationChart }; | ||||||
| @@ -296,19 +302,22 @@ const $ApQuestionService: Provider = { provide: 'ApQuestionService', useExisting | |||||||
| 		RoleService, | 		RoleService, | ||||||
| 		S3Service, | 		S3Service, | ||||||
| 		SignupService, | 		SignupService, | ||||||
| 		TwoFactorAuthenticationService, | 		WebAuthnService, | ||||||
| 		UserBlockingService, | 		UserBlockingService, | ||||||
| 		CacheService, | 		CacheService, | ||||||
|  | 		UserService, | ||||||
| 		UserFollowingService, | 		UserFollowingService, | ||||||
| 		UserKeypairService, | 		UserKeypairService, | ||||||
| 		UserListService, | 		UserListService, | ||||||
| 		UserMutingService, | 		UserMutingService, | ||||||
| 		UserSuspendService, | 		UserSuspendService, | ||||||
|  | 		UserAuthService, | ||||||
| 		VideoProcessingService, | 		VideoProcessingService, | ||||||
| 		WebhookService, | 		WebhookService, | ||||||
| 		UtilityService, | 		UtilityService, | ||||||
| 		FileInfoService, | 		FileInfoService, | ||||||
| 		SearchService, | 		SearchService, | ||||||
|  | 		ClipService, | ||||||
| 		ChartLoggerService, | 		ChartLoggerService, | ||||||
| 		FederationChart, | 		FederationChart, | ||||||
| 		NotesChart, | 		NotesChart, | ||||||
| @@ -417,19 +426,22 @@ const $ApQuestionService: Provider = { provide: 'ApQuestionService', useExisting | |||||||
| 		$RoleService, | 		$RoleService, | ||||||
| 		$S3Service, | 		$S3Service, | ||||||
| 		$SignupService, | 		$SignupService, | ||||||
| 		$TwoFactorAuthenticationService, | 		$WebAuthnService, | ||||||
| 		$UserBlockingService, | 		$UserBlockingService, | ||||||
| 		$CacheService, | 		$CacheService, | ||||||
|  | 		$UserService, | ||||||
| 		$UserFollowingService, | 		$UserFollowingService, | ||||||
| 		$UserKeypairService, | 		$UserKeypairService, | ||||||
| 		$UserListService, | 		$UserListService, | ||||||
| 		$UserMutingService, | 		$UserMutingService, | ||||||
| 		$UserSuspendService, | 		$UserSuspendService, | ||||||
|  | 		$UserAuthService, | ||||||
| 		$VideoProcessingService, | 		$VideoProcessingService, | ||||||
| 		$WebhookService, | 		$WebhookService, | ||||||
| 		$UtilityService, | 		$UtilityService, | ||||||
| 		$FileInfoService, | 		$FileInfoService, | ||||||
| 		$SearchService, | 		$SearchService, | ||||||
|  | 		$ClipService, | ||||||
| 		$ChartLoggerService, | 		$ChartLoggerService, | ||||||
| 		$FederationChart, | 		$FederationChart, | ||||||
| 		$NotesChart, | 		$NotesChart, | ||||||
| @@ -539,19 +551,22 @@ const $ApQuestionService: Provider = { provide: 'ApQuestionService', useExisting | |||||||
| 		RoleService, | 		RoleService, | ||||||
| 		S3Service, | 		S3Service, | ||||||
| 		SignupService, | 		SignupService, | ||||||
| 		TwoFactorAuthenticationService, | 		WebAuthnService, | ||||||
| 		UserBlockingService, | 		UserBlockingService, | ||||||
| 		CacheService, | 		CacheService, | ||||||
|  | 		UserService, | ||||||
| 		UserFollowingService, | 		UserFollowingService, | ||||||
| 		UserKeypairService, | 		UserKeypairService, | ||||||
| 		UserListService, | 		UserListService, | ||||||
| 		UserMutingService, | 		UserMutingService, | ||||||
| 		UserSuspendService, | 		UserSuspendService, | ||||||
|  | 		UserAuthService, | ||||||
| 		VideoProcessingService, | 		VideoProcessingService, | ||||||
| 		WebhookService, | 		WebhookService, | ||||||
| 		UtilityService, | 		UtilityService, | ||||||
| 		FileInfoService, | 		FileInfoService, | ||||||
| 		SearchService, | 		SearchService, | ||||||
|  | 		ClipService, | ||||||
| 		FederationChart, | 		FederationChart, | ||||||
| 		NotesChart, | 		NotesChart, | ||||||
| 		UsersChart, | 		UsersChart, | ||||||
| @@ -659,19 +674,22 @@ const $ApQuestionService: Provider = { provide: 'ApQuestionService', useExisting | |||||||
| 		$RoleService, | 		$RoleService, | ||||||
| 		$S3Service, | 		$S3Service, | ||||||
| 		$SignupService, | 		$SignupService, | ||||||
| 		$TwoFactorAuthenticationService, | 		$WebAuthnService, | ||||||
| 		$UserBlockingService, | 		$UserBlockingService, | ||||||
| 		$CacheService, | 		$CacheService, | ||||||
|  | 		$UserService, | ||||||
| 		$UserFollowingService, | 		$UserFollowingService, | ||||||
| 		$UserKeypairService, | 		$UserKeypairService, | ||||||
| 		$UserListService, | 		$UserListService, | ||||||
| 		$UserMutingService, | 		$UserMutingService, | ||||||
| 		$UserSuspendService, | 		$UserSuspendService, | ||||||
|  | 		$UserAuthService, | ||||||
| 		$VideoProcessingService, | 		$VideoProcessingService, | ||||||
| 		$WebhookService, | 		$WebhookService, | ||||||
| 		$UtilityService, | 		$UtilityService, | ||||||
| 		$FileInfoService, | 		$FileInfoService, | ||||||
| 		$SearchService, | 		$SearchService, | ||||||
|  | 		$ClipService, | ||||||
| 		$FederationChart, | 		$FederationChart, | ||||||
| 		$NotesChart, | 		$NotesChart, | ||||||
| 		$UsersChart, | 		$UsersChart, | ||||||
|   | |||||||
| @@ -8,11 +8,11 @@ import { Inject, Injectable } from '@nestjs/common'; | |||||||
| import bcrypt from 'bcryptjs'; | import bcrypt from 'bcryptjs'; | ||||||
| import { IsNull, DataSource } from 'typeorm'; | import { IsNull, DataSource } from 'typeorm'; | ||||||
| import { genRsaKeyPair } from '@/misc/gen-key-pair.js'; | import { genRsaKeyPair } from '@/misc/gen-key-pair.js'; | ||||||
| import { MiUser } from '@/models/entities/User.js'; | import { MiUser } from '@/models/User.js'; | ||||||
| import { MiUserProfile } from '@/models/entities/UserProfile.js'; | import { MiUserProfile } from '@/models/UserProfile.js'; | ||||||
| import { IdService } from '@/core/IdService.js'; | import { IdService } from '@/core/IdService.js'; | ||||||
| import { MiUserKeypair } from '@/models/entities/UserKeypair.js'; | import { MiUserKeypair } from '@/models/UserKeypair.js'; | ||||||
| import { MiUsedUsername } from '@/models/entities/UsedUsername.js'; | import { MiUsedUsername } from '@/models/UsedUsername.js'; | ||||||
| import { DI } from '@/di-symbols.js'; | import { DI } from '@/di-symbols.js'; | ||||||
| import generateNativeUserToken from '@/misc/generate-native-user-token.js'; | import generateNativeUserToken from '@/misc/generate-native-user-token.js'; | ||||||
| import { bindThis } from '@/decorators.js'; | import { bindThis } from '@/decorators.js'; | ||||||
|   | |||||||
| @@ -10,14 +10,15 @@ import { DI } from '@/di-symbols.js'; | |||||||
| import { IdService } from '@/core/IdService.js'; | import { IdService } from '@/core/IdService.js'; | ||||||
| import { EmojiEntityService } from '@/core/entities/EmojiEntityService.js'; | import { EmojiEntityService } from '@/core/entities/EmojiEntityService.js'; | ||||||
| import { GlobalEventService } from '@/core/GlobalEventService.js'; | import { GlobalEventService } from '@/core/GlobalEventService.js'; | ||||||
| import type { MiDriveFile } from '@/models/entities/DriveFile.js'; | import type { MiDriveFile } from '@/models/DriveFile.js'; | ||||||
| import type { MiEmoji } from '@/models/entities/Emoji.js'; | import type { MiEmoji } from '@/models/Emoji.js'; | ||||||
| import type { EmojisRepository, MiRole } from '@/models/index.js'; | import type { EmojisRepository, MiRole, MiUser } from '@/models/_.js'; | ||||||
| import { bindThis } from '@/decorators.js'; | import { bindThis } from '@/decorators.js'; | ||||||
| import { MemoryKVCache, RedisSingleCache } from '@/misc/cache.js'; | import { MemoryKVCache, RedisSingleCache } from '@/misc/cache.js'; | ||||||
| import { UtilityService } from '@/core/UtilityService.js'; | import { UtilityService } from '@/core/UtilityService.js'; | ||||||
| import { query } from '@/misc/prelude/url.js'; | import { query } from '@/misc/prelude/url.js'; | ||||||
| import type { Serialized } from '@/server/api/stream/types.js'; | import type { Serialized } from '@/types.js'; | ||||||
|  | import { ModerationLogService } from '@/core/ModerationLogService.js'; | ||||||
|  |  | ||||||
| const parseEmojiStrRegexp = /^(\w+)(?:@([\w.-]+))?$/; | const parseEmojiStrRegexp = /^(\w+)(?:@([\w.-]+))?$/; | ||||||
|  |  | ||||||
| @@ -36,6 +37,7 @@ export class CustomEmojiService implements OnApplicationShutdown { | |||||||
| 		private utilityService: UtilityService, | 		private utilityService: UtilityService, | ||||||
| 		private idService: IdService, | 		private idService: IdService, | ||||||
| 		private emojiEntityService: EmojiEntityService, | 		private emojiEntityService: EmojiEntityService, | ||||||
|  | 		private moderationLogService: ModerationLogService, | ||||||
| 		private globalEventService: GlobalEventService, | 		private globalEventService: GlobalEventService, | ||||||
| 	) { | 	) { | ||||||
| 		this.cache = new MemoryKVCache<MiEmoji | null>(1000 * 60 * 60 * 12); | 		this.cache = new MemoryKVCache<MiEmoji | null>(1000 * 60 * 60 * 12); | ||||||
| @@ -66,7 +68,7 @@ export class CustomEmojiService implements OnApplicationShutdown { | |||||||
| 		isSensitive: boolean; | 		isSensitive: boolean; | ||||||
| 		localOnly: boolean; | 		localOnly: boolean; | ||||||
| 		roleIdsThatCanBeUsedThisEmojiAsReaction: MiRole['id'][]; | 		roleIdsThatCanBeUsedThisEmojiAsReaction: MiRole['id'][]; | ||||||
| 	}): Promise<MiEmoji> { | 	}, moderator?: MiUser): Promise<MiEmoji> { | ||||||
| 		const emoji = await this.emojisRepository.insert({ | 		const emoji = await this.emojisRepository.insert({ | ||||||
| 			id: this.idService.genId(), | 			id: this.idService.genId(), | ||||||
| 			updatedAt: new Date(), | 			updatedAt: new Date(), | ||||||
| @@ -89,6 +91,13 @@ export class CustomEmojiService implements OnApplicationShutdown { | |||||||
| 			this.globalEventService.publishBroadcastStream('emojiAdded', { | 			this.globalEventService.publishBroadcastStream('emojiAdded', { | ||||||
| 				emoji: await this.emojiEntityService.packDetailed(emoji.id), | 				emoji: await this.emojiEntityService.packDetailed(emoji.id), | ||||||
| 			}); | 			}); | ||||||
|  |  | ||||||
|  | 			if (moderator) { | ||||||
|  | 				this.moderationLogService.log(moderator, 'addCustomEmoji', { | ||||||
|  | 					emojiId: emoji.id, | ||||||
|  | 					emoji: emoji, | ||||||
|  | 				}); | ||||||
|  | 			} | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		return emoji; | 		return emoji; | ||||||
| @@ -104,7 +113,7 @@ export class CustomEmojiService implements OnApplicationShutdown { | |||||||
| 		isSensitive?: boolean; | 		isSensitive?: boolean; | ||||||
| 		localOnly?: boolean; | 		localOnly?: boolean; | ||||||
| 		roleIdsThatCanBeUsedThisEmojiAsReaction?: MiRole['id'][]; | 		roleIdsThatCanBeUsedThisEmojiAsReaction?: MiRole['id'][]; | ||||||
| 	}): Promise<void> { | 	}, moderator?: MiUser): Promise<void> { | ||||||
| 		const emoji = await this.emojisRepository.findOneByOrFail({ id: id }); | 		const emoji = await this.emojisRepository.findOneByOrFail({ id: id }); | ||||||
| 		const sameNameEmoji = await this.emojisRepository.findOneBy({ name: data.name, host: IsNull() }); | 		const sameNameEmoji = await this.emojisRepository.findOneBy({ name: data.name, host: IsNull() }); | ||||||
| 		if (sameNameEmoji != null && sameNameEmoji.id !== id) throw new Error('name already exists'); | 		if (sameNameEmoji != null && sameNameEmoji.id !== id) throw new Error('name already exists'); | ||||||
| @@ -125,11 +134,11 @@ export class CustomEmojiService implements OnApplicationShutdown { | |||||||
|  |  | ||||||
| 		this.localEmojisCache.refresh(); | 		this.localEmojisCache.refresh(); | ||||||
|  |  | ||||||
| 		const updated = await this.emojiEntityService.packDetailed(emoji.id); | 		const packed = await this.emojiEntityService.packDetailed(emoji.id); | ||||||
|  |  | ||||||
| 		if (emoji.name === data.name) { | 		if (emoji.name === data.name) { | ||||||
| 			this.globalEventService.publishBroadcastStream('emojiUpdated', { | 			this.globalEventService.publishBroadcastStream('emojiUpdated', { | ||||||
| 				emojis: [updated], | 				emojis: [packed], | ||||||
| 			}); | 			}); | ||||||
| 		} else { | 		} else { | ||||||
| 			this.globalEventService.publishBroadcastStream('emojiDeleted', { | 			this.globalEventService.publishBroadcastStream('emojiDeleted', { | ||||||
| @@ -137,7 +146,16 @@ export class CustomEmojiService implements OnApplicationShutdown { | |||||||
| 			}); | 			}); | ||||||
|  |  | ||||||
| 			this.globalEventService.publishBroadcastStream('emojiAdded', { | 			this.globalEventService.publishBroadcastStream('emojiAdded', { | ||||||
| 				emoji: updated, | 				emoji: packed, | ||||||
|  | 			}); | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if (moderator) { | ||||||
|  | 			const updated = await this.emojisRepository.findOneByOrFail({ id: id }); | ||||||
|  | 			this.moderationLogService.log(moderator, 'updateCustomEmoji', { | ||||||
|  | 				emojiId: emoji.id, | ||||||
|  | 				before: emoji, | ||||||
|  | 				after: updated, | ||||||
| 			}); | 			}); | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| @@ -231,7 +249,7 @@ export class CustomEmojiService implements OnApplicationShutdown { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public async delete(id: MiEmoji['id']) { | 	public async delete(id: MiEmoji['id'], moderator?: MiUser) { | ||||||
| 		const emoji = await this.emojisRepository.findOneByOrFail({ id: id }); | 		const emoji = await this.emojisRepository.findOneByOrFail({ id: id }); | ||||||
|  |  | ||||||
| 		await this.emojisRepository.delete(emoji.id); | 		await this.emojisRepository.delete(emoji.id); | ||||||
| @@ -241,16 +259,30 @@ export class CustomEmojiService implements OnApplicationShutdown { | |||||||
| 		this.globalEventService.publishBroadcastStream('emojiDeleted', { | 		this.globalEventService.publishBroadcastStream('emojiDeleted', { | ||||||
| 			emojis: [await this.emojiEntityService.packDetailed(emoji)], | 			emojis: [await this.emojiEntityService.packDetailed(emoji)], | ||||||
| 		}); | 		}); | ||||||
|  |  | ||||||
|  | 		if (moderator) { | ||||||
|  | 			this.moderationLogService.log(moderator, 'deleteCustomEmoji', { | ||||||
|  | 				emojiId: emoji.id, | ||||||
|  | 				emoji: emoji, | ||||||
|  | 			}); | ||||||
|  | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public async deleteBulk(ids: MiEmoji['id'][]) { | 	public async deleteBulk(ids: MiEmoji['id'][], moderator?: MiUser) { | ||||||
| 		const emojis = await this.emojisRepository.findBy({ | 		const emojis = await this.emojisRepository.findBy({ | ||||||
| 			id: In(ids), | 			id: In(ids), | ||||||
| 		}); | 		}); | ||||||
|  |  | ||||||
| 		for (const emoji of emojis) { | 		for (const emoji of emojis) { | ||||||
| 			await this.emojisRepository.delete(emoji.id); | 			await this.emojisRepository.delete(emoji.id); | ||||||
|  |  | ||||||
|  | 			if (moderator) { | ||||||
|  | 				this.moderationLogService.log(moderator, 'deleteCustomEmoji', { | ||||||
|  | 					emojiId: emoji.id, | ||||||
|  | 					emoji: emoji, | ||||||
|  | 				}); | ||||||
|  | 			} | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		this.localEmojisCache.refresh(); | 		this.localEmojisCache.refresh(); | ||||||
|   | |||||||
| @@ -4,7 +4,7 @@ | |||||||
|  */ |  */ | ||||||
|  |  | ||||||
| import { Inject, Injectable } from '@nestjs/common'; | import { Inject, Injectable } from '@nestjs/common'; | ||||||
| import type { UsersRepository } from '@/models/index.js'; | import type { UsersRepository } from '@/models/_.js'; | ||||||
| import { QueueService } from '@/core/QueueService.js'; | import { QueueService } from '@/core/QueueService.js'; | ||||||
| import { UserSuspendService } from '@/core/UserSuspendService.js'; | import { UserSuspendService } from '@/core/UserSuspendService.js'; | ||||||
| import { DI } from '@/di-symbols.js'; | import { DI } from '@/di-symbols.js'; | ||||||
|   | |||||||
| @@ -11,12 +11,12 @@ import { sharpBmp } from 'sharp-read-bmp'; | |||||||
| import { IsNull } from 'typeorm'; | import { IsNull } from 'typeorm'; | ||||||
| import { DeleteObjectCommandInput, PutObjectCommandInput, NoSuchKey } from '@aws-sdk/client-s3'; | import { DeleteObjectCommandInput, PutObjectCommandInput, NoSuchKey } from '@aws-sdk/client-s3'; | ||||||
| import { DI } from '@/di-symbols.js'; | import { DI } from '@/di-symbols.js'; | ||||||
| import type { DriveFilesRepository, UsersRepository, DriveFoldersRepository, UserProfilesRepository } from '@/models/index.js'; | import type { DriveFilesRepository, UsersRepository, DriveFoldersRepository, UserProfilesRepository } from '@/models/_.js'; | ||||||
| import type { Config } from '@/config.js'; | import type { Config } from '@/config.js'; | ||||||
| import Logger from '@/logger.js'; | import Logger from '@/logger.js'; | ||||||
| import type { MiRemoteUser, MiUser } from '@/models/entities/User.js'; | import type { MiRemoteUser, MiUser } from '@/models/User.js'; | ||||||
| import { MetaService } from '@/core/MetaService.js'; | import { MetaService } from '@/core/MetaService.js'; | ||||||
| import { MiDriveFile } from '@/models/entities/DriveFile.js'; | import { MiDriveFile } from '@/models/DriveFile.js'; | ||||||
| import { IdService } from '@/core/IdService.js'; | import { IdService } from '@/core/IdService.js'; | ||||||
| import { isDuplicateKeyValueError } from '@/misc/is-duplicate-key-value-error.js'; | import { isDuplicateKeyValueError } from '@/misc/is-duplicate-key-value-error.js'; | ||||||
| import { FILE_TYPE_BROWSERSAFE } from '@/const.js'; | import { FILE_TYPE_BROWSERSAFE } from '@/const.js'; | ||||||
| @@ -27,7 +27,7 @@ import { VideoProcessingService } from '@/core/VideoProcessingService.js'; | |||||||
| import { ImageProcessingService } from '@/core/ImageProcessingService.js'; | import { ImageProcessingService } from '@/core/ImageProcessingService.js'; | ||||||
| import type { IImage } from '@/core/ImageProcessingService.js'; | import type { IImage } from '@/core/ImageProcessingService.js'; | ||||||
| import { QueueService } from '@/core/QueueService.js'; | import { QueueService } from '@/core/QueueService.js'; | ||||||
| import type { MiDriveFolder } from '@/models/entities/DriveFolder.js'; | import type { MiDriveFolder } from '@/models/DriveFolder.js'; | ||||||
| import { createTemp } from '@/misc/create-temp.js'; | import { createTemp } from '@/misc/create-temp.js'; | ||||||
| import DriveChart from '@/core/chart/charts/drive.js'; | import DriveChart from '@/core/chart/charts/drive.js'; | ||||||
| import PerUserDriveChart from '@/core/chart/charts/per-user-drive.js'; | import PerUserDriveChart from '@/core/chart/charts/per-user-drive.js'; | ||||||
| @@ -42,6 +42,7 @@ import { bindThis } from '@/decorators.js'; | |||||||
| import { RoleService } from '@/core/RoleService.js'; | import { RoleService } from '@/core/RoleService.js'; | ||||||
| import { correctFilename } from '@/misc/correct-filename.js'; | import { correctFilename } from '@/misc/correct-filename.js'; | ||||||
| import { isMimeImage } from '@/misc/is-mime-image.js'; | import { isMimeImage } from '@/misc/is-mime-image.js'; | ||||||
|  | import { ModerationLogService } from '@/core/ModerationLogService.js'; | ||||||
|  |  | ||||||
| type AddFileArgs = { | type AddFileArgs = { | ||||||
| 	/** User who wish to add file */ | 	/** User who wish to add file */ | ||||||
| @@ -86,6 +87,9 @@ type UploadFromUrlArgs = { | |||||||
|  |  | ||||||
| @Injectable() | @Injectable() | ||||||
| export class DriveService { | export class DriveService { | ||||||
|  | 	public static NoSuchFolderError = class extends Error {}; | ||||||
|  | 	public static InvalidFileNameError = class extends Error {}; | ||||||
|  | 	public static CannotUnmarkSensitiveError = class extends Error {}; | ||||||
| 	private registerLogger: Logger; | 	private registerLogger: Logger; | ||||||
| 	private downloaderLogger: Logger; | 	private downloaderLogger: Logger; | ||||||
| 	private deleteLogger: Logger; | 	private deleteLogger: Logger; | ||||||
| @@ -119,6 +123,7 @@ export class DriveService { | |||||||
| 		private globalEventService: GlobalEventService, | 		private globalEventService: GlobalEventService, | ||||||
| 		private queueService: QueueService, | 		private queueService: QueueService, | ||||||
| 		private roleService: RoleService, | 		private roleService: RoleService, | ||||||
|  | 		private moderationLogService: ModerationLogService, | ||||||
| 		private driveChart: DriveChart, | 		private driveChart: DriveChart, | ||||||
| 		private perUserDriveChart: PerUserDriveChart, | 		private perUserDriveChart: PerUserDriveChart, | ||||||
| 		private instanceChart: InstanceChart, | 		private instanceChart: InstanceChart, | ||||||
| @@ -648,7 +653,63 @@ export class DriveService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public async deleteFile(file: MiDriveFile, isExpired = false) { | 	public async updateFile(file: MiDriveFile, values: Partial<MiDriveFile>, updater: MiUser) { | ||||||
|  | 		const alwaysMarkNsfw = (await this.roleService.getUserPolicies(file.userId)).alwaysMarkNsfw; | ||||||
|  |  | ||||||
|  | 		if (values.name && !this.driveFileEntityService.validateFileName(file.name)) { | ||||||
|  | 			throw new DriveService.InvalidFileNameError(); | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if (values.isSensitive !== undefined && values.isSensitive !== file.isSensitive && alwaysMarkNsfw && !values.isSensitive) { | ||||||
|  | 			throw new DriveService.CannotUnmarkSensitiveError(); | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if (values.folderId != null) { | ||||||
|  | 			const folder = await this.driveFoldersRepository.findOneBy({ | ||||||
|  | 				id: values.folderId, | ||||||
|  | 				userId: file.userId!, | ||||||
|  | 			}); | ||||||
|  |  | ||||||
|  | 			if (folder == null) { | ||||||
|  | 				throw new DriveService.NoSuchFolderError(); | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		await this.driveFilesRepository.update(file.id, values); | ||||||
|  |  | ||||||
|  | 		const fileObj = await this.driveFileEntityService.pack(file.id, { self: true }); | ||||||
|  |  | ||||||
|  | 		// Publish fileUpdated event | ||||||
|  | 		if (file.userId) { | ||||||
|  | 			this.globalEventService.publishDriveStream(file.userId, 'fileUpdated', fileObj); | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if (await this.roleService.isModerator(updater) && (file.userId !== updater.id)) { | ||||||
|  | 			if (values.isSensitive !== undefined && values.isSensitive !== file.isSensitive) { | ||||||
|  | 				const user = file.userId ? await this.usersRepository.findOneByOrFail({ id: file.userId }) : null; | ||||||
|  | 				if (values.isSensitive) { | ||||||
|  | 					this.moderationLogService.log(updater, 'markSensitiveDriveFile', { | ||||||
|  | 						fileId: file.id, | ||||||
|  | 						fileUserId: file.userId, | ||||||
|  | 						fileUserUsername: user?.username ?? null, | ||||||
|  | 						fileUserHost: user?.host ?? null, | ||||||
|  | 					}); | ||||||
|  | 				} else { | ||||||
|  | 					this.moderationLogService.log(updater, 'unmarkSensitiveDriveFile', { | ||||||
|  | 						fileId: file.id, | ||||||
|  | 						fileUserId: file.userId, | ||||||
|  | 						fileUserUsername: user?.username ?? null, | ||||||
|  | 						fileUserHost: user?.host ?? null, | ||||||
|  | 					}); | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		return fileObj; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	@bindThis | ||||||
|  | 	public async deleteFile(file: MiDriveFile, isExpired = false, deleter?: MiUser) { | ||||||
| 		if (file.storedInternal) { | 		if (file.storedInternal) { | ||||||
| 			this.internalStorageService.del(file.accessKey!); | 			this.internalStorageService.del(file.accessKey!); | ||||||
|  |  | ||||||
| @@ -671,11 +732,11 @@ export class DriveService { | |||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		this.deletePostProcess(file, isExpired); | 		this.deletePostProcess(file, isExpired, deleter); | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public async deleteFileSync(file: MiDriveFile, isExpired = false) { | 	public async deleteFileSync(file: MiDriveFile, isExpired = false, deleter?: MiUser) { | ||||||
| 		if (file.storedInternal) { | 		if (file.storedInternal) { | ||||||
| 			this.internalStorageService.del(file.accessKey!); | 			this.internalStorageService.del(file.accessKey!); | ||||||
|  |  | ||||||
| @@ -702,11 +763,11 @@ export class DriveService { | |||||||
| 			await Promise.all(promises); | 			await Promise.all(promises); | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		this.deletePostProcess(file, isExpired); | 		this.deletePostProcess(file, isExpired, deleter); | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	private async deletePostProcess(file: MiDriveFile, isExpired = false) { | 	private async deletePostProcess(file: MiDriveFile, isExpired = false, deleter?: MiUser) { | ||||||
| 		// リモートファイル期限切れ削除後は直リンクにする | 		// リモートファイル期限切れ削除後は直リンクにする | ||||||
| 		if (isExpired && file.userHost !== null && file.uri != null) { | 		if (isExpired && file.userHost !== null && file.uri != null) { | ||||||
| 			this.driveFilesRepository.update(file.id, { | 			this.driveFilesRepository.update(file.id, { | ||||||
| @@ -733,6 +794,20 @@ export class DriveService { | |||||||
| 				this.instanceChart.updateDrive(file, false); | 				this.instanceChart.updateDrive(file, false); | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
|  | 		if (file.userId) { | ||||||
|  | 			this.globalEventService.publishDriveStream(file.userId, 'fileDeleted', file.id); | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if (deleter && await this.roleService.isModerator(deleter) && (file.userId !== deleter.id)) { | ||||||
|  | 			const user = file.userId ? await this.usersRepository.findOneByOrFail({ id: file.userId }) : null; | ||||||
|  | 			this.moderationLogService.log(deleter, 'deleteDriveFile', { | ||||||
|  | 				fileId: file.id, | ||||||
|  | 				fileUserId: file.userId, | ||||||
|  | 				fileUserUsername: user?.username ?? null, | ||||||
|  | 				fileUserHost: user?.host ?? null, | ||||||
|  | 			}); | ||||||
|  | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
|   | |||||||
| @@ -10,7 +10,7 @@ import { MetaService } from '@/core/MetaService.js'; | |||||||
| import { DI } from '@/di-symbols.js'; | import { DI } from '@/di-symbols.js'; | ||||||
| import type { Config } from '@/config.js'; | import type { Config } from '@/config.js'; | ||||||
| import type Logger from '@/logger.js'; | import type Logger from '@/logger.js'; | ||||||
| import type { UserProfilesRepository } from '@/models/index.js'; | import type { UserProfilesRepository } from '@/models/_.js'; | ||||||
| import { LoggerService } from '@/core/LoggerService.js'; | import { LoggerService } from '@/core/LoggerService.js'; | ||||||
| import { bindThis } from '@/decorators.js'; | import { bindThis } from '@/decorators.js'; | ||||||
|  |  | ||||||
|   | |||||||
| @@ -5,8 +5,8 @@ | |||||||
|  |  | ||||||
| import { Inject, Injectable, OnApplicationShutdown } from '@nestjs/common'; | import { Inject, Injectable, OnApplicationShutdown } from '@nestjs/common'; | ||||||
| import * as Redis from 'ioredis'; | import * as Redis from 'ioredis'; | ||||||
| import type { InstancesRepository } from '@/models/index.js'; | import type { InstancesRepository } from '@/models/_.js'; | ||||||
| import type { MiInstance } from '@/models/entities/Instance.js'; | import type { MiInstance } from '@/models/Instance.js'; | ||||||
| import { MemoryKVCache, RedisKVCache } from '@/misc/cache.js'; | import { MemoryKVCache, RedisKVCache } from '@/misc/cache.js'; | ||||||
| import { IdService } from '@/core/IdService.js'; | import { IdService } from '@/core/IdService.js'; | ||||||
| import { DI } from '@/di-symbols.js'; | import { DI } from '@/di-symbols.js'; | ||||||
|   | |||||||
| @@ -8,7 +8,7 @@ import { Inject, Injectable } from '@nestjs/common'; | |||||||
| import { JSDOM } from 'jsdom'; | import { JSDOM } from 'jsdom'; | ||||||
| import tinycolor from 'tinycolor2'; | import tinycolor from 'tinycolor2'; | ||||||
| import * as Redis from 'ioredis'; | import * as Redis from 'ioredis'; | ||||||
| import type { MiInstance } from '@/models/entities/Instance.js'; | import type { MiInstance } from '@/models/Instance.js'; | ||||||
| import type Logger from '@/logger.js'; | import type Logger from '@/logger.js'; | ||||||
| import { DI } from '@/di-symbols.js'; | import { DI } from '@/di-symbols.js'; | ||||||
| import { LoggerService } from '@/core/LoggerService.js'; | import { LoggerService } from '@/core/LoggerService.js'; | ||||||
|   | |||||||
| @@ -5,27 +5,254 @@ | |||||||
|  |  | ||||||
| import { Inject, Injectable } from '@nestjs/common'; | import { Inject, Injectable } from '@nestjs/common'; | ||||||
| import * as Redis from 'ioredis'; | import * as Redis from 'ioredis'; | ||||||
| import type { MiUser } from '@/models/entities/User.js'; | import type { MiChannel } from '@/models/Channel.js'; | ||||||
| import type { MiNote } from '@/models/entities/Note.js'; | import type { MiUser } from '@/models/User.js'; | ||||||
| import type { MiUserList } from '@/models/entities/UserList.js'; | import type { MiUserProfile } from '@/models/UserProfile.js'; | ||||||
| import type { MiAntenna } from '@/models/entities/Antenna.js'; | import type { MiNote } from '@/models/Note.js'; | ||||||
| import type { | import type { MiAntenna } from '@/models/Antenna.js'; | ||||||
| 	StreamChannels, | import type { MiDriveFile } from '@/models/DriveFile.js'; | ||||||
| 	AdminStreamTypes, | import type { MiDriveFolder } from '@/models/DriveFolder.js'; | ||||||
| 	AntennaStreamTypes, | import type { MiUserList } from '@/models/UserList.js'; | ||||||
| 	BroadcastTypes, | import type { MiAbuseUserReport } from '@/models/AbuseUserReport.js'; | ||||||
| 	DriveStreamTypes, | import type { MiSignin } from '@/models/Signin.js'; | ||||||
| 	InternalStreamTypes, | import type { MiPage } from '@/models/Page.js'; | ||||||
| 	MainStreamTypes, | import type { MiWebhook } from '@/models/Webhook.js'; | ||||||
| 	NoteStreamTypes, | import type { MiMeta } from '@/models/Meta.js'; | ||||||
| 	UserListStreamTypes, | import { MiRole, MiRoleAssignment } from '@/models/_.js'; | ||||||
| 	RoleTimelineStreamTypes, |  | ||||||
| } from '@/server/api/stream/types.js'; |  | ||||||
| import type { Packed } from '@/misc/json-schema.js'; | import type { Packed } from '@/misc/json-schema.js'; | ||||||
| import { DI } from '@/di-symbols.js'; | import { DI } from '@/di-symbols.js'; | ||||||
| import type { Config } from '@/config.js'; | import type { Config } from '@/config.js'; | ||||||
| import { bindThis } from '@/decorators.js'; | import { bindThis } from '@/decorators.js'; | ||||||
| import { MiRole } from '@/models/index.js'; | import { Serialized } from '@/types.js'; | ||||||
|  | import type Emitter from 'strict-event-emitter-types'; | ||||||
|  | import type { EventEmitter } from 'events'; | ||||||
|  |  | ||||||
|  | //#region Stream type-body definitions | ||||||
|  | export interface BroadcastTypes { | ||||||
|  | 	emojiAdded: { | ||||||
|  | 		emoji: Packed<'EmojiDetailed'>; | ||||||
|  | 	}; | ||||||
|  | 	emojiUpdated: { | ||||||
|  | 		emojis: Packed<'EmojiDetailed'>[]; | ||||||
|  | 	}; | ||||||
|  | 	emojiDeleted: { | ||||||
|  | 		emojis: { | ||||||
|  | 			id?: string; | ||||||
|  | 			name: string; | ||||||
|  | 			[other: string]: any; | ||||||
|  | 		}[]; | ||||||
|  | 	}; | ||||||
|  | 	announcementCreated: { | ||||||
|  | 		announcement: Packed<'Announcement'>; | ||||||
|  | 	}; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export interface MainEventTypes { | ||||||
|  | 	notification: Packed<'Notification'>; | ||||||
|  | 	mention: Packed<'Note'>; | ||||||
|  | 	reply: Packed<'Note'>; | ||||||
|  | 	renote: Packed<'Note'>; | ||||||
|  | 	follow: Packed<'UserDetailedNotMe'>; | ||||||
|  | 	followed: Packed<'User'>; | ||||||
|  | 	unfollow: Packed<'User'>; | ||||||
|  | 	meUpdated: Packed<'User'>; | ||||||
|  | 	pageEvent: { | ||||||
|  | 		pageId: MiPage['id']; | ||||||
|  | 		event: string; | ||||||
|  | 		var: any; | ||||||
|  | 		userId: MiUser['id']; | ||||||
|  | 		user: Packed<'User'>; | ||||||
|  | 	}; | ||||||
|  | 	urlUploadFinished: { | ||||||
|  | 		marker?: string | null; | ||||||
|  | 		file: Packed<'DriveFile'>; | ||||||
|  | 	}; | ||||||
|  | 	readAllNotifications: undefined; | ||||||
|  | 	unreadNotification: Packed<'Notification'>; | ||||||
|  | 	unreadMention: MiNote['id']; | ||||||
|  | 	readAllUnreadMentions: undefined; | ||||||
|  | 	unreadSpecifiedNote: MiNote['id']; | ||||||
|  | 	readAllUnreadSpecifiedNotes: undefined; | ||||||
|  | 	readAllAntennas: undefined; | ||||||
|  | 	unreadAntenna: MiAntenna; | ||||||
|  | 	readAllAnnouncements: undefined; | ||||||
|  | 	myTokenRegenerated: undefined; | ||||||
|  | 	signin: MiSignin; | ||||||
|  | 	registryUpdated: { | ||||||
|  | 		scope?: string[]; | ||||||
|  | 		key: string; | ||||||
|  | 		value: any | null; | ||||||
|  | 	}; | ||||||
|  | 	driveFileCreated: Packed<'DriveFile'>; | ||||||
|  | 	readAntenna: MiAntenna; | ||||||
|  | 	receiveFollowRequest: Packed<'User'>; | ||||||
|  | 	announcementCreated: { | ||||||
|  | 		announcement: Packed<'Announcement'>; | ||||||
|  | 	}; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export interface DriveEventTypes { | ||||||
|  | 	fileCreated: Packed<'DriveFile'>; | ||||||
|  | 	fileDeleted: MiDriveFile['id']; | ||||||
|  | 	fileUpdated: Packed<'DriveFile'>; | ||||||
|  | 	folderCreated: Packed<'DriveFolder'>; | ||||||
|  | 	folderDeleted: MiDriveFolder['id']; | ||||||
|  | 	folderUpdated: Packed<'DriveFolder'>; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export interface NoteEventTypes { | ||||||
|  | 	pollVoted: { | ||||||
|  | 		choice: number; | ||||||
|  | 		userId: MiUser['id']; | ||||||
|  | 	}; | ||||||
|  | 	deleted: { | ||||||
|  | 		deletedAt: Date; | ||||||
|  | 	}; | ||||||
|  | 	updated: { | ||||||
|  | 		cw: string | null; | ||||||
|  | 		text: string; | ||||||
|  | 	}; | ||||||
|  | 	reacted: { | ||||||
|  | 		reaction: string; | ||||||
|  | 		emoji?: { | ||||||
|  | 			name: string; | ||||||
|  | 			url: string; | ||||||
|  | 		} | null; | ||||||
|  | 		userId: MiUser['id']; | ||||||
|  | 	}; | ||||||
|  | 	unreacted: { | ||||||
|  | 		reaction: string; | ||||||
|  | 		userId: MiUser['id']; | ||||||
|  | 	}; | ||||||
|  | } | ||||||
|  | type NoteStreamEventTypes = { | ||||||
|  | 	[key in keyof NoteEventTypes]: { | ||||||
|  | 		id: MiNote['id']; | ||||||
|  | 		body: NoteEventTypes[key]; | ||||||
|  | 	}; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | export interface UserListEventTypes { | ||||||
|  | 	userAdded: Packed<'User'>; | ||||||
|  | 	userRemoved: Packed<'User'>; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export interface AntennaEventTypes { | ||||||
|  | 	note: MiNote; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export interface RoleTimelineEventTypes { | ||||||
|  | 	note: Packed<'Note'>; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export interface AdminEventTypes { | ||||||
|  | 	newAbuseUserReport: { | ||||||
|  | 		id: MiAbuseUserReport['id']; | ||||||
|  | 		targetUserId: MiUser['id'], | ||||||
|  | 		reporterId: MiUser['id'], | ||||||
|  | 		comment: string; | ||||||
|  | 	}; | ||||||
|  | } | ||||||
|  | //#endregion | ||||||
|  |  | ||||||
|  | // 辞書(interface or type)から{ type, body }ユニオンを定義 | ||||||
|  | // https://stackoverflow.com/questions/49311989/can-i-infer-the-type-of-a-value-using-extends-keyof-type | ||||||
|  | // VS Codeの展開を防止するためにEvents型を定義 | ||||||
|  | type Events<T extends object> = { [K in keyof T]: { type: K; body: T[K]; } }; | ||||||
|  | type EventUnionFromDictionary< | ||||||
|  | 	T extends object, | ||||||
|  | 	U = Events<T> | ||||||
|  | > = U[keyof U]; | ||||||
|  |  | ||||||
|  | type SerializedAll<T> = { | ||||||
|  | 	[K in keyof T]: Serialized<T[K]>; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | export interface InternalEventTypes { | ||||||
|  | 	userChangeSuspendedState: { id: MiUser['id']; isSuspended: MiUser['isSuspended']; }; | ||||||
|  | 	userTokenRegenerated: { id: MiUser['id']; oldToken: string; newToken: string; }; | ||||||
|  | 	remoteUserUpdated: { id: MiUser['id']; }; | ||||||
|  | 	follow: { followerId: MiUser['id']; followeeId: MiUser['id']; }; | ||||||
|  | 	unfollow: { followerId: MiUser['id']; followeeId: MiUser['id']; }; | ||||||
|  | 	blockingCreated: { blockerId: MiUser['id']; blockeeId: MiUser['id']; }; | ||||||
|  | 	blockingDeleted: { blockerId: MiUser['id']; blockeeId: MiUser['id']; }; | ||||||
|  | 	policiesUpdated: MiRole['policies']; | ||||||
|  | 	roleCreated: MiRole; | ||||||
|  | 	roleDeleted: MiRole; | ||||||
|  | 	roleUpdated: MiRole; | ||||||
|  | 	userRoleAssigned: MiRoleAssignment; | ||||||
|  | 	userRoleUnassigned: MiRoleAssignment; | ||||||
|  | 	webhookCreated: MiWebhook; | ||||||
|  | 	webhookDeleted: MiWebhook; | ||||||
|  | 	webhookUpdated: MiWebhook; | ||||||
|  | 	antennaCreated: MiAntenna; | ||||||
|  | 	antennaDeleted: MiAntenna; | ||||||
|  | 	antennaUpdated: MiAntenna; | ||||||
|  | 	metaUpdated: MiMeta; | ||||||
|  | 	followChannel: { userId: MiUser['id']; channelId: MiChannel['id']; }; | ||||||
|  | 	unfollowChannel: { userId: MiUser['id']; channelId: MiChannel['id']; }; | ||||||
|  | 	updateUserProfile: MiUserProfile; | ||||||
|  | 	mute: { muterId: MiUser['id']; muteeId: MiUser['id']; }; | ||||||
|  | 	unmute: { muterId: MiUser['id']; muteeId: MiUser['id']; }; | ||||||
|  | 	userListMemberAdded: { userListId: MiUserList['id']; memberId: MiUser['id']; }; | ||||||
|  | 	userListMemberRemoved: { userListId: MiUserList['id']; memberId: MiUser['id']; }; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // name/messages(spec) pairs dictionary | ||||||
|  | export type GlobalEvents = { | ||||||
|  | 	internal: { | ||||||
|  | 		name: 'internal'; | ||||||
|  | 		payload: EventUnionFromDictionary<SerializedAll<InternalEventTypes>>; | ||||||
|  | 	}; | ||||||
|  | 	broadcast: { | ||||||
|  | 		name: 'broadcast'; | ||||||
|  | 		payload: EventUnionFromDictionary<SerializedAll<BroadcastTypes>>; | ||||||
|  | 	}; | ||||||
|  | 	main: { | ||||||
|  | 		name: `mainStream:${MiUser['id']}`; | ||||||
|  | 		payload: EventUnionFromDictionary<SerializedAll<MainEventTypes>>; | ||||||
|  | 	}; | ||||||
|  | 	drive: { | ||||||
|  | 		name: `driveStream:${MiUser['id']}`; | ||||||
|  | 		payload: EventUnionFromDictionary<SerializedAll<DriveEventTypes>>; | ||||||
|  | 	}; | ||||||
|  | 	note: { | ||||||
|  | 		name: `noteStream:${MiNote['id']}`; | ||||||
|  | 		payload: EventUnionFromDictionary<SerializedAll<NoteStreamEventTypes>>; | ||||||
|  | 	}; | ||||||
|  | 	userList: { | ||||||
|  | 		name: `userListStream:${MiUserList['id']}`; | ||||||
|  | 		payload: EventUnionFromDictionary<SerializedAll<UserListEventTypes>>; | ||||||
|  | 	}; | ||||||
|  | 	roleTimeline: { | ||||||
|  | 		name: `roleTimelineStream:${MiRole['id']}`; | ||||||
|  | 		payload: EventUnionFromDictionary<SerializedAll<RoleTimelineEventTypes>>; | ||||||
|  | 	}; | ||||||
|  | 	antenna: { | ||||||
|  | 		name: `antennaStream:${MiAntenna['id']}`; | ||||||
|  | 		payload: EventUnionFromDictionary<SerializedAll<AntennaEventTypes>>; | ||||||
|  | 	}; | ||||||
|  | 	admin: { | ||||||
|  | 		name: `adminStream:${MiUser['id']}`; | ||||||
|  | 		payload: EventUnionFromDictionary<SerializedAll<AdminEventTypes>>; | ||||||
|  | 	}; | ||||||
|  | 	notes: { | ||||||
|  | 		name: 'notesStream'; | ||||||
|  | 		payload: Serialized<Packed<'Note'>>; | ||||||
|  | 	}; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | // API event definitions | ||||||
|  | // ストリームごとのEmitterの辞書を用意 | ||||||
|  | type EventEmitterDictionary = { [x in keyof GlobalEvents]: Emitter.default<EventEmitter, { [y in GlobalEvents[x]['name']]: (e: GlobalEvents[x]['payload']) => void }> }; | ||||||
|  | // 共用体型を交差型にする型 https://stackoverflow.com/questions/54938141/typescript-convert-union-to-intersection | ||||||
|  | type UnionToIntersection<U> = (U extends any ? (k: U) => void : never) extends ((k: infer I) => void) ? I : never; | ||||||
|  | // Emitter辞書から共用体型を作り、UnionToIntersectionで交差型にする | ||||||
|  | export type StreamEventEmitter = UnionToIntersection<EventEmitterDictionary[keyof GlobalEvents]>; | ||||||
|  | // { [y in name]: (e: spec) => void }をまとめてその交差型をEmitterにかけるとts(2590)にひっかかる | ||||||
|  |  | ||||||
|  | // provide stream channels union | ||||||
|  | export type StreamChannels = GlobalEvents[keyof GlobalEvents]['name']; | ||||||
|  |  | ||||||
| @Injectable() | @Injectable() | ||||||
| export class GlobalEventService { | export class GlobalEventService { | ||||||
| @@ -51,7 +278,7 @@ export class GlobalEventService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public publishInternalEvent<K extends keyof InternalStreamTypes>(type: K, value?: InternalStreamTypes[K]): void { | 	public publishInternalEvent<K extends keyof InternalEventTypes>(type: K, value?: InternalEventTypes[K]): void { | ||||||
| 		this.publish('internal', type, typeof value === 'undefined' ? null : value); | 		this.publish('internal', type, typeof value === 'undefined' ? null : value); | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| @@ -61,17 +288,17 @@ export class GlobalEventService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public publishMainStream<K extends keyof MainStreamTypes>(userId: MiUser['id'], type: K, value?: MainStreamTypes[K]): void { | 	public publishMainStream<K extends keyof MainEventTypes>(userId: MiUser['id'], type: K, value?: MainEventTypes[K]): void { | ||||||
| 		this.publish(`mainStream:${userId}`, type, typeof value === 'undefined' ? null : value); | 		this.publish(`mainStream:${userId}`, type, typeof value === 'undefined' ? null : value); | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public publishDriveStream<K extends keyof DriveStreamTypes>(userId: MiUser['id'], type: K, value?: DriveStreamTypes[K]): void { | 	public publishDriveStream<K extends keyof DriveEventTypes>(userId: MiUser['id'], type: K, value?: DriveEventTypes[K]): void { | ||||||
| 		this.publish(`driveStream:${userId}`, type, typeof value === 'undefined' ? null : value); | 		this.publish(`driveStream:${userId}`, type, typeof value === 'undefined' ? null : value); | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public publishNoteStream<K extends keyof NoteStreamTypes>(noteId: MiNote['id'], type: K, value?: NoteStreamTypes[K]): void { | 	public publishNoteStream<K extends keyof NoteEventTypes>(noteId: MiNote['id'], type: K, value?: NoteEventTypes[K]): void { | ||||||
| 		this.publish(`noteStream:${noteId}`, type, { | 		this.publish(`noteStream:${noteId}`, type, { | ||||||
| 			id: noteId, | 			id: noteId, | ||||||
| 			body: value, | 			body: value, | ||||||
| @@ -79,17 +306,17 @@ export class GlobalEventService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public publishUserListStream<K extends keyof UserListStreamTypes>(listId: MiUserList['id'], type: K, value?: UserListStreamTypes[K]): void { | 	public publishUserListStream<K extends keyof UserListEventTypes>(listId: MiUserList['id'], type: K, value?: UserListEventTypes[K]): void { | ||||||
| 		this.publish(`userListStream:${listId}`, type, typeof value === 'undefined' ? null : value); | 		this.publish(`userListStream:${listId}`, type, typeof value === 'undefined' ? null : value); | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public publishAntennaStream<K extends keyof AntennaStreamTypes>(antennaId: MiAntenna['id'], type: K, value?: AntennaStreamTypes[K]): void { | 	public publishAntennaStream<K extends keyof AntennaEventTypes>(antennaId: MiAntenna['id'], type: K, value?: AntennaEventTypes[K]): void { | ||||||
| 		this.publish(`antennaStream:${antennaId}`, type, typeof value === 'undefined' ? null : value); | 		this.publish(`antennaStream:${antennaId}`, type, typeof value === 'undefined' ? null : value); | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public publishRoleTimelineStream<K extends keyof RoleTimelineStreamTypes>(roleId: MiRole['id'], type: K, value?: RoleTimelineStreamTypes[K]): void { | 	public publishRoleTimelineStream<K extends keyof RoleTimelineEventTypes>(roleId: MiRole['id'], type: K, value?: RoleTimelineEventTypes[K]): void { | ||||||
| 		this.publish(`roleTimelineStream:${roleId}`, type, typeof value === 'undefined' ? null : value); | 		this.publish(`roleTimelineStream:${roleId}`, type, typeof value === 'undefined' ? null : value); | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| @@ -99,7 +326,7 @@ export class GlobalEventService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public publishAdminStream<K extends keyof AdminStreamTypes>(userId: MiUser['id'], type: K, value?: AdminStreamTypes[K]): void { | 	public publishAdminStream<K extends keyof AdminEventTypes>(userId: MiUser['id'], type: K, value?: AdminEventTypes[K]): void { | ||||||
| 		this.publish(`adminStream:${userId}`, type, typeof value === 'undefined' ? null : value); | 		this.publish(`adminStream:${userId}`, type, typeof value === 'undefined' ? null : value); | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|   | |||||||
| @@ -5,11 +5,11 @@ | |||||||
|  |  | ||||||
| import { Inject, Injectable } from '@nestjs/common'; | import { Inject, Injectable } from '@nestjs/common'; | ||||||
| import { DI } from '@/di-symbols.js'; | import { DI } from '@/di-symbols.js'; | ||||||
| import type { MiUser } from '@/models/entities/User.js'; | import type { MiUser } from '@/models/User.js'; | ||||||
| import { normalizeForSearch } from '@/misc/normalize-for-search.js'; | import { normalizeForSearch } from '@/misc/normalize-for-search.js'; | ||||||
| import { IdService } from '@/core/IdService.js'; | import { IdService } from '@/core/IdService.js'; | ||||||
| import type { MiHashtag } from '@/models/entities/Hashtag.js'; | import type { MiHashtag } from '@/models/Hashtag.js'; | ||||||
| import type { HashtagsRepository } from '@/models/index.js'; | import type { HashtagsRepository } from '@/models/_.js'; | ||||||
| import { UserEntityService } from '@/core/entities/UserEntityService.js'; | import { UserEntityService } from '@/core/entities/UserEntityService.js'; | ||||||
| import { bindThis } from '@/decorators.js'; | import { bindThis } from '@/decorators.js'; | ||||||
|  |  | ||||||
|   | |||||||
| @@ -5,8 +5,8 @@ | |||||||
|  |  | ||||||
| import { Inject, Injectable } from '@nestjs/common'; | import { Inject, Injectable } from '@nestjs/common'; | ||||||
| import { IsNull } from 'typeorm'; | import { IsNull } from 'typeorm'; | ||||||
| import type { MiLocalUser } from '@/models/entities/User.js'; | import type { MiLocalUser } from '@/models/User.js'; | ||||||
| import type { UsersRepository } from '@/models/index.js'; | import type { UsersRepository } from '@/models/_.js'; | ||||||
| import { MemorySingleCache } from '@/misc/cache.js'; | import { MemorySingleCache } from '@/misc/cache.js'; | ||||||
| import { DI } from '@/di-symbols.js'; | import { DI } from '@/di-symbols.js'; | ||||||
| import { CreateSystemUserService } from '@/core/CreateSystemUserService.js'; | import { CreateSystemUserService } from '@/core/CreateSystemUserService.js'; | ||||||
|   | |||||||
| @@ -7,10 +7,10 @@ import { Inject, Injectable } from '@nestjs/common'; | |||||||
| import { DataSource } from 'typeorm'; | import { DataSource } from 'typeorm'; | ||||||
| import * as Redis from 'ioredis'; | import * as Redis from 'ioredis'; | ||||||
| import { DI } from '@/di-symbols.js'; | import { DI } from '@/di-symbols.js'; | ||||||
| import { MiMeta } from '@/models/entities/Meta.js'; | import { MiMeta } from '@/models/Meta.js'; | ||||||
| import { GlobalEventService } from '@/core/GlobalEventService.js'; | import { GlobalEventService } from '@/core/GlobalEventService.js'; | ||||||
| import { bindThis } from '@/decorators.js'; | import { bindThis } from '@/decorators.js'; | ||||||
| import { StreamMessages } from '@/server/api/stream/types.js'; | import type { GlobalEvents } from '@/core/GlobalEventService.js'; | ||||||
| import type { OnApplicationShutdown } from '@nestjs/common'; | import type { OnApplicationShutdown } from '@nestjs/common'; | ||||||
|  |  | ||||||
| @Injectable() | @Injectable() | ||||||
| @@ -46,7 +46,7 @@ export class MetaService implements OnApplicationShutdown { | |||||||
| 		const obj = JSON.parse(data); | 		const obj = JSON.parse(data); | ||||||
|  |  | ||||||
| 		if (obj.channel === 'internal') { | 		if (obj.channel === 'internal') { | ||||||
| 			const { type, body } = obj.message as StreamMessages['internal']['payload']; | 			const { type, body } = obj.message as GlobalEvents['internal']['payload']; | ||||||
| 			switch (type) { | 			switch (type) { | ||||||
| 				case 'metaUpdated': { | 				case 'metaUpdated': { | ||||||
| 					this.cache = body; | 					this.cache = body; | ||||||
|   | |||||||
| @@ -10,7 +10,7 @@ import { Window } from 'happy-dom'; | |||||||
| import { DI } from '@/di-symbols.js'; | import { DI } from '@/di-symbols.js'; | ||||||
| import type { Config } from '@/config.js'; | import type { Config } from '@/config.js'; | ||||||
| import { intersperse } from '@/misc/prelude/array.js'; | import { intersperse } from '@/misc/prelude/array.js'; | ||||||
| import type { IMentionedRemoteUsers } from '@/models/entities/Note.js'; | import type { IMentionedRemoteUsers } from '@/models/Note.js'; | ||||||
| import { bindThis } from '@/decorators.js'; | import { bindThis } from '@/decorators.js'; | ||||||
| import * as TreeAdapter from '../../node_modules/parse5/dist/tree-adapters/default.js'; | import * as TreeAdapter from '../../node_modules/parse5/dist/tree-adapters/default.js'; | ||||||
| import type * as mfm from 'mfm-js'; | import type * as mfm from 'mfm-js'; | ||||||
|   | |||||||
| @@ -5,10 +5,11 @@ | |||||||
|  |  | ||||||
| import { Inject, Injectable } from '@nestjs/common'; | import { Inject, Injectable } from '@nestjs/common'; | ||||||
| import { DI } from '@/di-symbols.js'; | import { DI } from '@/di-symbols.js'; | ||||||
| import type { ModerationLogsRepository } from '@/models/index.js'; | import type { ModerationLogsRepository } from '@/models/_.js'; | ||||||
| import type { MiUser } from '@/models/entities/User.js'; | import type { MiUser } from '@/models/User.js'; | ||||||
| import { IdService } from '@/core/IdService.js'; | import { IdService } from '@/core/IdService.js'; | ||||||
| import { bindThis } from '@/decorators.js'; | import { bindThis } from '@/decorators.js'; | ||||||
|  | import { ModerationLogPayloads, moderationLogTypes } from '@/types.js'; | ||||||
|  |  | ||||||
| @Injectable() | @Injectable() | ||||||
| export class ModerationLogService { | export class ModerationLogService { | ||||||
| @@ -21,13 +22,13 @@ export class ModerationLogService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public async insertModerationLog(moderator: { id: MiUser['id'] }, type: string, info?: Record<string, any>) { | 	public async log<T extends typeof moderationLogTypes[number]>(moderator: { id: MiUser['id'] }, type: T, info?: ModerationLogPayloads[T]) { | ||||||
| 		await this.moderationLogsRepository.insert({ | 		await this.moderationLogsRepository.insert({ | ||||||
| 			id: this.idService.genId(), | 			id: this.idService.genId(), | ||||||
| 			createdAt: new Date(), | 			createdAt: new Date(), | ||||||
| 			userId: moderator.id, | 			userId: moderator.id, | ||||||
| 			type: type, | 			type: type, | ||||||
| 			info: info ?? {}, | 			info: (info as any) ?? {}, | ||||||
| 		}); | 		}); | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|   | |||||||
| @@ -5,29 +5,29 @@ | |||||||
|  |  | ||||||
| import { setImmediate } from 'node:timers/promises'; | import { setImmediate } from 'node:timers/promises'; | ||||||
| import * as mfm from 'mfm-js'; | import * as mfm from 'mfm-js'; | ||||||
| import { In, DataSource } from 'typeorm'; | import { In, DataSource, IsNull, LessThan } from 'typeorm'; | ||||||
| import * as Redis from 'ioredis'; | import * as Redis from 'ioredis'; | ||||||
| import { Inject, Injectable, OnApplicationShutdown } from '@nestjs/common'; | import { Inject, Injectable, OnApplicationShutdown } from '@nestjs/common'; | ||||||
| import RE2 from 're2'; | import RE2 from 're2'; | ||||||
| import { extractMentions } from '@/misc/extract-mentions.js'; | import { extractMentions } from '@/misc/extract-mentions.js'; | ||||||
| import { extractCustomEmojisFromMfm } from '@/misc/extract-custom-emojis-from-mfm.js'; | import { extractCustomEmojisFromMfm } from '@/misc/extract-custom-emojis-from-mfm.js'; | ||||||
| import { extractHashtags } from '@/misc/extract-hashtags.js'; | import { extractHashtags } from '@/misc/extract-hashtags.js'; | ||||||
| import type { IMentionedRemoteUsers } from '@/models/entities/Note.js'; | import type { IMentionedRemoteUsers } from '@/models/Note.js'; | ||||||
| import { MiNote } from '@/models/entities/Note.js'; | import { MiNote } from '@/models/Note.js'; | ||||||
| import type { ChannelsRepository, InstancesRepository, MutedNotesRepository, MutingsRepository, NotesRepository, NoteThreadMutingsRepository, UserProfilesRepository, UsersRepository } from '@/models/index.js'; | import type { ChannelFollowingsRepository, ChannelsRepository, FollowingsRepository, InstancesRepository, MiFollowing, MutingsRepository, NotesRepository, NoteThreadMutingsRepository, UserListMembershipsRepository, UserProfilesRepository, UsersRepository } from '@/models/_.js'; | ||||||
| import type { MiDriveFile } from '@/models/entities/DriveFile.js'; | import type { MiDriveFile } from '@/models/DriveFile.js'; | ||||||
| import type { MiApp } from '@/models/entities/App.js'; | import type { MiApp } from '@/models/App.js'; | ||||||
| import { concat } from '@/misc/prelude/array.js'; | import { concat } from '@/misc/prelude/array.js'; | ||||||
| import { IdService } from '@/core/IdService.js'; | import { IdService } from '@/core/IdService.js'; | ||||||
| import type { MiUser, MiLocalUser, MiRemoteUser } from '@/models/entities/User.js'; | import type { MiUser, MiLocalUser, MiRemoteUser } from '@/models/User.js'; | ||||||
| import type { IPoll } from '@/models/entities/Poll.js'; | import type { IPoll } from '@/models/Poll.js'; | ||||||
| import { MiPoll } from '@/models/entities/Poll.js'; | import { MiPoll } from '@/models/Poll.js'; | ||||||
| import { isDuplicateKeyValueError } from '@/misc/is-duplicate-key-value-error.js'; | import { isDuplicateKeyValueError } from '@/misc/is-duplicate-key-value-error.js'; | ||||||
| import { checkWordMute } from '@/misc/check-word-mute.js'; | import { checkWordMute } from '@/misc/check-word-mute.js'; | ||||||
| import type { MiChannel } from '@/models/entities/Channel.js'; | import type { MiChannel } from '@/models/Channel.js'; | ||||||
| import { normalizeForSearch } from '@/misc/normalize-for-search.js'; | import { normalizeForSearch } from '@/misc/normalize-for-search.js'; | ||||||
| import { MemorySingleCache } from '@/misc/cache.js'; | import { MemorySingleCache } from '@/misc/cache.js'; | ||||||
| import type { MiUserProfile } from '@/models/entities/UserProfile.js'; | import type { MiUserProfile } from '@/models/UserProfile.js'; | ||||||
| import { RelayService } from '@/core/RelayService.js'; | import { RelayService } from '@/core/RelayService.js'; | ||||||
| import { FederatedInstanceService } from '@/core/FederatedInstanceService.js'; | import { FederatedInstanceService } from '@/core/FederatedInstanceService.js'; | ||||||
| import { DI } from '@/di-symbols.js'; | import { DI } from '@/di-symbols.js'; | ||||||
| @@ -54,8 +54,6 @@ import { RoleService } from '@/core/RoleService.js'; | |||||||
| import { MetaService } from '@/core/MetaService.js'; | import { MetaService } from '@/core/MetaService.js'; | ||||||
| import { SearchService } from '@/core/SearchService.js'; | import { SearchService } from '@/core/SearchService.js'; | ||||||
|  |  | ||||||
| const mutedWordsCache = new MemorySingleCache<{ userId: MiUserProfile['userId']; mutedWords: MiUserProfile['mutedWords']; }[]>(1000 * 60 * 5); |  | ||||||
|  |  | ||||||
| type NotificationType = 'reply' | 'renote' | 'quote' | 'mention'; | type NotificationType = 'reply' | 'renote' | 'quote' | 'mention'; | ||||||
|  |  | ||||||
| class NotificationManager { | class NotificationManager { | ||||||
| @@ -110,9 +108,8 @@ class NotificationManager { | |||||||
| 			// 通知される側のユーザーが通知する側のユーザーをミュートしていない限りは通知する | 			// 通知される側のユーザーが通知する側のユーザーをミュートしていない限りは通知する | ||||||
| 			if (!mentioneesMutedUserIds.includes(this.notifier.id)) { | 			if (!mentioneesMutedUserIds.includes(this.notifier.id)) { | ||||||
| 				this.notificationService.createNotification(x.target, x.reason, { | 				this.notificationService.createNotification(x.target, x.reason, { | ||||||
| 					notifierId: this.notifier.id, |  | ||||||
| 					noteId: this.note.id, | 					noteId: this.note.id, | ||||||
| 				}); | 				}, this.notifier.id); | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| @@ -158,8 +155,8 @@ export class NoteCreateService implements OnApplicationShutdown { | |||||||
| 		@Inject(DI.db) | 		@Inject(DI.db) | ||||||
| 		private db: DataSource, | 		private db: DataSource, | ||||||
|  |  | ||||||
| 		@Inject(DI.redis) | 		@Inject(DI.redisForTimelines) | ||||||
| 		private redisClient: Redis.Redis, | 		private redisForTimelines: Redis.Redis, | ||||||
|  |  | ||||||
| 		@Inject(DI.usersRepository) | 		@Inject(DI.usersRepository) | ||||||
| 		private usersRepository: UsersRepository, | 		private usersRepository: UsersRepository, | ||||||
| @@ -176,8 +173,8 @@ export class NoteCreateService implements OnApplicationShutdown { | |||||||
| 		@Inject(DI.userProfilesRepository) | 		@Inject(DI.userProfilesRepository) | ||||||
| 		private userProfilesRepository: UserProfilesRepository, | 		private userProfilesRepository: UserProfilesRepository, | ||||||
|  |  | ||||||
| 		@Inject(DI.mutedNotesRepository) | 		@Inject(DI.userListMembershipsRepository) | ||||||
| 		private mutedNotesRepository: MutedNotesRepository, | 		private userListMembershipsRepository: UserListMembershipsRepository, | ||||||
|  |  | ||||||
| 		@Inject(DI.channelsRepository) | 		@Inject(DI.channelsRepository) | ||||||
| 		private channelsRepository: ChannelsRepository, | 		private channelsRepository: ChannelsRepository, | ||||||
| @@ -185,6 +182,12 @@ export class NoteCreateService implements OnApplicationShutdown { | |||||||
| 		@Inject(DI.noteThreadMutingsRepository) | 		@Inject(DI.noteThreadMutingsRepository) | ||||||
| 		private noteThreadMutingsRepository: NoteThreadMutingsRepository, | 		private noteThreadMutingsRepository: NoteThreadMutingsRepository, | ||||||
|  |  | ||||||
|  | 		@Inject(DI.followingsRepository) | ||||||
|  | 		private followingsRepository: FollowingsRepository, | ||||||
|  |  | ||||||
|  | 		@Inject(DI.channelFollowingsRepository) | ||||||
|  | 		private channelFollowingsRepository: ChannelFollowingsRepository, | ||||||
|  |  | ||||||
| 		private userEntityService: UserEntityService, | 		private userEntityService: UserEntityService, | ||||||
| 		private noteEntityService: NoteEntityService, | 		private noteEntityService: NoteEntityService, | ||||||
| 		private idService: IdService, | 		private idService: IdService, | ||||||
| @@ -332,7 +335,7 @@ export class NoteCreateService implements OnApplicationShutdown { | |||||||
| 		const note = await this.insertNote(user, data, tags, emojis, mentionedUsers); | 		const note = await this.insertNote(user, data, tags, emojis, mentionedUsers); | ||||||
|  |  | ||||||
| 		if (data.channel) { | 		if (data.channel) { | ||||||
| 			this.redisClient.xadd( | 			this.redisForTimelines.xadd( | ||||||
| 				`channelTimeline:${data.channel.id}`, | 				`channelTimeline:${data.channel.id}`, | ||||||
| 				'MAXLEN', '~', this.config.perChannelMaxNoteCacheCount.toString(), | 				'MAXLEN', '~', this.config.perChannelMaxNoteCacheCount.toString(), | ||||||
| 				'*', | 				'*', | ||||||
| @@ -478,26 +481,13 @@ export class NoteCreateService implements OnApplicationShutdown { | |||||||
| 		// Increment notes count (user) | 		// Increment notes count (user) | ||||||
| 		this.incNotesCountOfUser(user); | 		this.incNotesCountOfUser(user); | ||||||
|  |  | ||||||
| 		// Word mute | 		if (data.visibility === 'public' || data.visibility === 'home') { | ||||||
| 		mutedWordsCache.fetch(() => this.userProfilesRepository.find({ | 			this.pushToTl(note, user); | ||||||
| 			where: { | 		} else if (data.visibility === 'followers') { | ||||||
| 				enableWordMute: true, | 			this.pushToTl(note, user); | ||||||
| 			}, | 		} else if (data.visibility === 'specified') { | ||||||
| 			select: ['userId', 'mutedWords'], | 			// TODO | ||||||
| 		})).then(us => { | 		} | ||||||
| 			for (const u of us) { |  | ||||||
| 				checkWordMute(note, { id: u.userId }, u.mutedWords).then(shouldMute => { |  | ||||||
| 					if (shouldMute) { |  | ||||||
| 						this.mutedNotesRepository.insert({ |  | ||||||
| 							id: this.idService.genId(), |  | ||||||
| 							userId: u.userId, |  | ||||||
| 							noteId: note.id, |  | ||||||
| 							reason: 'word', |  | ||||||
| 						}); |  | ||||||
| 					} |  | ||||||
| 				}); |  | ||||||
| 			} |  | ||||||
| 		}); |  | ||||||
|  |  | ||||||
| 		this.antennaService.addNoteToAntennas(note, user); | 		this.antennaService.addNoteToAntennas(note, user); | ||||||
|  |  | ||||||
| @@ -505,6 +495,21 @@ export class NoteCreateService implements OnApplicationShutdown { | |||||||
| 			this.saveReply(data.reply, note); | 			this.saveReply(data.reply, note); | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
|  | 		if (data.reply == null) { | ||||||
|  | 			// TODO: キャッシュ | ||||||
|  | 			this.followingsRepository.findBy({ | ||||||
|  | 				followeeId: user.id, | ||||||
|  | 				notify: 'normal', | ||||||
|  | 			}).then(followings => { | ||||||
|  | 				for (const following of followings) { | ||||||
|  | 					// TODO: ワードミュート考慮 | ||||||
|  | 					this.notificationService.createNotification(following.followerId, 'note', { | ||||||
|  | 						noteId: note.id, | ||||||
|  | 					}, user.id); | ||||||
|  | 				} | ||||||
|  | 			}); | ||||||
|  | 		} | ||||||
|  |  | ||||||
| 		// この投稿を除く指定したユーザーによる指定したノートのリノートが存在しないとき | 		// この投稿を除く指定したユーザーによる指定したノートのリノートが存在しないとき | ||||||
| 		if (data.renote && (await this.noteEntityService.countSameRenotes(user.id, data.renote.id, note.id) === 0)) { | 		if (data.renote && (await this.noteEntityService.countSameRenotes(user.id, data.renote.id, note.id) === 0)) { | ||||||
| 			if (!user.isBot) this.incRenoteCount(data.renote); | 			if (!user.isBot) this.incRenoteCount(data.renote); | ||||||
| @@ -796,6 +801,205 @@ export class NoteCreateService implements OnApplicationShutdown { | |||||||
| 		return mentionedUsers; | 		return mentionedUsers; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	@bindThis | ||||||
|  | 	private async pushToTl(note: MiNote, user: { id: MiUser['id']; host: MiUser['host']; }) { | ||||||
|  | 		const redisPipeline = this.redisForTimelines.pipeline(); | ||||||
|  |  | ||||||
|  | 		if (note.channelId) { | ||||||
|  | 			const channelFollowings = await this.channelFollowingsRepository.find({ | ||||||
|  | 				where: { | ||||||
|  | 					followeeId: note.channelId, | ||||||
|  | 				}, | ||||||
|  | 				select: ['followerId'], | ||||||
|  | 			}); | ||||||
|  |  | ||||||
|  | 			for (const channelFollowing of channelFollowings) { | ||||||
|  | 				redisPipeline.xadd( | ||||||
|  | 					`homeTimeline:${channelFollowing.followerId}`, | ||||||
|  | 					'MAXLEN', '~', '200', | ||||||
|  | 					'*', | ||||||
|  | 					'note', note.id); | ||||||
|  |  | ||||||
|  | 				if (note.fileIds.length > 0) { | ||||||
|  | 					redisPipeline.xadd( | ||||||
|  | 						`homeTimelineWithFiles:${channelFollowing.followerId}`, | ||||||
|  | 						'MAXLEN', '~', '100', | ||||||
|  | 						'*', | ||||||
|  | 						'note', note.id); | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 		} else { | ||||||
|  | 			// TODO: キャッシュ? | ||||||
|  | 			const followings = await this.followingsRepository.find({ | ||||||
|  | 				where: { | ||||||
|  | 					followeeId: user.id, | ||||||
|  | 					followerHost: IsNull(), | ||||||
|  | 					isFollowerHibernated: false, | ||||||
|  | 				}, | ||||||
|  | 				select: ['followerId', 'withReplies'], | ||||||
|  | 			}); | ||||||
|  |  | ||||||
|  | 			const userListMemberships = await this.userListMembershipsRepository.find({ | ||||||
|  | 				where: { | ||||||
|  | 					userId: user.id, | ||||||
|  | 				}, | ||||||
|  | 				select: ['userListId', 'withReplies'], | ||||||
|  | 			}); | ||||||
|  |  | ||||||
|  | 			// TODO: あまりにも数が多いと redisPipeline.exec に失敗する(理由は不明)ため、3万件程度を目安に分割して実行するようにする | ||||||
|  | 			for (const following of followings) { | ||||||
|  | 				// 自分自身以外への返信 | ||||||
|  | 				if (note.replyId && note.replyUserId !== note.userId) { | ||||||
|  | 					if (!following.withReplies) continue; | ||||||
|  | 				} | ||||||
|  |  | ||||||
|  | 				redisPipeline.xadd( | ||||||
|  | 					`homeTimeline:${following.followerId}`, | ||||||
|  | 					'MAXLEN', '~', '200', | ||||||
|  | 					'*', | ||||||
|  | 					'note', note.id); | ||||||
|  |  | ||||||
|  | 				if (note.fileIds.length > 0) { | ||||||
|  | 					redisPipeline.xadd( | ||||||
|  | 						`homeTimelineWithFiles:${following.followerId}`, | ||||||
|  | 						'MAXLEN', '~', '100', | ||||||
|  | 						'*', | ||||||
|  | 						'note', note.id); | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			// TODO | ||||||
|  | 			//if (note.visibility === 'followers') { | ||||||
|  | 			//	// TODO: 重そうだから何とかしたい Set 使う? | ||||||
|  | 			//	userLists = userLists.filter(x => followings.some(f => f.followerId === x.userListUserId)); | ||||||
|  | 			//} | ||||||
|  |  | ||||||
|  | 			for (const userListMembership of userListMemberships) { | ||||||
|  | 				// 自分自身以外への返信 | ||||||
|  | 				if (note.replyId && note.replyUserId !== note.userId) { | ||||||
|  | 					if (!userListMembership.withReplies) continue; | ||||||
|  | 				} | ||||||
|  |  | ||||||
|  | 				redisPipeline.xadd( | ||||||
|  | 					`userListTimeline:${userListMembership.userListId}`, | ||||||
|  | 					'MAXLEN', '~', '200', | ||||||
|  | 					'*', | ||||||
|  | 					'note', note.id); | ||||||
|  |  | ||||||
|  | 				if (note.fileIds.length > 0) { | ||||||
|  | 					redisPipeline.xadd( | ||||||
|  | 						`userListTimelineWithFiles:${userListMembership.userListId}`, | ||||||
|  | 						'MAXLEN', '~', '100', | ||||||
|  | 						'*', | ||||||
|  | 						'note', note.id); | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			{ // 自分自身のHTL | ||||||
|  | 				redisPipeline.xadd( | ||||||
|  | 					`homeTimeline:${user.id}`, | ||||||
|  | 					'MAXLEN', '~', '200', | ||||||
|  | 					'*', | ||||||
|  | 					'note', note.id); | ||||||
|  |  | ||||||
|  | 				if (note.fileIds.length > 0) { | ||||||
|  | 					redisPipeline.xadd( | ||||||
|  | 						`homeTimelineWithFiles:${user.id}`, | ||||||
|  | 						'MAXLEN', '~', '100', | ||||||
|  | 						'*', | ||||||
|  | 						'note', note.id); | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			if (note.visibility === 'public' || note.visibility === 'home') { | ||||||
|  | 				// 自分自身以外への返信 | ||||||
|  | 				if (note.replyId && note.replyUserId !== note.userId) { | ||||||
|  | 					redisPipeline.xadd( | ||||||
|  | 						`userTimelineWithReplies:${user.id}`, | ||||||
|  | 						'MAXLEN', '~', '1000', | ||||||
|  | 						'*', | ||||||
|  | 						'note', note.id); | ||||||
|  | 				} else { | ||||||
|  | 					redisPipeline.xadd( | ||||||
|  | 						`userTimeline:${user.id}`, | ||||||
|  | 						'MAXLEN', '~', '1000', | ||||||
|  | 						'*', | ||||||
|  | 						'note', note.id); | ||||||
|  |  | ||||||
|  | 					if (note.fileIds.length > 0) { | ||||||
|  | 						redisPipeline.xadd( | ||||||
|  | 							`userTimelineWithFiles:${user.id}`, | ||||||
|  | 							'MAXLEN', '~', '500', | ||||||
|  | 							'*', | ||||||
|  | 							'note', note.id); | ||||||
|  | 					} | ||||||
|  |  | ||||||
|  | 					if (note.visibility === 'public' && note.userHost == null) { | ||||||
|  | 						redisPipeline.xadd( | ||||||
|  | 							'localTimeline', | ||||||
|  | 							'MAXLEN', '~', '1000', | ||||||
|  | 							'*', | ||||||
|  | 							'note', note.id); | ||||||
|  |  | ||||||
|  | 						if (note.fileIds.length > 0) { | ||||||
|  | 							redisPipeline.xadd( | ||||||
|  | 								'localTimelineWithFiles', | ||||||
|  | 								'MAXLEN', '~', '500', | ||||||
|  | 								'*', | ||||||
|  | 								'note', note.id); | ||||||
|  | 						} | ||||||
|  | 					} | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			if (Math.random() < 0.1) { | ||||||
|  | 				process.nextTick(() => { | ||||||
|  | 					this.checkHibernation(followings); | ||||||
|  | 				}); | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		redisPipeline.exec(); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	@bindThis | ||||||
|  | 	public async checkHibernation(followings: MiFollowing[]) { | ||||||
|  | 		if (followings.length === 0) return; | ||||||
|  |  | ||||||
|  | 		const shuffle = (array: MiFollowing[]) => { | ||||||
|  | 			for (let i = array.length - 1; i > 0; i--) { | ||||||
|  | 				const j = Math.floor(Math.random() * (i + 1)); | ||||||
|  | 				[array[i], array[j]] = [array[j], array[i]]; | ||||||
|  | 			} | ||||||
|  | 			return array; | ||||||
|  | 		}; | ||||||
|  |  | ||||||
|  | 		// ランダムに最大1000件サンプリング | ||||||
|  | 		const samples = shuffle(followings).slice(0, Math.min(followings.length, 1000)); | ||||||
|  |  | ||||||
|  | 		const hibernatedUsers = await this.usersRepository.find({ | ||||||
|  | 			where: { | ||||||
|  | 				id: In(samples.map(x => x.followerId)), | ||||||
|  | 				lastActiveDate: LessThan(new Date(Date.now() - (1000 * 60 * 60 * 24 * 50))), | ||||||
|  | 			}, | ||||||
|  | 			select: ['id'], | ||||||
|  | 		}); | ||||||
|  |  | ||||||
|  | 		if (hibernatedUsers.length > 0) { | ||||||
|  | 			this.usersRepository.update({ | ||||||
|  | 				id: In(hibernatedUsers.map(x => x.id)), | ||||||
|  | 			}, { | ||||||
|  | 				isHibernated: true, | ||||||
|  | 			}); | ||||||
|  |  | ||||||
|  | 			this.followingsRepository.update({ | ||||||
|  | 				followerId: In(hibernatedUsers.map(x => x.id)), | ||||||
|  | 			}, { | ||||||
|  | 				isFollowerHibernated: true, | ||||||
|  | 			}); | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public dispose(): void { | 	public dispose(): void { | ||||||
| 		this.#shutdownController.abort(); | 		this.#shutdownController.abort(); | ||||||
|   | |||||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user