Compare commits
	
		
			216 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | c5ef6bf38a | ||
|   | b427bf70a8 | ||
|   | c75fc266e9 | ||
|   | e98740c285 | ||
|   | 56b23a64a3 | ||
|   | ee5b417354 | ||
|   | 2f48d109dd | ||
|   | 784fc7b3f5 | ||
|   | b55d26387b | ||
|   | 9ddf62d8b7 | ||
|   | a8feed1eff | ||
|   | f5bfc6f0c1 | ||
|   | ee03ab8d2c | ||
|   | 3c7e1ff92e | ||
|   | ac7e2ecb59 | ||
|   | f28aea9e30 | ||
|   | 1ac7c154d7 | ||
|   | ef860a8f84 | ||
|   | f6f269194f | ||
|   | e1d41063cd | ||
|   | 5d02405a98 | ||
|   | 998c2b692a | ||
|   | 19c0027605 | ||
|   | 9349f72227 | ||
|   | 5af8b77d28 | ||
|   | f74d9c7ed0 | ||
|   | 7d9c273dac | ||
|   | f2da79ad43 | ||
|   | 73a1372940 | ||
|   | 0138c3b00e | ||
|   | 6f33be6c75 | ||
|   | 3004fe573d | ||
|   | 040f9927dd | ||
|   | abc1bdf218 | ||
|   | e73e56be8f | ||
|   | b0616b52ea | ||
|   | 6b6b767199 | ||
|   | e1bdecb9c1 | ||
|   | a32c6267be | ||
|   | 54df243b90 | ||
|   | b44597d5d8 | ||
|   | 7b70b6c3cd | ||
|   | 5cc0219ff2 | ||
|   | 4a0b0b135a | ||
|   | 8bd2d6328a | ||
|   | 9351fb9617 | ||
|   | 7b29e36d64 | ||
|   | 9cc36ef32d | ||
|   | 000f876084 | ||
|   | 2d11c558fa | ||
|   | ac6b02af40 | ||
|   | 7d91912cfd | ||
|   | 3c504b4b08 | ||
|   | adad4bcfe3 | ||
|   | b3e8671dd9 | ||
|   | 0f8c890761 | ||
|   | 512e451f24 | ||
|   | ca0d53ec5d | ||
|   | 686a709e87 | ||
|   | 83fb629f0b | ||
|   | 35eeeb25e3 | ||
|   | 19035c676c | ||
|   | 61ffe7417c | ||
|   | 7651353f39 | ||
|   | 3f5b81060f | ||
|   | 63dc66769f | ||
|   | e0fc8cbf8f | ||
|   | f9d1bc340e | ||
|   | 0b269e79fd | ||
|   | 6159cfd138 | ||
|   | 6a5bbd335b | ||
|   | 39e269db8c | ||
|   | 70fe23a3ce | ||
|   | a6a8a7fb85 | ||
|   | 6641b13b4c | ||
|   | 5136b05c9b | ||
|   | 803c2144f4 | ||
|   | b69a079514 | ||
|   | 2aa800cd55 | ||
|   | 6e61a36d05 | ||
|   | f80bf1fb1c | ||
|   | d465e85239 | ||
|   | deed25a2ff | ||
|   | a486716520 | ||
|   | 2361e11e98 | ||
|   | cd1f2adca7 | ||
|   | a558767b7a | ||
|   | 399ce9b999 | ||
|   | a94a0b5b0b | ||
|   | 76faec2115 | ||
|   | 33c4e57994 | ||
|   | bc23496998 | ||
|   | d35ad95c18 | ||
|   | 5facd11592 | ||
|   | e1e885d6b2 | ||
|   | 5b6695114f | ||
|   | 71dd7f89e9 | ||
|   | 21331e53fe | ||
|   | 7afee5977f | ||
|   | d195b0dec7 | ||
|   | 8a95e850ad | ||
|   | a4d74d7d7e | ||
|   | 256e0db36d | ||
|   | d593c1358a | ||
|   | 1ff14d81c1 | ||
|   | 4369d12eec | ||
|   | 91cc033eb5 | ||
|   | 57543e6b44 | ||
|   | a1b8cd15c4 | ||
|   | 73f06e591a | ||
|   | 6f7cfa82b5 | ||
|   | ff97a003d1 | ||
|   | 53c92e3e23 | ||
|   | 13d13bc2f6 | ||
|   | 03744a25ed | ||
|   | eac3bf8bff | ||
|   | 2e1fbb5b16 | ||
|   | 98b3517d36 | ||
|   | dee662705e | ||
|   | 0da0cc80b9 | ||
|   | 650187deaf | ||
|   | 2e565cac2c | ||
|   | ac7537278c | ||
|   | f9a2e98831 | ||
|   | 54f789bd55 | ||
|   | 5ac9d13516 | ||
|   | 2be1a39d13 | ||
|   | f3c5edc852 | ||
|   | 30704e6de8 | ||
|   | 41932ac409 | ||
|   | 9843c596d8 | ||
|   | baf65bfa69 | ||
|   | 6501f80fc7 | ||
|   | b037f6566b | ||
|   | 0ec8ebeba3 | ||
|   | af1c9251fc | ||
|   | 4ad399c593 | ||
|   | 55a9646f23 | ||
|   | 46017f5725 | ||
|   | c20ce12f86 | ||
|   | 1e28db2396 | ||
|   | 5f3640c7fd | ||
|   | d65e5f6794 | ||
|   | e67d7bc0ea | ||
|   | 1139632f95 | ||
|   | b51a8c3f82 | ||
|   | 0d7256678e | ||
|   | eea33d07fd | ||
|   | f599337320 | ||
|   | 7df019db0e | ||
|   | 04f92bd688 | ||
|   | 505ecf6c1f | ||
|   | c9ec08704e | ||
|   | 6a3039f7b7 | ||
|   | 868c8fffb3 | ||
|   | faed3b438e | ||
|   | 6c982629ea | ||
|   | 110bbbc7dc | ||
|   | 4ad0345f20 | ||
|   | 9d84214462 | ||
|   | 3f199c7113 | ||
|   | e9417fb741 | ||
|   | ee74df6823 | ||
|   | 26630bae81 | ||
|   | 9bde9edcf6 | ||
|   | a12f07c42b | ||
|   | e7334c4fb0 | ||
|   | 38f9d1e764 | ||
|   | 2dfed75402 | ||
|   | 0c12e80106 | ||
|   | b7522f69e7 | ||
|   | 24705a7e39 | ||
|   | 8add8025a0 | ||
|   | 32fa79d928 | ||
|   | 534be6ff25 | ||
|   | f684c07567 | ||
|   | 788ae2f6ca | ||
|   | 572000f868 | ||
|   | 57f5df2d22 | ||
|   | b2a67ba5ca | ||
|   | d78e15cc1a | ||
|   | ceab34f5f3 | ||
|   | 3a62625bbc | ||
|   | ad6844ac4a | ||
|   | a8c252a613 | ||
|   | 1d39f785f1 | ||
|   | 4b8b29b862 | ||
|   | 0d148bd23b | ||
|   | ebedb81e3f | ||
|   | d195406fdc | ||
|   | 5173ed37f9 | ||
|   | 825551d64f | ||
|   | 449761bada | ||
|   | 5859df389f | ||
|   | 562b02310f | ||
|   | 65ed702d87 | ||
|   | c559a9843f | ||
|   | 88c3957085 | ||
|   | 01778e11dc | ||
|   | 9d9e8a3c4e | ||
|   | ed3e035ad6 | ||
|   | 07f885fea8 | ||
|   | 2cc98226ca | ||
|   | 8a6f73c5ff | ||
|   | 00e3453ce1 | ||
|   | 16646dd77a | ||
|   | 1f39d1fe26 | ||
|   | e8f3c587c9 | ||
|   | 4b43745e7c | ||
|   | 9db2f60053 | ||
|   | 4610d8dfe3 | ||
|   | fa296efdf6 | ||
|   | d9d98f84bf | ||
|   | 7c3143b8e5 | ||
|   | 387fcd5c5d | ||
|   | ebc6437977 | 
| @@ -114,11 +114,6 @@ id: 'aid' | |||||||
| # IP address family used for outgoing request (ipv4, ipv6 or dual) | # IP address family used for outgoing request (ipv4, ipv6 or dual) | ||||||
| #outgoingAddressFamily: ipv4 | #outgoingAddressFamily: ipv4 | ||||||
|  |  | ||||||
| # Syslog option |  | ||||||
| #syslog: |  | ||||||
| #  host: localhost |  | ||||||
| #  port: 514 |  | ||||||
|  |  | ||||||
| # Proxy for HTTP/HTTPS | # Proxy for HTTP/HTTPS | ||||||
| #proxy: http://127.0.0.1:3128 | #proxy: http://127.0.0.1:3128 | ||||||
|  |  | ||||||
|   | |||||||
| @@ -114,11 +114,6 @@ id: 'aid' | |||||||
| # IP address family used for outgoing request (ipv4, ipv6 or dual) | # IP address family used for outgoing request (ipv4, ipv6 or dual) | ||||||
| #outgoingAddressFamily: ipv4 | #outgoingAddressFamily: ipv4 | ||||||
|  |  | ||||||
| # Syslog option |  | ||||||
| #syslog: |  | ||||||
| #  host: localhost |  | ||||||
| #  port: 514 |  | ||||||
|  |  | ||||||
| # Proxy for HTTP/HTTPS | # Proxy for HTTP/HTTPS | ||||||
| #proxy: http://127.0.0.1:3128 | #proxy: http://127.0.0.1:3128 | ||||||
|  |  | ||||||
| @@ -135,11 +130,21 @@ proxyBypassHosts: | |||||||
| #proxySmtp: socks5://127.0.0.1:1080 # use SOCKS5 | #proxySmtp: socks5://127.0.0.1:1080 # use SOCKS5 | ||||||
|  |  | ||||||
| # Media Proxy | # Media Proxy | ||||||
|  | # Reference Implementation: https://github.com/misskey-dev/media-proxy | ||||||
|  | # * Deliver a common cache between instances | ||||||
|  | # * Perform image compression (on a different server resource than the main process) | ||||||
| #mediaProxy: https://example.com/proxy | #mediaProxy: https://example.com/proxy | ||||||
|  |  | ||||||
| # Proxy remote files (default: false) | # Proxy remote files (default: false) | ||||||
|  | # Proxy remote files by this instance or mediaProxy to prevent remote files from running in remote domains. | ||||||
| #proxyRemoteFiles: true | #proxyRemoteFiles: true | ||||||
|  |  | ||||||
|  | # Movie Thumbnail Generation URL | ||||||
|  | # There is no reference implementation. | ||||||
|  | # For example, Misskey will point to the following URL: | ||||||
|  | #   https://example.com/thumbnail.webp?thumbnail=1&url=https%3A%2F%2Fstorage.example.com%2Fpath%2Fto%2Fvideo.mp4 | ||||||
|  | #videoThumbnailGenerator: https://example.com | ||||||
|  |  | ||||||
| # Sign to ActivityPub GET request (default: true) | # Sign to ActivityPub GET request (default: true) | ||||||
| signToActivityPubGet: true | signToActivityPubGet: true | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										1
									
								
								.devcontainer/Dockerfile
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								.devcontainer/Dockerfile
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | |||||||
|  | FROM mcr.microsoft.com/devcontainers/javascript-node:0-18 | ||||||
							
								
								
									
										11
									
								
								.devcontainer/devcontainer.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								.devcontainer/devcontainer.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,11 @@ | |||||||
|  | { | ||||||
|  | 	"name": "Misskey", | ||||||
|  | 	"dockerComposeFile": "docker-compose.yml", | ||||||
|  | 	"service": "app", | ||||||
|  | 	"workspaceFolder": "/workspaces/${localWorkspaceFolderBasename}", | ||||||
|  | 	"features": { | ||||||
|  | 		"ghcr.io/devcontainers-contrib/features/pnpm:2": {} | ||||||
|  | 	}, | ||||||
|  | 	"forwardPorts": [3000], | ||||||
|  | 	"postCreateCommand": ".devcontainer/init.sh" | ||||||
|  | } | ||||||
							
								
								
									
										146
									
								
								.devcontainer/devcontainer.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										146
									
								
								.devcontainer/devcontainer.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,146 @@ | |||||||
|  | #━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━ | ||||||
|  | # Misskey configuration | ||||||
|  | #━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━ | ||||||
|  |  | ||||||
|  | #   ┌─────┐ | ||||||
|  | #───┘ URL └───────────────────────────────────────────────────── | ||||||
|  |  | ||||||
|  | # Final accessible URL seen by a user. | ||||||
|  | url: http://127.0.0.1:3000/ | ||||||
|  |  | ||||||
|  | # ONCE YOU HAVE STARTED THE INSTANCE, DO NOT CHANGE THE | ||||||
|  | # URL SETTINGS AFTER THAT! | ||||||
|  |  | ||||||
|  | #   ┌───────────────────────┐ | ||||||
|  | #───┘ Port and TLS settings └─────────────────────────────────── | ||||||
|  |  | ||||||
|  | # | ||||||
|  | # Misskey requires a reverse proxy to support HTTPS connections. | ||||||
|  | # | ||||||
|  | #                 +----- https://example.tld/ ------------+ | ||||||
|  | #   +------+      |+-------------+      +----------------+| | ||||||
|  | #   | User | ---> || Proxy (443) | ---> | Misskey (3000) || | ||||||
|  | #   +------+      |+-------------+      +----------------+| | ||||||
|  | #                 +---------------------------------------+ | ||||||
|  | # | ||||||
|  | #   You need to set up a reverse proxy. (e.g. nginx) | ||||||
|  | #   An encrypted connection with HTTPS is highly recommended | ||||||
|  | #   because tokens may be transferred in GET requests. | ||||||
|  |  | ||||||
|  | # The port that your Misskey server should listen on. | ||||||
|  | port: 3000 | ||||||
|  |  | ||||||
|  | #   ┌──────────────────────────┐ | ||||||
|  | #───┘ PostgreSQL configuration └──────────────────────────────── | ||||||
|  |  | ||||||
|  | db: | ||||||
|  |   host: db | ||||||
|  |   port: 5432 | ||||||
|  |  | ||||||
|  |   # Database name | ||||||
|  |   db: misskey | ||||||
|  |  | ||||||
|  |   # Auth | ||||||
|  |   user: postgres | ||||||
|  |   pass: postgres | ||||||
|  |  | ||||||
|  |   # Whether disable Caching queries | ||||||
|  |   #disableCache: true | ||||||
|  |  | ||||||
|  |   # Extra Connection options | ||||||
|  |   #extra: | ||||||
|  |   #  ssl: true | ||||||
|  |  | ||||||
|  | #   ┌─────────────────────┐ | ||||||
|  | #───┘ Redis configuration └───────────────────────────────────── | ||||||
|  |  | ||||||
|  | redis: | ||||||
|  |   host: redis | ||||||
|  |   port: 6379 | ||||||
|  |   #family: 0  # 0=Both, 4=IPv4, 6=IPv6 | ||||||
|  |   #pass: example-pass | ||||||
|  |   #prefix: example-prefix | ||||||
|  |   #db: 1 | ||||||
|  |  | ||||||
|  | #   ┌─────────────────────────────┐ | ||||||
|  | #───┘ Elasticsearch configuration └───────────────────────────── | ||||||
|  |  | ||||||
|  | #elasticsearch: | ||||||
|  | #  host: localhost | ||||||
|  | #  port: 9200 | ||||||
|  | #  ssl: false | ||||||
|  | #  user: | ||||||
|  | #  pass: | ||||||
|  |  | ||||||
|  | #   ┌───────────────┐ | ||||||
|  | #───┘ ID generation └─────────────────────────────────────────── | ||||||
|  |  | ||||||
|  | # You can select the ID generation method. | ||||||
|  | # You don't usually need to change this setting, but you can | ||||||
|  | # change it according to your preferences. | ||||||
|  |  | ||||||
|  | # Available methods: | ||||||
|  | # aid ... Short, Millisecond accuracy | ||||||
|  | # meid ... Similar to ObjectID, Millisecond accuracy | ||||||
|  | # ulid ... Millisecond accuracy | ||||||
|  | # objectid ... This is left for backward compatibility | ||||||
|  |  | ||||||
|  | # ONCE YOU HAVE STARTED THE INSTANCE, DO NOT CHANGE THE | ||||||
|  | # ID SETTINGS AFTER THAT! | ||||||
|  |  | ||||||
|  | id: 'aid' | ||||||
|  |  | ||||||
|  | #   ┌─────────────────────┐ | ||||||
|  | #───┘ Other configuration └───────────────────────────────────── | ||||||
|  |  | ||||||
|  | # Whether disable HSTS | ||||||
|  | #disableHsts: true | ||||||
|  |  | ||||||
|  | # Number of worker processes | ||||||
|  | #clusterLimit: 1 | ||||||
|  |  | ||||||
|  | # Job concurrency per worker | ||||||
|  | # deliverJobConcurrency: 128 | ||||||
|  | # inboxJobConcurrency: 16 | ||||||
|  |  | ||||||
|  | # Job rate limiter | ||||||
|  | # deliverJobPerSec: 128 | ||||||
|  | # inboxJobPerSec: 16 | ||||||
|  |  | ||||||
|  | # Job attempts | ||||||
|  | # deliverJobMaxAttempts: 12 | ||||||
|  | # inboxJobMaxAttempts: 8 | ||||||
|  |  | ||||||
|  | # IP address family used for outgoing request (ipv4, ipv6 or dual) | ||||||
|  | #outgoingAddressFamily: ipv4 | ||||||
|  |  | ||||||
|  | # Proxy for HTTP/HTTPS | ||||||
|  | #proxy: http://127.0.0.1:3128 | ||||||
|  |  | ||||||
|  | proxyBypassHosts: | ||||||
|  |   - api.deepl.com | ||||||
|  |   - api-free.deepl.com | ||||||
|  |   - www.recaptcha.net | ||||||
|  |   - hcaptcha.com | ||||||
|  |   - challenges.cloudflare.com | ||||||
|  |  | ||||||
|  | # Proxy for SMTP/SMTPS | ||||||
|  | #proxySmtp: http://127.0.0.1:3128   # use HTTP/1.1 CONNECT | ||||||
|  | #proxySmtp: socks4://127.0.0.1:1080 # use SOCKS4 | ||||||
|  | #proxySmtp: socks5://127.0.0.1:1080 # use SOCKS5 | ||||||
|  |  | ||||||
|  | # Media Proxy | ||||||
|  | #mediaProxy: https://example.com/proxy | ||||||
|  |  | ||||||
|  | # Proxy remote files (default: false) | ||||||
|  | #proxyRemoteFiles: true | ||||||
|  |  | ||||||
|  | # Sign to ActivityPub GET request (default: true) | ||||||
|  | signToActivityPubGet: true | ||||||
|  |  | ||||||
|  | allowedPrivateNetworks: [ | ||||||
|  |   '127.0.0.1/32' | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | # Upload or download file size limits (bytes) | ||||||
|  | #maxFileSize: 262144000 | ||||||
							
								
								
									
										52
									
								
								.devcontainer/docker-compose.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										52
									
								
								.devcontainer/docker-compose.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,52 @@ | |||||||
|  | version: '3.8' | ||||||
|  |  | ||||||
|  | services: | ||||||
|  |   app: | ||||||
|  |     build:  | ||||||
|  |       context: . | ||||||
|  |       dockerfile: Dockerfile | ||||||
|  |  | ||||||
|  |     volumes: | ||||||
|  |       - ../..:/workspaces:cached | ||||||
|  |  | ||||||
|  |     command: sleep infinity | ||||||
|  |  | ||||||
|  |     networks: | ||||||
|  |       - internal_network | ||||||
|  |       - external_network | ||||||
|  |  | ||||||
|  |   redis: | ||||||
|  |     restart: always | ||||||
|  |     image: redis:7-alpine | ||||||
|  |     networks: | ||||||
|  |       - internal_network | ||||||
|  |     volumes: | ||||||
|  |       - ../redis:/data | ||||||
|  |     healthcheck: | ||||||
|  |       test: "redis-cli ping" | ||||||
|  |       interval: 5s | ||||||
|  |       retries: 20 | ||||||
|  |  | ||||||
|  |   db: | ||||||
|  |     restart: unless-stopped | ||||||
|  |     image: postgres:15-alpine | ||||||
|  |     networks: | ||||||
|  |       - internal_network | ||||||
|  |     environment: | ||||||
|  |       POSTGRES_USER: postgres | ||||||
|  |       POSTGRES_PASSWORD: postgres | ||||||
|  |       POSTGRES_DB: misskey | ||||||
|  |     volumes: | ||||||
|  |       - ../db:/var/lib/postgresql/data | ||||||
|  |     healthcheck: | ||||||
|  |       test: "pg_isready -U $$POSTGRES_USER -d $$POSTGRES_DB" | ||||||
|  |       interval: 5s | ||||||
|  |       retries: 20 | ||||||
|  |  | ||||||
|  | volumes: | ||||||
|  |   postgres-data: | ||||||
|  |  | ||||||
|  | networks: | ||||||
|  |   internal_network: | ||||||
|  |     internal: true | ||||||
|  |   external_network: | ||||||
							
								
								
									
										9
									
								
								.devcontainer/init.sh
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										9
									
								
								.devcontainer/init.sh
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,9 @@ | |||||||
|  | #!/bin/bash | ||||||
|  |  | ||||||
|  | set -xe | ||||||
|  |  | ||||||
|  | git submodule update --init | ||||||
|  | pnpm install --frozen-lockfile | ||||||
|  | cp .devcontainer/devcontainer.yml .config/default.yml | ||||||
|  | pnpm build | ||||||
|  | pnpm migrate | ||||||
| @@ -16,9 +16,15 @@ files/ | |||||||
| misskey-assets/ | misskey-assets/ | ||||||
| fluent-emojis/ | fluent-emojis/ | ||||||
| .pnp.* | .pnp.* | ||||||
|  |  | ||||||
|  | # .yarn関連 | ||||||
| .yarn/* | .yarn/* | ||||||
| !.yarn/patches | !.yarn/patches | ||||||
| !.yarn/plugins | !.yarn/plugins | ||||||
| !.yarn/releases | !.yarn/releases | ||||||
| !.yarn/sdks | !.yarn/sdks | ||||||
| !.yarn/versions | !.yarn/versions | ||||||
|  |  | ||||||
|  | .idea/ | ||||||
|  | packages/*/.vscode/ | ||||||
|  | packages/backend/test/docker-compose.yml | ||||||
|   | |||||||
							
								
								
									
										3
									
								
								.dockleignore
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								.dockleignore
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | |||||||
|  | DKL-DI-0005 | ||||||
|  | DKL-DI-0006 | ||||||
|  | DKL-LI-0003 | ||||||
							
								
								
									
										2
									
								
								.github/workflows/docker-develop.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/docker-develop.yml
									
									
									
									
										vendored
									
									
								
							| @@ -14,6 +14,8 @@ jobs: | |||||||
|     steps: |     steps: | ||||||
|       - name: Check out the repo |       - name: Check out the repo | ||||||
|         uses: actions/checkout@v3.3.0 |         uses: actions/checkout@v3.3.0 | ||||||
|  |       - name: Set up Docker Buildx | ||||||
|  |         uses: docker/setup-buildx-action@v2.3.0 | ||||||
|       - name: Docker meta |       - name: Docker meta | ||||||
|         id: meta |         id: meta | ||||||
|         uses: docker/metadata-action@v4 |         uses: docker/metadata-action@v4 | ||||||
|   | |||||||
							
								
								
									
										30
									
								
								.github/workflows/dockle.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										30
									
								
								.github/workflows/dockle.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,30 @@ | |||||||
|  | --- | ||||||
|  | name: Dockle | ||||||
|  |  | ||||||
|  | on: | ||||||
|  |   push: | ||||||
|  |     branches: | ||||||
|  |       - master | ||||||
|  |       - develop | ||||||
|  |   pull_request: | ||||||
|  |  | ||||||
|  | jobs: | ||||||
|  |   dockle: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     env: | ||||||
|  |       DOCKER_CONTENT_TRUST: 1 | ||||||
|  |     steps: | ||||||
|  |       - uses: actions/checkout@v3.2.0 | ||||||
|  |       - run: | | ||||||
|  |           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 | ||||||
|  |       - run: | | ||||||
|  |           cp .config/docker_example.env .config/docker.env | ||||||
|  |           cp ./docker-compose.yml.example ./docker-compose.yml | ||||||
|  |       - run: | | ||||||
|  |           docker compose up -d web | ||||||
|  |           docker tag "$(docker compose images web | awk 'OFS=":" {print $4}' | tail -n +2)" misskey-web:latest | ||||||
|  |       - run: | | ||||||
|  |           cmd="dockle --exit-code 1 misskey-web:latest ${image_name}" | ||||||
|  |           echo "> ${cmd}" | ||||||
|  |           eval "${cmd}" | ||||||
							
								
								
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -33,6 +33,7 @@ coverage | |||||||
| !/.config/docker_example.yml | !/.config/docker_example.yml | ||||||
| !/.config/docker_example.env | !/.config/docker_example.env | ||||||
| docker-compose.yml | docker-compose.yml | ||||||
|  | !/.devcontainer/docker-compose.yml | ||||||
|  |  | ||||||
| # misskey | # misskey | ||||||
| /build | /build | ||||||
|   | |||||||
							
								
								
									
										130
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						
									
										130
									
								
								CHANGELOG.md
									
									
									
									
									
								
							| @@ -8,6 +8,136 @@ | |||||||
|  |  | ||||||
| You should also include the user name that made the change. | You should also include the user name that made the change. | ||||||
| --> | --> | ||||||
|  | ## 13.6.1 (2023/02/12) | ||||||
|  |  | ||||||
|  | ### Improvements | ||||||
|  | - アニメーションを少なくする設定の時、MkPageHeaderのタブアニメーションを無効化 | ||||||
|  | - Backend: activitypub情報がcorsでブロックされないようヘッダーを追加 | ||||||
|  | - enhance: レートリミットを0%にできるように | ||||||
|  | - チャンネル内Renoteを行えるように | ||||||
|  |  | ||||||
|  | ### Bugfixes | ||||||
|  | - Client: ユーザーページでアクティビティを見ることができない問題を修正 | ||||||
|  |  | ||||||
|  | ## 13.6.0 (2023/02/11) | ||||||
|  |  | ||||||
|  | ### Improvements | ||||||
|  | - MkPageHeaderをごっそり変えた | ||||||
|  |   * モバイルではヘッダーは上下に分割され、下段にタブが表示されるように | ||||||
|  |   * iconOnlyのタブ項目がアクティブな場合にはタブのタイトルを表示するように | ||||||
|  |   * メインタイムラインではタイトルを表示しない | ||||||
|  |   * メインタイムラインかつモバイルで表示される左上のアバターを選択するとアカウントメニューが開くように | ||||||
|  | - ユーザーページのノート一覧をタブとして分離 | ||||||
|  | - コンディショナルロールもバッジとして表示可能に | ||||||
|  | - enhance(client): ロールをより簡単に付与できるように | ||||||
|  | - enhance(client): 一度見たノートのRenoteは省略して表示するように | ||||||
|  | - enhance(client): 迷惑になる可能性のある投稿を行う前に警告を表示 | ||||||
|  | - リアクションの数が多い場合の表示を改善 | ||||||
|  | - 一部のMFM構文をopt-outに | ||||||
|  |  | ||||||
|  | ### Bugfixes | ||||||
|  | - Client: ユーザーページでタブがほとんど見れないことがないように | ||||||
|  |  | ||||||
|  | ## 13.5.6 (2023/02/10) | ||||||
|  |  | ||||||
|  | ### Improvements | ||||||
|  | - 非ログイン時にMiAuthを踏んだ際にMiAuthであることを表示する | ||||||
|  | - /auth/のUIをアップデート | ||||||
|  | - 利用規約同意UIの調整 | ||||||
|  | - クロップ時の質問を分かりやすく | ||||||
|  |  | ||||||
|  | ### Bugfixes | ||||||
|  | - fix: prevent clipping audio plyr's tooltip | ||||||
|  |  | ||||||
|  | ## 13.5.4 (2023/02/09) | ||||||
|  |  | ||||||
|  | ### Improvements | ||||||
|  | - Server: UIのHTML(ノートなどの特別なページを除く)のキャッシュ時間を15秒から30秒に | ||||||
|  | - i/notificationsのレートリミットを緩和 | ||||||
|  |  | ||||||
|  | ### Bugfixes | ||||||
|  | - fix(client): validate url to improve security | ||||||
|  | - fix(client): dateの初期値が正常に入らない時がある | ||||||
|  |  | ||||||
|  | ## 13.5.3 (2023/02/09) | ||||||
|  |  | ||||||
|  | ### Improvements | ||||||
|  | - Client: デッキにチャンネルカラムを追加 | ||||||
|  |  | ||||||
|  | ## 13.5.2 (2023/02/08) | ||||||
|  |  | ||||||
|  | ### Changes | ||||||
|  | - Revert: perf(client): do not render custom emojis in user names | ||||||
|  |  | ||||||
|  | ### Bugfixes | ||||||
|  | - Client: register_note_view_interruptor not working | ||||||
|  | - Client: ログイントークンの再生成が出来ない | ||||||
|  |  | ||||||
|  | ## 13.5.0 (2023/02/08) | ||||||
|  |  | ||||||
|  | ### Changes | ||||||
|  | - perf(client): do not render custom emojis in user names | ||||||
|  |  | ||||||
|  | ### Improvements | ||||||
|  | - Client: disableShowingAnimatedImagesのデフォルト値をprefers-reduced-motionにする | ||||||
|  | - enhance(client): tweak medialist style | ||||||
|  |  | ||||||
|  | ### Bugfixes | ||||||
|  | - fix docker health check | ||||||
|  | - Client: MkEmojiPickerでもChromeで検索ダイアログで変換確定するとそのまま検索されてしまうのを修正 | ||||||
|  | - fix(mfm): default degree not used in rotate | ||||||
|  | - fix(server): validate urls from ap to improve security | ||||||
|  |  | ||||||
|  | ## 13.4.0 (2023/02/05) | ||||||
|  |  | ||||||
|  | ### Improvements | ||||||
|  | - ロールにアイコンを設定してユーザー名の横に表示できるように | ||||||
|  | - feat: timeline page for non-login users | ||||||
|  | - 実績の単なるラッキーの獲得確立を調整 | ||||||
|  | - Add Thai language support | ||||||
|  |  | ||||||
|  | ### Bugfixes | ||||||
|  | - fix(server): 自分のノートをお気に入りに登録しても実績解除される問題を修正 | ||||||
|  | - fix(server): clean up file in FileServer | ||||||
|  | - fix(server): Deny UNIX domain socket | ||||||
|  | - fix(server): validate filename and emoji name to improve security | ||||||
|  | - fix(client): validate input response in aiscript | ||||||
|  | - fix(client): add webhook delete button | ||||||
|  | - fix(client): tweak notification style | ||||||
|  | - fix(client): インラインコードを折り返して表示する | ||||||
|  |  | ||||||
|  | ## 13.3.3 (2023/02/04) | ||||||
|  |  | ||||||
|  | ### Bugfixes | ||||||
|  | - Server: improve security | ||||||
|  |  | ||||||
|  | ## 13.3.2 (2023/02/04) | ||||||
|  |  | ||||||
|  | ### Improvements | ||||||
|  | - 外部メディアプロキシへの対応を強化しました   | ||||||
|  |   外部メディアプロキシのFastify実装を作りました   | ||||||
|  |   https://github.com/misskey-dev/media-proxy | ||||||
|  | - Server: improve performance | ||||||
|  |  | ||||||
|  | ### Bugfixes | ||||||
|  | - Client: validate urls to improve security | ||||||
|  |  | ||||||
|  | ## 13.3.1 (2023/02/04) | ||||||
|  |  | ||||||
|  | ### Bugfixes | ||||||
|  | - Client: カスタム絵文字にアニメーション画像を再生しない設定が適用されていない問題を修正 | ||||||
|  | - Client: オートコンプリートでUnicode絵文字がカスタム絵文字として表示されてしまうのを修正 | ||||||
|  | - Client: Fix Vue-plyr CORS issue | ||||||
|  | - Client: validate urls to improve security | ||||||
|  |  | ||||||
|  | ## 13.3.0 (2023/02/03) | ||||||
|  | ### Changes | ||||||
|  | - twitter/github/discord連携機能が削除されました | ||||||
|  | - ハッシュタグごとのチャートが削除されました | ||||||
|  | - syslogのサポートが削除されました | ||||||
|  |  | ||||||
|  | ### Improvements | ||||||
|  | - ロールで広告の非表示が有効になっている場合は最初から広告を非表示にするように | ||||||
|  |  | ||||||
| ## 13.2.6 (2023/02/01) | ## 13.2.6 (2023/02/01) | ||||||
| ### Changes | ### Changes | ||||||
|   | |||||||
| @@ -44,7 +44,7 @@ Thank you for your PR! Before creating a PR, please check the following: | |||||||
| - Check if there are any documents that need to be created or updated due to this change. | - Check if there are any documents that need to be created or updated due to this change. | ||||||
| - If you have added a feature or fixed a bug, please add a test case if possible. | - If you have added a feature or fixed a bug, please add a test case if possible. | ||||||
| - Please make sure that tests and Lint are passed in advance. | - Please make sure that tests and Lint are passed in advance. | ||||||
|   - You can run it with `yarn test` and `yarn lint`. [See more info](#testing) |   - You can run it with `pnpm test` and `pnpm lint`. [See more info](#testing) | ||||||
| - If this PR includes UI changes, please attach a screenshot in the text. | - If this PR includes UI changes, please attach a screenshot in the text. | ||||||
|  |  | ||||||
| Thanks for your cooperation 🤗 | Thanks for your cooperation 🤗 | ||||||
| @@ -102,7 +102,7 @@ If your language is not listed in Crowdin, please open an issue. | |||||||
| During development, it is useful to use the  | During development, it is useful to use the  | ||||||
|  |  | ||||||
| ``` | ``` | ||||||
| yarn dev | pnpm dev | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| command. | command. | ||||||
| @@ -111,8 +111,27 @@ command. | |||||||
| - Vite HMR (just the `vite` command) is available. The behavior may be different from production. | - Vite HMR (just the `vite` command) is available. The behavior may be different from production. | ||||||
| - Service Worker is watched by esbuild. | - Service Worker is watched by esbuild. | ||||||
|  |  | ||||||
|  | ### Dev Container | ||||||
|  | Instead of running `pnpm` locally, you can use Dev Container to set up your development environment. | ||||||
|  | To use Dev Container, open the project directory on VSCode with Dev Containers installed. | ||||||
|  |  | ||||||
|  | It will run the following command automatically inside the container. | ||||||
|  | ``` bash | ||||||
|  | git submodule update --init | ||||||
|  | pnpm install --frozen-lockfile | ||||||
|  | cp .devcontainer/devcontainer.yml .config/default.yml | ||||||
|  | pnpm build | ||||||
|  | pnpm migrate | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | After finishing the migration, run the `pnpm dev` command to start the development server. | ||||||
|  |  | ||||||
|  | ``` bash | ||||||
|  | pnpm dev | ||||||
|  | ``` | ||||||
|  |  | ||||||
| ## Testing | ## Testing | ||||||
| - Test codes are located in [`/test`](/test). | - Test codes are located in [`/packages/backend/test`](/packages/backend/test). | ||||||
|  |  | ||||||
| ### Run test | ### Run test | ||||||
| Create a config file. | Create a config file. | ||||||
| @@ -121,18 +140,18 @@ cp .github/misskey/test.yml .config/ | |||||||
| ``` | ``` | ||||||
| Prepare DB/Redis for testing. | Prepare DB/Redis for testing. | ||||||
| ``` | ``` | ||||||
| docker-compose -f packages/backend/test/docker-compose.yml up | docker compose -f packages/backend/test/docker-compose.yml up | ||||||
| ``` | ``` | ||||||
| Alternatively, prepare an empty (data can be erased) DB and edit `.config/test.yml`.  | Alternatively, prepare an empty (data can be erased) DB and edit `.config/test.yml`.  | ||||||
|  |  | ||||||
| Run all test. | Run all test. | ||||||
| ``` | ``` | ||||||
| yarn test | pnpm test | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| #### Run specify test | #### Run specify test | ||||||
| ``` | ``` | ||||||
| yarn jest -- foo.ts | pnpm jest -- foo.ts | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| ### e2e tests | ### e2e tests | ||||||
| @@ -177,9 +196,9 @@ vue-routerとの最大の違いは、niraxは複数のルーターが存在す | |||||||
| これにより、アプリ内ウィンドウでブラウザとは個別にルーティングすることなどが可能になります。 | これにより、アプリ内ウィンドウでブラウザとは個別にルーティングすることなどが可能になります。 | ||||||
|  |  | ||||||
| ## Notes | ## Notes | ||||||
| ### How to resolve conflictions occurred at yarn.lock? | ### How to resolve conflictions occurred at pnpm-lock.yaml? | ||||||
|  |  | ||||||
| Just execute `yarn` to fix it. | Just execute `pnpm` to fix it. | ||||||
|  |  | ||||||
| ### INSERTするときにはsaveではなくinsertを使用する | ### INSERTするときにはsaveではなくinsertを使用する | ||||||
| #6441 | #6441 | ||||||
| @@ -265,7 +284,7 @@ MongoDBは`null`で返してきてたので、その感覚で`if (x === null)` | |||||||
| ### Migration作成方法 | ### Migration作成方法 | ||||||
| packages/backendで: | packages/backendで: | ||||||
| ```sh | ```sh | ||||||
| yarn dlx typeorm migration:generate -d ormconfig.js -o <migration name> | pnpm dlx typeorm migration:generate -d ormconfig.js -o <migration name> | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| - 生成後、ファイルをmigration下に移してください | - 生成後、ファイルをmigration下に移してください | ||||||
|   | |||||||
| @@ -29,6 +29,7 @@ ARG NODE_ENV=production | |||||||
|  |  | ||||||
| RUN git submodule update --init | RUN git submodule update --init | ||||||
| RUN pnpm build | RUN pnpm build | ||||||
|  | RUN rm -rf .git/ | ||||||
|  |  | ||||||
| FROM node:${NODE_VERSION}-slim AS runner | FROM node:${NODE_VERSION}-slim AS runner | ||||||
|  |  | ||||||
| @@ -41,10 +42,12 @@ RUN --mount=type=cache,target=/var/cache/apt,sharing=locked \ | |||||||
| 	; echo 'Binary::apt::APT::Keep-Downloaded-Packages "true";' > /etc/apt/apt.conf.d/keep-cache \ | 	; echo 'Binary::apt::APT::Keep-Downloaded-Packages "true";' > /etc/apt/apt.conf.d/keep-cache \ | ||||||
| 	&& apt-get update \ | 	&& apt-get update \ | ||||||
| 	&& apt-get install -y --no-install-recommends \ | 	&& apt-get install -y --no-install-recommends \ | ||||||
| 	ffmpeg tini \ | 	ffmpeg tini curl \ | ||||||
| 	&& 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 \ | ||||||
|  | 	&& find / -type f -perm /u+s -ignore_readdir_race -exec chmod u-s {} \; \ | ||||||
|  | 	&& find / -type f -perm /g+s -ignore_readdir_race -exec chmod g-s {} \; | ||||||
|  |  | ||||||
| USER misskey | USER misskey | ||||||
| WORKDIR /misskey | WORKDIR /misskey | ||||||
| @@ -58,5 +61,6 @@ COPY --chown=misskey:misskey --from=builder /misskey/fluent-emojis /misskey/flue | |||||||
| COPY --chown=misskey:misskey . ./ | COPY --chown=misskey:misskey . ./ | ||||||
|  |  | ||||||
| ENV NODE_ENV=production | ENV NODE_ENV=production | ||||||
|  | HEALTHCHECK --interval=5s --retries=20 CMD ["/bin/bash", "/misskey/healthcheck.sh"] | ||||||
| ENTRYPOINT ["/usr/bin/tini", "--"] | ENTRYPOINT ["/usr/bin/tini", "--"] | ||||||
| CMD ["pnpm", "run", "migrateandstart"] | CMD ["pnpm", "run", "migrateandstart"] | ||||||
|   | |||||||
| @@ -24,6 +24,8 @@ | |||||||
| 	 | 	 | ||||||
| --- | --- | ||||||
|  |  | ||||||
|  | [](https://codecov.io/gh/misskey-dev/misskey) | ||||||
|  |  | ||||||
| </div> | </div> | ||||||
|  |  | ||||||
| <div> | <div> | ||||||
|   | |||||||
| @@ -6,16 +6,13 @@ Also, the later tasks are more indefinite and are subject to change as developme | |||||||
| This is the phase we are at now. We need to make a high-maintenance environment that can withstand future development. | This is the phase we are at now. We need to make a high-maintenance environment that can withstand future development. | ||||||
|  |  | ||||||
| - Make the number of type errors zero (backend) | - Make the number of type errors zero (backend) | ||||||
| 	- Probably need to switch some libraries to others that make it difficult to reduce type errors |  | ||||||
| 		- e.g. koa to fastify https://github.com/misskey-dev/misskey/issues/7537 |  | ||||||
| - Improve CI | - Improve CI | ||||||
| 	- Fix tests | 	- Fix tests | ||||||
| 		- mocha, jest, etc. do not support the combination of `TypeScript + ESM + Path alias`, and the tests currently do not work. |  | ||||||
| 	- Fix random test failures - https://github.com/misskey-dev/misskey/issues/7985 and https://github.com/misskey-dev/misskey/issues/7986 | 	- Fix random test failures - https://github.com/misskey-dev/misskey/issues/7985 and https://github.com/misskey-dev/misskey/issues/7986 | ||||||
| 	- Add more tests | 	- Add more tests | ||||||
| 		- May need to implement a mechanism that allows for DI | 		- ~~May need to implement a mechanism that allows for DI~~ → Done ✔️ | ||||||
| 			- https://github.com/misskey-dev/misskey/pull/9085 | 			- https://github.com/misskey-dev/misskey/pull/9085 | ||||||
| 	- Measure coverage | 	- ~~Measure coverage~~ → Done ✔️ | ||||||
| 		- https://github.com/misskey-dev/misskey/pull/9081 | 		- https://github.com/misskey-dev/misskey/pull/9081 | ||||||
| - Improve documentation | - Improve documentation | ||||||
| - Refactoring | - Refactoring | ||||||
|   | |||||||
| @@ -133,11 +133,6 @@ id: "aid" | |||||||
| # IP address family used for outgoing request (ipv4, ipv6 or dual) | # IP address family used for outgoing request (ipv4, ipv6 or dual) | ||||||
| #outgoingAddressFamily: ipv4 | #outgoingAddressFamily: ipv4 | ||||||
|  |  | ||||||
| # Syslog option |  | ||||||
| #syslog: |  | ||||||
| #  host: localhost |  | ||||||
| #  port: 514 |  | ||||||
|  |  | ||||||
| # Proxy for HTTP/HTTPS | # Proxy for HTTP/HTTPS | ||||||
| #proxy: http://127.0.0.1:3128 | #proxy: http://127.0.0.1:3128 | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										4
									
								
								healthcheck.sh
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								healthcheck.sh
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | |||||||
|  | #!/bin/bash | ||||||
|  |  | ||||||
|  | PORT=$(grep '^port:' /misskey/.config/default.yml | awk 'NR==1{print $2; exit}') | ||||||
|  | curl -s -S -o /dev/null "http://localhost:${PORT}" | ||||||
| @@ -886,56 +886,6 @@ _nsfw: | |||||||
|   respect: "اخف الوسائط ذات المحتوى الحساس" |   respect: "اخف الوسائط ذات المحتوى الحساس" | ||||||
|   ignore: "اعرض الوسائط ذات المحتوى الحساس" |   ignore: "اعرض الوسائط ذات المحتوى الحساس" | ||||||
|   force: "اخف كل الوسائط" |   force: "اخف كل الوسائط" | ||||||
| _mfm: |  | ||||||
|   cheatSheet: "مرجع ملخص عن MFM" |  | ||||||
|   intro: "MFM هي لغة ترميزية مخصصة  يمكن استخدامها في عدّة أماكن في ميسكي. يمكنك مراجعة كل تعابيرها مع كيفية استخدامها هنا." |  | ||||||
|   mention: "أشر الى" |  | ||||||
|   mentionDescription: "يمكنك الإشارة لمستخدم معيّن من خلال كتابة @ متبوعة باسم مستخدم." |  | ||||||
|   hashtag: "الوسوم" |  | ||||||
|   hashtagDescription: "يمكنك تعيين وسم من خلال كتابة # متبوعة بالنص المطلوب." |  | ||||||
|   url: "الرابط" |  | ||||||
|   urlDescription: "يمكن عرض الروابط" |  | ||||||
|   link: "رابط" |  | ||||||
|   bold: "عريض" |  | ||||||
|   boldDescription: "جعل الحروف أثخن لإبرازها." |  | ||||||
|   small: "صغير" |  | ||||||
|   smallDescription: "يعرض المحتوى صغيرًا ورفيعًا." |  | ||||||
|   center: "وسط" |  | ||||||
|   centerDescription: "يمركز المحتوى في الوَسَط." |  | ||||||
|   quote: "اقتبس" |  | ||||||
|   quoteDescription: "يعرض المحتوى كاقتباس" |  | ||||||
|   emoji: "إيموجي مخصص" |  | ||||||
|   emojiDescription: "إحاطة اسم الإيموجي بنقطتي تفسير سيستبدله بصورة الإيموجي." |  | ||||||
|   search: "البحث" |  | ||||||
|   searchDescription: "يعرض نصًا في صندوق البحث" |  | ||||||
|   flip: "اقلب" |  | ||||||
|   flipDescription: "يقلب المحتوى عموديًا أو أفقيًا" |  | ||||||
|   jelly: "تأثير (هلام)" |  | ||||||
|   jellyDescription: "يمنح المحتوى حركة هلامية." |  | ||||||
|   tada: "تأثير (تادا)" |  | ||||||
|   tadaDescription: "يمنح للمحتوى تأثير تادا" |  | ||||||
|   jump: "تأثير (قفز)" |  | ||||||
|   jumpDescription: "يمنح للمحتوى حركة قفز." |  | ||||||
|   bounce: "تأثير (ارتداد)" |  | ||||||
|   bounceDescription: "يمنح للمحتوى حركة ارتدادية" |  | ||||||
|   shake: "تأثير (اهتزاز)" |  | ||||||
|   shakeDescription: "يمنح المحتوى حركة اهتزازية." |  | ||||||
|   spin: "تأثير (دوران)" |  | ||||||
|   spinDescription: "يمنح المحتوى حركة دورانية." |  | ||||||
|   x2: "كبير" |  | ||||||
|   x2Description: "يُكبر المحتوى" |  | ||||||
|   x3: "كبير جداً" |  | ||||||
|   x3Description: "يُضخم المحتوى" |  | ||||||
|   x4: "هائل" |  | ||||||
|   x4Description: "يُضخم المحتوى أكثر مما سبق." |  | ||||||
|   blur: "طمس" |  | ||||||
|   blurDescription: "يطمس المحتوى، لكن بالتمرير فوقه سيظهر بوضوح." |  | ||||||
|   font: "الخط" |  | ||||||
|   fontDescription: "الخط المستخدم لعرض المحتوى." |  | ||||||
|   rainbow: "قوس قزح" |  | ||||||
|   rainbowDescription: "اجعل المحتوى يظهر بألوان الطيف" |  | ||||||
|   rotate: "تدوير" |  | ||||||
|   rotateDescription: "يُدير المحتوى بزاوية معيّنة." |  | ||||||
| _instanceTicker: | _instanceTicker: | ||||||
|   none: "لا تظهره بتاتًا" |   none: "لا تظهره بتاتًا" | ||||||
|   remote: "أظهر للمستخدمين البِعاد" |   remote: "أظهر للمستخدمين البِعاد" | ||||||
| @@ -1345,5 +1295,6 @@ _deck: | |||||||
|     tl: "الخيط الزمني" |     tl: "الخيط الزمني" | ||||||
|     antenna: "الهوائيات" |     antenna: "الهوائيات" | ||||||
|     list: "القوائم" |     list: "القوائم" | ||||||
|  |     channel: "القنوات" | ||||||
|     mentions: "الإشارات" |     mentions: "الإشارات" | ||||||
|     direct: "مباشرة" |     direct: "مباشرة" | ||||||
|   | |||||||
| @@ -455,7 +455,6 @@ youHaveNoGroups: "আপনার কোন গ্রুপ নেই " | |||||||
| joinOrCreateGroup: "একটি বিদ্যমান গ্রুপের আমন্ত্রণ পান বা একটি নতুন গ্রুপ তৈরি করুন৷" | joinOrCreateGroup: "একটি বিদ্যমান গ্রুপের আমন্ত্রণ পান বা একটি নতুন গ্রুপ তৈরি করুন৷" | ||||||
| noHistory: "কোনো ইতিহাস নেই" | noHistory: "কোনো ইতিহাস নেই" | ||||||
| signinHistory: "প্রবেশ করার ইতিহাস" | signinHistory: "প্রবেশ করার ইতিহাস" | ||||||
| disableAnimatedMfm: "অ্যানিমেটেড MFM অক্ষম করুন" |  | ||||||
| doing: "প্রক্রিয়া করছে..." | doing: "প্রক্রিয়া করছে..." | ||||||
| category: "বিভাগ" | category: "বিভাগ" | ||||||
| tags: "ট্যাগসমূহ" | tags: "ট্যাগসমূহ" | ||||||
| @@ -923,70 +922,6 @@ _nsfw: | |||||||
|   respect: "স্পর্শকাতর মিডিয়া লুকান" |   respect: "স্পর্শকাতর মিডিয়া লুকান" | ||||||
|   ignore: "স্পর্শকাতর মিডিয়া লুকাবেন না" |   ignore: "স্পর্শকাতর মিডিয়া লুকাবেন না" | ||||||
|   force: "সকল মিডিয়া লুকান" |   force: "সকল মিডিয়া লুকান" | ||||||
| _mfm: |  | ||||||
|   cheatSheet: "MFM চিটশিট" |  | ||||||
|   intro: "MFM একটি মার্কআপ ভাষা যা Misskey-এর মধ্যে বিভিন্ন জায়গায় ব্যবহার করা যেতে পারে। এখানে আপনি MFM-এর সিনট্যাক্সগুলির একটি তালিকা দেখতে পারবেন।" |  | ||||||
|   dummy: "মিসকি ফেডিভার্সের বিশ্বকে প্রসারিত করে" |  | ||||||
|   mention: "উল্লেখ" |  | ||||||
|   mentionDescription: "@ চিহ্ন + ব্যবহারকারীর নাম একটি নির্দিষ্ট ব্যবহারকারীকে নির্দেশ করতে ব্যবহার করা যায়।" |  | ||||||
|   hashtag: "হ্যাশট্যাগ" |  | ||||||
|   hashtagDescription: "আপনি একটি # চিহ্ন + ট্যাগ সহ একটি হ্যাশট্যাগ নির্দেশ করতে পারেন।" |  | ||||||
|   url: "URL" |  | ||||||
|   urlDescription: "URL দেখানো সম্ভব।" |  | ||||||
|   link: "লিংক" |  | ||||||
|   linkDescription: "আপনি পাঠ্যের একটি নির্দিষ্ট অংশকে URL হিসাবে দেখাতে পারেন৷" |  | ||||||
|   bold: "গাঢ়" |  | ||||||
|   boldDescription: "অক্ষরগুলিকে মোটাকরে প্রদর্শন করা হবে।" |  | ||||||
|   small: "ছোট" |  | ||||||
|   smallDescription: "লেখা ছোট এবং পাতলা করে দেখানো হবে।" |  | ||||||
|   center: "সেন্টার" |  | ||||||
|   centerDescription: "লেখা মাঝ বরাবর দেখানো হবে" |  | ||||||
|   inlineCode: "কোড (ইনলাইন)" |  | ||||||
|   inlineCodeDescription: " প্রোগ্রামের কোডের জন্য ইনলাইন সিনট্যাক্স হাইলাইটিং করা হবে" |  | ||||||
|   blockCode: "কোড (ব্লক)" |  | ||||||
|   blockCodeDescription: "মাল্টি-লাইন প্রোগ্রামের কোডের জন্য সিনট্যাক্স হাইলাইট করে।" |  | ||||||
|   inlineMath: "গাণিতিক সূত্র (ইনলাইন)" |  | ||||||
|   inlineMathDescription: "গাণিতিক সূত্র প্রদর্শন করুন (KaTeX) ইনলাইন।" |  | ||||||
|   blockMath: "গাণিতিক সূত্র (ব্লক)" |  | ||||||
|   blockMathDescription: "একটি ব্লকে একাধিক লাইনের গাণিতিক সূত্র প্রদর্শন করুন (KaTeX)।" |  | ||||||
|   quote: "উদ্ধৃতি" |  | ||||||
|   quoteDescription: "বিষয়বস্তুকে একটি উদ্ধৃতি হিসাবে দেখানো হবে।" |  | ||||||
|   emoji: "স্বনির্ধারিত ইমোজিগুলি" |  | ||||||
|   emojiDescription: "আপনি একটি কাস্টম ইমোজির নাম কোলনে আবদ্ধ করে কাস্টম ইমোজিটি দেখাতে পারেন৷" |  | ||||||
|   search: "খুঁজুন" |  | ||||||
|   searchDescription: "পূর্ব-টাইপ করা পাঠ্য সহ একটি অনুসন্ধান বাক্স প্রদর্শন করে।" |  | ||||||
|   flip: "উল্টান" |  | ||||||
|   flipDescription: "বিষয়বস্তু উপরে/নীচে বা বাম/ডানে উল্টান।" |  | ||||||
|   jelly: "অ্যানিমেশন (জেলি)" |  | ||||||
|   jellyDescription: "জেলির মত অ্যানিমেশন দেখায়।" |  | ||||||
|   tada: "অ্যানিমেশন (টাডা)" |  | ||||||
|   tadaDescription: "\"টাডা!\" এর মত অ্যানিমেশন দেখায়।" |  | ||||||
|   jump: "অ্যানিমেশন (লাফ)" |  | ||||||
|   jumpDescription: "বিষয়বস্তুতে লাফ মারার মত অ্যানিমেশন দেখায়।" |  | ||||||
|   bounce: "অ্যানিমেশন (তিড়িং বিড়িং)" |  | ||||||
|   bounceDescription: "তিড়িং বিড়িং করার মত অ্যানিমেশন দেখায়।" |  | ||||||
|   shake: "অ্যানিমেশন (ঝাঁকি)" |  | ||||||
|   shakeDescription: "ঝাঁকির মত অ্যানিমেশন দেখায়।" |  | ||||||
|   twitch: "অ্যানিমেশন (মোচড়ানো)" |  | ||||||
|   twitchDescription: "মোচড়ানোর মত অ্যানিমেশন দেখায়।" |  | ||||||
|   spin: "অ্যানিমেশন (ঘুরা)" |  | ||||||
|   spinDescription: "ঘুরার মত অ্যানিমেশন দেখায়।" |  | ||||||
|   x2: "বড়" |  | ||||||
|   x2Description: "বিষয়বস্তু বড় করে দেখায়।" |  | ||||||
|   x3: "অনেক বড়" |  | ||||||
|   x3Description: "বিষয়বস্তু আরও বড় করে দেখায়।" |  | ||||||
|   x4: "অস্বাভাবিক বড়" |  | ||||||
|   x4Description: "বিষয়বস্তুকে আগের থেকেও আরও বড় করে দেখায়।" |  | ||||||
|   blur: "ব্লার" |  | ||||||
|   blurDescription: "বিষয়বস্তুকে ব্লার করতে পারেন। আপনি এর উপর মাউস কার্সার রাখলে, এটি পরিষ্কারভাবে দেখতে পাবেন।" |  | ||||||
|   font: "ফন্ট" |  | ||||||
|   fontDescription: "বিষয়বস্তুকে কোন ফন্টে দেখানো হবে তা নির্ধারণ করে।" |  | ||||||
|   rainbow: "রেইনবো" |  | ||||||
|   rainbowDescription: "বিষয়বস্তুকে রংধনুর রং গুলিতে প্রদর্শন করে।" |  | ||||||
|   sparkle: "চিক চিক" |  | ||||||
|   sparkleDescription: "বিষয়বস্তুকে একটি চিকচিকে কণা প্রভাব দেয়।" |  | ||||||
|   rotate: "ঘুরান" |  | ||||||
|   rotateDescription: "বিষয়বস্তুকে একটি নির্দিষ্ট কোনে ঘুরায়।" |  | ||||||
| _instanceTicker: | _instanceTicker: | ||||||
|   none: "দেখাবেন না" |   none: "দেখাবেন না" | ||||||
|   remote: "রিমোট ব্যাবহারকারীদের জন্য দেখান" |   remote: "রিমোট ব্যাবহারকারীদের জন্য দেখান" | ||||||
| @@ -1441,5 +1376,6 @@ _deck: | |||||||
|     tl: "টাইমলাইন" |     tl: "টাইমলাইন" | ||||||
|     antenna: "অ্যান্টেনা" |     antenna: "অ্যান্টেনা" | ||||||
|     list: "লিস্ট" |     list: "লিস্ট" | ||||||
|  |     channel: "চ্যানেলগুলি" | ||||||
|     mentions: "উল্লেখসমূহ" |     mentions: "উল্লেখসমূহ" | ||||||
|     direct: "ডাইরেক্ট নোটগুলি" |     direct: "ডাইরেক্ট নোটগুলি" | ||||||
|   | |||||||
| @@ -375,11 +375,6 @@ file: "Fitxers" | |||||||
| _email: | _email: | ||||||
|   _follow: |   _follow: | ||||||
|     title: "t'ha seguit" |     title: "t'ha seguit" | ||||||
| _mfm: |  | ||||||
|   mention: "Menció" |  | ||||||
|   quote: "Citar" |  | ||||||
|   emoji: "Emojis personalitzats" |  | ||||||
|   search: "Cercar" |  | ||||||
| _instanceMute: | _instanceMute: | ||||||
|   instanceMuteDescription: "Silencia tots els impulsos dels servidors seleccionats, també els usuaris que responen a altres d'un servidor silenciat." |   instanceMuteDescription: "Silencia tots els impulsos dels servidors seleccionats, també els usuaris que responen a altres d'un servidor silenciat." | ||||||
| _theme: | _theme: | ||||||
|   | |||||||
| @@ -642,19 +642,6 @@ _registry: | |||||||
| _aboutMisskey: | _aboutMisskey: | ||||||
|   allContributors: "Všichni přispěvatelé" |   allContributors: "Všichni přispěvatelé" | ||||||
|   source: "Zdrojový kód" |   source: "Zdrojový kód" | ||||||
| _mfm: |  | ||||||
|   mention: "Zmínění" |  | ||||||
|   hashtag: "Hashtag" |  | ||||||
|   link: "Odkaz" |  | ||||||
|   bold: "Tučně" |  | ||||||
|   quote: "Citovat" |  | ||||||
|   emoji: "Vlastní emoji" |  | ||||||
|   search: "Vyhledávání" |  | ||||||
|   flip: "Otočit" |  | ||||||
|   tada: "Animace (tadá)" |  | ||||||
|   blur: "Rozmazání" |  | ||||||
|   font: "Font" |  | ||||||
|   rainbow: "Duha" |  | ||||||
| _channel: | _channel: | ||||||
|   featured: "Trendy" |   featured: "Trendy" | ||||||
| _menuDisplay: | _menuDisplay: | ||||||
| @@ -804,4 +791,5 @@ _deck: | |||||||
|     tl: "Časová osa" |     tl: "Časová osa" | ||||||
|     antenna: "Antény" |     antenna: "Antény" | ||||||
|     list: "Seznamy" |     list: "Seznamy" | ||||||
|  |     channel: "Kanály" | ||||||
|     mentions: "Zmínění" |     mentions: "Zmínění" | ||||||
|   | |||||||
| @@ -68,7 +68,7 @@ export: "Export" | |||||||
| files: "Dateien" | files: "Dateien" | ||||||
| download: "Herunterladen" | download: "Herunterladen" | ||||||
| driveFileDeleteConfirm: "Möchtest du die Datei „{name}“ wirklich löschen? Notizen mit dieser Datei werden ebenso verschwinden." | driveFileDeleteConfirm: "Möchtest du die Datei „{name}“ wirklich löschen? Notizen mit dieser Datei werden ebenso verschwinden." | ||||||
| unfollowConfirm: "Möchtest du {name} 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." | ||||||
| lists: "Listen" | lists: "Listen" | ||||||
| @@ -94,7 +94,7 @@ defaultNoteVisibility: "Standardsichtbarkeit" | |||||||
| follow: "Folgen" | follow: "Folgen" | ||||||
| followRequest: "Follow-Anfrage senden" | followRequest: "Follow-Anfrage senden" | ||||||
| followRequests: "Follow-Anfragen" | followRequests: "Follow-Anfragen" | ||||||
| unfollow: "Nicht mehr folgen" | unfollow: "Entfolgen" | ||||||
| followRequestPending: "Follow-Anfrage ausstehend" | followRequestPending: "Follow-Anfrage ausstehend" | ||||||
| enterEmoji: "Gib ein Emoji ein" | enterEmoji: "Gib ein Emoji ein" | ||||||
| renote: "Renote" | renote: "Renote" | ||||||
| @@ -129,6 +129,7 @@ unblockConfirm: "Möchtest du diese Blockierung wirklich aufheben?" | |||||||
| suspendConfirm: "Möchtest du diesen Benutzer wirklich sperren?" | suspendConfirm: "Möchtest du diesen Benutzer wirklich sperren?" | ||||||
| unsuspendConfirm: "Möchtest du diesen Benutzer wirklich entsperren?" | unsuspendConfirm: "Möchtest du diesen Benutzer wirklich entsperren?" | ||||||
| selectList: "Liste auswählen" | selectList: "Liste auswählen" | ||||||
|  | selectChannel: "Kanal auswählen" | ||||||
| selectAntenna: "Antenne auswählen" | selectAntenna: "Antenne auswählen" | ||||||
| selectWidget: "Widget auswählen" | selectWidget: "Widget auswählen" | ||||||
| editWidgets: "Widgets bearbeiten" | editWidgets: "Widgets bearbeiten" | ||||||
| @@ -256,6 +257,8 @@ noMoreHistory: "Kein weiterer Verlauf vorhanden" | |||||||
| startMessaging: "Neuen Chat erstellen" | startMessaging: "Neuen Chat erstellen" | ||||||
| nUsersRead: "Von {n} Benutzern gelesen" | nUsersRead: "Von {n} Benutzern gelesen" | ||||||
| agreeTo: "Ich stimme {0} zu" | agreeTo: "Ich stimme {0} zu" | ||||||
|  | agreeBelow: "Ich stimme Untenstehendem zu" | ||||||
|  | basicNotesBeforeCreateAccount: "Wichtige Infos" | ||||||
| tos: "Nutzungsbedingungen" | tos: "Nutzungsbedingungen" | ||||||
| start: "Anfangen" | start: "Anfangen" | ||||||
| home: "Startseite" | home: "Startseite" | ||||||
| @@ -464,7 +467,6 @@ youHaveNoGroups: "Keine Gruppen vorhanden" | |||||||
| joinOrCreateGroup: "Lass dich zu einer Gruppe einladen oder erstelle deine eigene." | joinOrCreateGroup: "Lass dich zu einer Gruppe einladen oder erstelle deine eigene." | ||||||
| noHistory: "Kein Verlauf gefunden" | noHistory: "Kein Verlauf gefunden" | ||||||
| signinHistory: "Anmeldungsverlauf" | signinHistory: "Anmeldungsverlauf" | ||||||
| disableAnimatedMfm: "MFM, die Animationen enthalten, deaktivieren" |  | ||||||
| doing: "In Bearbeitung …" | doing: "In Bearbeitung …" | ||||||
| category: "Kategorie" | category: "Kategorie" | ||||||
| tags: "Schlagwörter" | tags: "Schlagwörter" | ||||||
| @@ -861,6 +863,8 @@ failedToFetchAccountInformation: "Benutzerkontoinformationen konnten nicht abgef | |||||||
| rateLimitExceeded: "Versuchsanzahl überschritten" | rateLimitExceeded: "Versuchsanzahl überschritten" | ||||||
| cropImage: "Bild zuschneiden" | cropImage: "Bild zuschneiden" | ||||||
| cropImageAsk: "Möchtest du das Bild zuschneiden?" | cropImageAsk: "Möchtest du das Bild zuschneiden?" | ||||||
|  | cropYes: "Zuschneiden" | ||||||
|  | cropNo: "Unbearbeitet verwenden" | ||||||
| file: "Datei" | file: "Datei" | ||||||
| recentNHours: "Letzten {n} Stunden" | recentNHours: "Letzten {n} Stunden" | ||||||
| recentNDays: "Letzten {n} Tage" | recentNDays: "Letzten {n} Tage" | ||||||
| @@ -939,6 +943,8 @@ cannotPerformTemporaryDescription: "Diese Aktion ist wegen des Überschreitenes | |||||||
| preset: "Vorlage" | preset: "Vorlage" | ||||||
| selectFromPresets: "Aus Vorlagen wählen" | selectFromPresets: "Aus Vorlagen wählen" | ||||||
| achievements: "Errungenschaften" | achievements: "Errungenschaften" | ||||||
|  | gotInvalidResponseError: "Ungültige Antwort des Servers" | ||||||
|  | gotInvalidResponseErrorDescription: "Eventuell ist der Server momentan nicht erreichbar oder untergeht Wartungsarbeiten. Bitte versuche es später noch einmal." | ||||||
| _achievements: | _achievements: | ||||||
|   earnedAt: "Freigeschaltet am" |   earnedAt: "Freigeschaltet am" | ||||||
|   _types: |   _types: | ||||||
| @@ -1195,6 +1201,9 @@ _role: | |||||||
|   baseRole: "Rollenvorlage" |   baseRole: "Rollenvorlage" | ||||||
|   useBaseValue: "Wert der Rollenvorlage verwenden" |   useBaseValue: "Wert der Rollenvorlage verwenden" | ||||||
|   chooseRoleToAssign: "Zuzuweisende Rolle auswählen" |   chooseRoleToAssign: "Zuzuweisende Rolle auswählen" | ||||||
|  |   iconUrl: "Icon-URL" | ||||||
|  |   asBadge: "Als Abzeichen anzeigen" | ||||||
|  |   descriptionOfAsBadge: "Ist dies aktiviert, so wird das Icon dieser Rolle an der Seite der Namen von Benutzern mit dieser Rolle angezeigt." | ||||||
|   canEditMembersByModerator: "Moderatoren können Benutzern diese Rolle zuweisen" |   canEditMembersByModerator: "Moderatoren können Benutzern diese Rolle zuweisen" | ||||||
|   descriptionOfCanEditMembersByModerator: "Wenn aktiviert, so können Moderatoren und Adminstratoren anderen Benutzern diese Rolle zuweisen bzw. diese Zuweisung aufheben. Wenn deaktiviert, so ist es nur Administratoren möglich, Zuweisungen dieser Rolle zu verwalten." |   descriptionOfCanEditMembersByModerator: "Wenn aktiviert, so können Moderatoren und Adminstratoren anderen Benutzern diese Rolle zuweisen bzw. diese Zuweisung aufheben. Wenn deaktiviert, so ist es nur Administratoren möglich, Zuweisungen dieser Rolle zu verwalten." | ||||||
|   priority: "Priorität" |   priority: "Priorität" | ||||||
| @@ -1320,72 +1329,6 @@ _nsfw: | |||||||
|   respect: "Als NSFW markierte Bilder verbergen" |   respect: "Als NSFW markierte Bilder verbergen" | ||||||
|   ignore: "Als NSFW markierte Bilder nicht verbergen" |   ignore: "Als NSFW markierte Bilder nicht verbergen" | ||||||
|   force: "Alle Medien verbergen" |   force: "Alle Medien verbergen" | ||||||
| _mfm: |  | ||||||
|   cheatSheet: "MFM Spickzettel" |  | ||||||
|   intro: "MFM ist eine Misskey-exklusive Markup-Sprache, die in Misskey an vielen Stellen verwendet werden kann. Hier kannst du eine Liste von verfügbarer MFM-Syntax einsehen." |  | ||||||
|   dummy: "Misskey erweitert die Welt des Fediverse" |  | ||||||
|   mention: "Erwähnung" |  | ||||||
|   mentionDescription: "Mit At-Zeichen und Benutzername kann ein individueller Nutzer angegeben werden." |  | ||||||
|   hashtag: "Hashtag" |  | ||||||
|   hashtagDescription: "Mit einer Raute und Text kann ein Hashtag angegeben werden." |  | ||||||
|   url: "URL" |  | ||||||
|   urlDescription: "Zeigt URLs an." |  | ||||||
|   link: "Link" |  | ||||||
|   linkDescription: "Zeigt spezifische Textabschnitte als URL an." |  | ||||||
|   bold: "Fett" |  | ||||||
|   boldDescription: "Zeichen zur Betonung dicker erscheinen lassen." |  | ||||||
|   small: "Klein" |  | ||||||
|   smallDescription: "Inhalt klein und dünn erscheinen lassen." |  | ||||||
|   center: "Zentrieren" |  | ||||||
|   centerDescription: "Inhalt zentriert anzeigen." |  | ||||||
|   inlineCode: "Code (Eingebettet)" |  | ||||||
|   inlineCodeDescription: "Syntax-Hervorhebung für (Programm-)Code eingebettet anzeigen." |  | ||||||
|   blockCode: "Code (Block)" |  | ||||||
|   blockCodeDescription: "Syntax-Hervorhebung für mehrzeiligen (Programm-)Code als Block anzeigen." |  | ||||||
|   inlineMath: "Mathe (Eingebettet)" |  | ||||||
|   inlineMathDescription: "Mathematische Formeln (KaTeX) eingebettet anzeigen." |  | ||||||
|   blockMath: "Mathe (Block)" |  | ||||||
|   blockMathDescription: "Mehrzeilige mathematische Formeln (KaTeX) als Block einbetten." |  | ||||||
|   quote: "Zitationen" |  | ||||||
|   quoteDescription: "Inhalt als Zitat anzeigen." |  | ||||||
|   emoji: "Benutzerdefinierte Emojis" |  | ||||||
|   emojiDescription: "Durch das Umschließen von Emoji-Namen durch Doppelpunkte können benutzerdefinierte Emojis angezeigt werden." |  | ||||||
|   search: "Suche" |  | ||||||
|   searchDescription: "Eine vorgefertige Suchanfragebox anzeigen." |  | ||||||
|   flip: "Spiegelung" |  | ||||||
|   flipDescription: "Inhalt horizontal oder vertikal gespiegelt anzeigen." |  | ||||||
|   jelly: "Animation (Dehnen)" |  | ||||||
|   jellyDescription: "Verleiht Inhalt eine sich dehnende Animation." |  | ||||||
|   tada: "Animation (Tada)" |  | ||||||
|   tadaDescription: "Verleiht Inhalt eine Animation mit \"Tada!\"-Gefühl" |  | ||||||
|   jump: "Animation (Sprung)" |  | ||||||
|   jumpDescription: "Verleiht Inhalt eine springende Animation." |  | ||||||
|   bounce: "Animation (Federn)" |  | ||||||
|   bounceDescription: "Verleiht Inhalt eine federnde Animation." |  | ||||||
|   shake: "Animation (Zittern)" |  | ||||||
|   shakeDescription: "Verleiht Inhalt eine zitternde Animation." |  | ||||||
|   twitch: "Animation (Zucken)" |  | ||||||
|   twitchDescription: "Verleiht Inhalt eine sehr stark zuckende Animation." |  | ||||||
|   spin: "Animation (Rotieren)" |  | ||||||
|   spinDescription: "Verleiht Inhalt eine rotierende Animation." |  | ||||||
|   x2: "Groß" |  | ||||||
|   x2Description: "Inhalte größer anzeigen." |  | ||||||
|   x3: "Sehr groß" |  | ||||||
|   x3Description: "Inhalte noch größer anzeigen." |  | ||||||
|   x4: "Unglaublich groß" |  | ||||||
|   x4Description: "Lässt Inhalte noch größer als größer als groß angezeigt werden." |  | ||||||
|   blur: "Weichzeichnen" |  | ||||||
|   blurDescription: "Inhalte durch Weihzeichnung verschwimmen lassen. Durch das Bewegen des Mauszeigers über den Inhalt wird er klar angezeigt." |  | ||||||
|   font: "Schriftart" |  | ||||||
|   fontDescription: "Setzt die Schriftart des Inhaltes fest." |  | ||||||
|   rainbow: "Regenbogen" |  | ||||||
|   rainbowDescription: "Lässt den Inhalt in Regenbogenfarben erscheinen." |  | ||||||
|   sparkle: "Glitzer" |  | ||||||
|   sparkleDescription: "Verleiht Inhalt einen glitzernden Partikeleffekt." |  | ||||||
|   rotate: "Drehen" |  | ||||||
|   rotateDescription: "Dreht den Inhalt um einen angegebenen Winkel." |  | ||||||
|   plain: "Schlicht" |  | ||||||
|   plainDescription: "Deaktiviert jegliche MFM-Syntax, die sich innerhalb dieses MFM-Effekts befindet." |  | ||||||
| _instanceTicker: | _instanceTicker: | ||||||
|   none: "Nie anzeigen" |   none: "Nie anzeigen" | ||||||
|   remote: "Für Benutzer fremder Instanzen anzeigen" |   remote: "Für Benutzer fremder Instanzen anzeigen" | ||||||
| @@ -1590,12 +1533,15 @@ _permissions: | |||||||
|   "read:gallery-likes": "Liste deiner mit \"Gefällt mir\" markierten Galerie-Beiträge lesen" |   "read:gallery-likes": "Liste deiner mit \"Gefällt mir\" markierten Galerie-Beiträge lesen" | ||||||
|   "write:gallery-likes": "Liste deiner mit \"Gefällt mir\" markierten Galerie-Beiträge bearbeiten" |   "write:gallery-likes": "Liste deiner mit \"Gefällt mir\" markierten Galerie-Beiträge bearbeiten" | ||||||
| _auth: | _auth: | ||||||
|  |   shareAccessTitle: "Verteilung von App-Berechtigungen" | ||||||
|   shareAccess: "Möchtest du „{name}“ authorisieren, auf dieses Benutzerkonto zugreifen zu können?" |   shareAccess: "Möchtest du „{name}“ authorisieren, auf dieses Benutzerkonto zugreifen zu können?" | ||||||
|   shareAccessAsk: "Bist du dir sicher, dass du diese Anwendung authorisieren möchtest, auf dein Benutzerkonto zugreifen zu können?" |   shareAccessAsk: "Bist du dir sicher, dass du diese Anwendung authorisieren möchtest, auf dein Benutzerkonto zugreifen zu können?" | ||||||
|  |   permission: "{name} fordert folgende Berechtigungen" | ||||||
|   permissionAsk: "Diese Anwendung fordert folgende Berechtigungen" |   permissionAsk: "Diese Anwendung fordert folgende Berechtigungen" | ||||||
|   pleaseGoBack: "Bitte kehre zur Anwendung zurück" |   pleaseGoBack: "Bitte kehre zur Anwendung zurück" | ||||||
|   callback: "Es wird zur Anwendung zurückgekehrt" |   callback: "Es wird zur Anwendung zurückgekehrt" | ||||||
|   denied: "Zugriff verweigert" |   denied: "Zugriff verweigert" | ||||||
|  |   pleaseLogin: "Bitte logge dich ein, um Apps zu authorisieren." | ||||||
| _antennaSources: | _antennaSources: | ||||||
|   all: "Alle Notizen" |   all: "Alle Notizen" | ||||||
|   homeTimeline: "Notizen von Benutzern, denen gefolgt wird" |   homeTimeline: "Notizen von Benutzern, denen gefolgt wird" | ||||||
| @@ -1866,5 +1812,6 @@ _deck: | |||||||
|     tl: "Chronik" |     tl: "Chronik" | ||||||
|     antenna: "Antennen" |     antenna: "Antennen" | ||||||
|     list: "Listen" |     list: "Listen" | ||||||
|  |     channel: "Kanal" | ||||||
|     mentions: "Erwähnungen" |     mentions: "Erwähnungen" | ||||||
|     direct: "Direktnachrichten" |     direct: "Direktnachrichten" | ||||||
|   | |||||||
| @@ -298,11 +298,6 @@ cannotUploadBecauseNoFreeSpace: "Το ανέβασμα απέτυχε λόγω  | |||||||
| _email: | _email: | ||||||
|   _follow: |   _follow: | ||||||
|     title: "Έχετε ένα νέο ακόλουθο" |     title: "Έχετε ένα νέο ακόλουθο" | ||||||
| _mfm: |  | ||||||
|   mention: "Επισήμανση" |  | ||||||
|   quote: "Παράθεση" |  | ||||||
|   emoji: "Επιπλέον emoji" |  | ||||||
|   search: "Αναζήτηση" |  | ||||||
| _channel: | _channel: | ||||||
|   featured: "Δημοφιλή" |   featured: "Δημοφιλή" | ||||||
| _theme: | _theme: | ||||||
|   | |||||||
| @@ -68,7 +68,7 @@ export: "Export" | |||||||
| files: "Files" | files: "Files" | ||||||
| download: "Download" | download: "Download" | ||||||
| driveFileDeleteConfirm: "Are you sure you want to delete the file \"{name}\"? Notes with this file attached will also be deleted." | driveFileDeleteConfirm: "Are you sure you want to delete the file \"{name}\"? Notes with this file attached will also be deleted." | ||||||
| unfollowConfirm: "Are you sure that you want to unfollow {name}?" | unfollowConfirm: "Are you sure you want to unfollow {name}?" | ||||||
| exportRequested: "You've requested an export. This may take a while. It will be added to your Drive once completed." | exportRequested: "You've requested an export. This may take a while. It will be added to your Drive once completed." | ||||||
| importRequested: "You've requested an import. This may take a while." | importRequested: "You've requested an import. This may take a while." | ||||||
| lists: "Lists" | lists: "Lists" | ||||||
| @@ -129,6 +129,7 @@ unblockConfirm: "Are you sure that you want to unblock this account?" | |||||||
| suspendConfirm: "Are you sure that you want to suspend this account?" | suspendConfirm: "Are you sure that you want to suspend this account?" | ||||||
| unsuspendConfirm: "Are you sure that you want to unsuspend this account?" | unsuspendConfirm: "Are you sure that you want to unsuspend this account?" | ||||||
| selectList: "Select a list" | selectList: "Select a list" | ||||||
|  | selectChannel: "Select a channel" | ||||||
| selectAntenna: "Select an antenna" | selectAntenna: "Select an antenna" | ||||||
| selectWidget: "Select a widget" | selectWidget: "Select a widget" | ||||||
| editWidgets: "Edit widgets" | editWidgets: "Edit widgets" | ||||||
| @@ -256,6 +257,8 @@ noMoreHistory: "There is no further history" | |||||||
| startMessaging: "Start a new chat" | startMessaging: "Start a new chat" | ||||||
| nUsersRead: "read by {n}" | nUsersRead: "read by {n}" | ||||||
| agreeTo: "I agree to {0}" | agreeTo: "I agree to {0}" | ||||||
|  | agreeBelow: "I agree to the below" | ||||||
|  | basicNotesBeforeCreateAccount: "Important notes" | ||||||
| tos: "Terms of Service" | tos: "Terms of Service" | ||||||
| start: "Begin" | start: "Begin" | ||||||
| home: "Home" | home: "Home" | ||||||
| @@ -464,7 +467,8 @@ youHaveNoGroups: "You have no groups" | |||||||
| joinOrCreateGroup: "Get invited to a group or create your own." | joinOrCreateGroup: "Get invited to a group or create your own." | ||||||
| noHistory: "No history available" | noHistory: "No history available" | ||||||
| signinHistory: "Login history" | signinHistory: "Login history" | ||||||
| disableAnimatedMfm: "Disable MFM with animation" | enableAdvancedMfm: "Enable advanced MFM" | ||||||
|  | enableAnimatedMfm: "Enable MFM with animation" | ||||||
| doing: "Processing..." | doing: "Processing..." | ||||||
| category: "Category" | category: "Category" | ||||||
| tags: "Tags" | tags: "Tags" | ||||||
| @@ -861,6 +865,8 @@ failedToFetchAccountInformation: "Could not fetch account information" | |||||||
| rateLimitExceeded: "Rate limit exceeded" | rateLimitExceeded: "Rate limit exceeded" | ||||||
| cropImage: "Crop image" | cropImage: "Crop image" | ||||||
| cropImageAsk: "Do you want to crop this image?" | cropImageAsk: "Do you want to crop this image?" | ||||||
|  | cropYes: "Crop" | ||||||
|  | cropNo: "Use as-is" | ||||||
| file: "File" | file: "File" | ||||||
| recentNHours: "Last {n} hours" | recentNHours: "Last {n} hours" | ||||||
| recentNDays: "Last {n} days" | recentNDays: "Last {n} days" | ||||||
| @@ -939,6 +945,12 @@ cannotPerformTemporaryDescription: "This action cannot be performed temporarily | |||||||
| preset: "Preset" | preset: "Preset" | ||||||
| selectFromPresets: "Choose from presets" | selectFromPresets: "Choose from presets" | ||||||
| achievements: "Achievements" | achievements: "Achievements" | ||||||
|  | gotInvalidResponseError: "Invalid server response" | ||||||
|  | gotInvalidResponseErrorDescription: "The server may be unreachable or undergoing maintenance. Please try again later." | ||||||
|  | thisPostMayBeAnnoying: "This note may annoy others." | ||||||
|  | thisPostMayBeAnnoyingHome: "Post to home timeline" | ||||||
|  | thisPostMayBeAnnoyingCancel: "Cancel" | ||||||
|  | thisPostMayBeAnnoyingIgnore: "Post anyway" | ||||||
| _achievements: | _achievements: | ||||||
|   earnedAt: "Unlocked at" |   earnedAt: "Unlocked at" | ||||||
|   _types: |   _types: | ||||||
| @@ -1195,6 +1207,9 @@ _role: | |||||||
|   baseRole: "Role template" |   baseRole: "Role template" | ||||||
|   useBaseValue: "Use role template value" |   useBaseValue: "Use role template value" | ||||||
|   chooseRoleToAssign: "Select the role to assign" |   chooseRoleToAssign: "Select the role to assign" | ||||||
|  |   iconUrl: "Icon URL" | ||||||
|  |   asBadge: "Show as badge" | ||||||
|  |   descriptionOfAsBadge: "This role's icon will be displayed next to the username of users with this role if turned on." | ||||||
|   canEditMembersByModerator: "Allow moderators to edit the list of members for this role" |   canEditMembersByModerator: "Allow moderators to edit the list of members for this role" | ||||||
|   descriptionOfCanEditMembersByModerator: "When turned on, moderators as well as administrators will be able to assign and unassign users to this role. When turned off, only administrators will be able to assign users." |   descriptionOfCanEditMembersByModerator: "When turned on, moderators as well as administrators will be able to assign and unassign users to this role. When turned off, only administrators will be able to assign users." | ||||||
|   priority: "Priority" |   priority: "Priority" | ||||||
| @@ -1320,72 +1335,6 @@ _nsfw: | |||||||
|   respect: "Hide NSFW media" |   respect: "Hide NSFW media" | ||||||
|   ignore: "Don't hide NSFW media" |   ignore: "Don't hide NSFW media" | ||||||
|   force: "Hide all media" |   force: "Hide all media" | ||||||
| _mfm: |  | ||||||
|   cheatSheet: "MFM Cheatsheet" |  | ||||||
|   intro: "MFM is a Misskey-exclusive markup language that can be used in many places. Here you can view a list of all available MFM syntax." |  | ||||||
|   dummy: "Misskey expands the world of the Fediverse" |  | ||||||
|   mention: "Mention" |  | ||||||
|   mentionDescription: "You can specify a user by using an At-Symbol and a username." |  | ||||||
|   hashtag: "Hashtag" |  | ||||||
|   hashtagDescription: "You can specify a hashtag using a number sign and text." |  | ||||||
|   url: "URL" |  | ||||||
|   urlDescription: "URLs can be displayed." |  | ||||||
|   link: "Link" |  | ||||||
|   linkDescription: "Specific parts of text can be displayed as a URL." |  | ||||||
|   bold: "Bold" |  | ||||||
|   boldDescription: "Highlights letters by making them thicker." |  | ||||||
|   small: "Small" |  | ||||||
|   smallDescription: "Displays content small and thin." |  | ||||||
|   center: "Center" |  | ||||||
|   centerDescription: "Displays content centered." |  | ||||||
|   inlineCode: "Code (Inline)" |  | ||||||
|   inlineCodeDescription: "Displays inline syntax highlighting for (program) code." |  | ||||||
|   blockCode: "Code (Block)" |  | ||||||
|   blockCodeDescription: "Displays syntax highlighting for multi-line (program) code in a block." |  | ||||||
|   inlineMath: "Math (Inline)" |  | ||||||
|   inlineMathDescription: "Display math formulas (KaTeX) in-line" |  | ||||||
|   blockMath: "Math (Block)" |  | ||||||
|   blockMathDescription: "Display multi-line math formulas (KaTeX) in a block" |  | ||||||
|   quote: "Quote" |  | ||||||
|   quoteDescription: "Displays content as a quote." |  | ||||||
|   emoji: "Custom Emoji" |  | ||||||
|   emojiDescription: "By surrounding a custom emoji name with colons, custom emoji can be displayed." |  | ||||||
|   search: "Search" |  | ||||||
|   searchDescription: "Displays a search box with pre-entered text." |  | ||||||
|   flip: "Flip" |  | ||||||
|   flipDescription: "Flips content horizontally or vertically." |  | ||||||
|   jelly: "Animation (Jelly)" |  | ||||||
|   jellyDescription: "Gives content a jelly-like animation." |  | ||||||
|   tada: "Animation (Tada)" |  | ||||||
|   tadaDescription: "Gives content a \"Tada!\"-like animation." |  | ||||||
|   jump: "Animation (Jump)" |  | ||||||
|   jumpDescription: "Gives content a jumping animation." |  | ||||||
|   bounce: "Animation (Bounce)" |  | ||||||
|   bounceDescription: "Gives content a bouncy animation." |  | ||||||
|   shake: "Animation (Shake)" |  | ||||||
|   shakeDescription: "Gives content a shaking animation." |  | ||||||
|   twitch: "Animation (Twitch)" |  | ||||||
|   twitchDescription: "Gives content a strongly twitching animation." |  | ||||||
|   spin: "Animation (Spin)" |  | ||||||
|   spinDescription: "Gives content a spinning animation." |  | ||||||
|   x2: "Big" |  | ||||||
|   x2Description: "Displays content bigger." |  | ||||||
|   x3: "Very big" |  | ||||||
|   x3Description: "Displays content even bigger." |  | ||||||
|   x4: "Unbelievably big" |  | ||||||
|   x4Description: "Displays content even bigger than bigger than big." |  | ||||||
|   blur: "Blur" |  | ||||||
|   blurDescription: "Blurs content. It will be displayed clearly when hovered over." |  | ||||||
|   font: "Font" |  | ||||||
|   fontDescription: "Sets the font to display content in." |  | ||||||
|   rainbow: "Rainbow" |  | ||||||
|   rainbowDescription: "Makes the content appear in rainbow colors." |  | ||||||
|   sparkle: "Sparkle" |  | ||||||
|   sparkleDescription: "Gives content a sparkling particle effect." |  | ||||||
|   rotate: "Rotate" |  | ||||||
|   rotateDescription: "Turns content by a specified angle." |  | ||||||
|   plain: "Plain" |  | ||||||
|   plainDescription: "Deactivates the effects of all MFM contained within this MFM effect." |  | ||||||
| _instanceTicker: | _instanceTicker: | ||||||
|   none: "Never show" |   none: "Never show" | ||||||
|   remote: "Show for remote users" |   remote: "Show for remote users" | ||||||
| @@ -1590,12 +1539,15 @@ _permissions: | |||||||
|   "read:gallery-likes": "View your list of liked gallery posts" |   "read:gallery-likes": "View your list of liked gallery posts" | ||||||
|   "write:gallery-likes": "Edit your list of liked gallery posts" |   "write:gallery-likes": "Edit your list of liked gallery posts" | ||||||
| _auth: | _auth: | ||||||
|  |   shareAccessTitle: "Granting application permissions" | ||||||
|   shareAccess: "Would you like to authorize \"{name}\" to access this account?" |   shareAccess: "Would you like to authorize \"{name}\" to access this account?" | ||||||
|   shareAccessAsk: "Are you sure you want to authorize this application to access your account?" |   shareAccessAsk: "Are you sure you want to authorize this application to access your account?" | ||||||
|  |   permission: "{name} requests the following permissions" | ||||||
|   permissionAsk: "This application requests the following permissions" |   permissionAsk: "This application requests the following permissions" | ||||||
|   pleaseGoBack: "Please go back to the application" |   pleaseGoBack: "Please go back to the application" | ||||||
|   callback: "Returning to the application" |   callback: "Returning to the application" | ||||||
|   denied: "Access denied" |   denied: "Access denied" | ||||||
|  |   pleaseLogin: "Please log in to authorize applications." | ||||||
| _antennaSources: | _antennaSources: | ||||||
|   all: "All notes" |   all: "All notes" | ||||||
|   homeTimeline: "Notes from followed users" |   homeTimeline: "Notes from followed users" | ||||||
| @@ -1866,5 +1818,6 @@ _deck: | |||||||
|     tl: "Timeline" |     tl: "Timeline" | ||||||
|     antenna: "Antennas" |     antenna: "Antennas" | ||||||
|     list: "List" |     list: "List" | ||||||
|  |     channel: "Channel" | ||||||
|     mentions: "Mentions" |     mentions: "Mentions" | ||||||
|     direct: "Direct notes" |     direct: "Direct notes" | ||||||
|   | |||||||
| @@ -56,7 +56,7 @@ reply: "Responder" | |||||||
| loadMore: "Ver más" | loadMore: "Ver más" | ||||||
| showMore: "Ver más" | showMore: "Ver más" | ||||||
| showLess: "Cerrar" | showLess: "Cerrar" | ||||||
| youGotNewFollower: "te ha seguido" | youGotNewFollower: "ahora te sigue" | ||||||
| receiveFollowRequest: "Recibiste una solicitud de seguimiento" | receiveFollowRequest: "Recibiste una solicitud de seguimiento" | ||||||
| followRequestAccepted: "La solicitud de seguimiento fue aceptada" | followRequestAccepted: "La solicitud de seguimiento fue aceptada" | ||||||
| mention: "Menciones" | mention: "Menciones" | ||||||
| @@ -129,6 +129,7 @@ unblockConfirm: "¿Quiere dejar de bloquear esta cuenta?" | |||||||
| suspendConfirm: "¿Quiere suspender esta cuenta?" | suspendConfirm: "¿Quiere suspender esta cuenta?" | ||||||
| unsuspendConfirm: "¿Quiere dejar de suspender esta cuenta?" | unsuspendConfirm: "¿Quiere dejar de suspender esta cuenta?" | ||||||
| selectList: "Seleccione una lista" | selectList: "Seleccione una lista" | ||||||
|  | selectChannel: "Seleccionar canal" | ||||||
| selectAntenna: "Seleccionar antena" | selectAntenna: "Seleccionar antena" | ||||||
| selectWidget: "Seleccionar widget" | selectWidget: "Seleccionar widget" | ||||||
| editWidgets: "Editar widgets" | editWidgets: "Editar widgets" | ||||||
| @@ -256,6 +257,8 @@ noMoreHistory: "El historial se ha acabado" | |||||||
| startMessaging: "Iniciar chat" | startMessaging: "Iniciar chat" | ||||||
| nUsersRead: "Leído por {n} personas" | nUsersRead: "Leído por {n} personas" | ||||||
| agreeTo: "De acuerdo con {0}" | agreeTo: "De acuerdo con {0}" | ||||||
|  | agreeBelow: "Estoy de acuerdo con lo siguiente" | ||||||
|  | basicNotesBeforeCreateAccount: "Notas básicas" | ||||||
| tos: "Términos de uso" | tos: "Términos de uso" | ||||||
| start: "Comenzar" | start: "Comenzar" | ||||||
| home: "Inicio" | home: "Inicio" | ||||||
| @@ -464,7 +467,6 @@ youHaveNoGroups: "Sin grupos" | |||||||
| joinOrCreateGroup: "Obtenga una invitación para unirse al grupos o puede crear su propio grupo." | joinOrCreateGroup: "Obtenga una invitación para unirse al grupos o puede crear su propio grupo." | ||||||
| noHistory: "No hay datos en el historial" | noHistory: "No hay datos en el historial" | ||||||
| signinHistory: "Historial de ingresos" | signinHistory: "Historial de ingresos" | ||||||
| disableAnimatedMfm: "Deshabilitar MFM que tiene animaciones" |  | ||||||
| doing: "Voy en camino" | doing: "Voy en camino" | ||||||
| category: "Categoría" | category: "Categoría" | ||||||
| tags: "Etiqueta" | tags: "Etiqueta" | ||||||
| @@ -509,7 +511,7 @@ objectStorageSetPublicRead: "Seleccionar \"public-read\" al subir " | |||||||
| serverLogs: "Registros del servidor" | serverLogs: "Registros del servidor" | ||||||
| deleteAll: "Eliminar todos" | deleteAll: "Eliminar todos" | ||||||
| showFixedPostForm: "Mostrar el formulario de las entradas encima de la línea de tiempo" | showFixedPostForm: "Mostrar el formulario de las entradas encima de la línea de tiempo" | ||||||
| newNoteRecived: "Tienes una nota nuevo" | newNoteRecived: "Tienes una nota nueva" | ||||||
| sounds: "Sonidos" | sounds: "Sonidos" | ||||||
| sound: "Sonidos" | sound: "Sonidos" | ||||||
| listen: "Escuchar" | listen: "Escuchar" | ||||||
| @@ -918,17 +920,328 @@ tools: "Utilidades" | |||||||
| cannotLoad: "No se puede cargar." | cannotLoad: "No se puede cargar." | ||||||
| numberOfProfileView: "Número de vistas de perfil" | numberOfProfileView: "Número de vistas de perfil" | ||||||
| like: "¡Muy bien!" | like: "¡Muy bien!" | ||||||
|  | unlike: "Quitar 'me gusta'" | ||||||
|  | numberOfLikes: "Cantidad de 'Me gusta'" | ||||||
| show: "Apariencia" | show: "Apariencia" | ||||||
|  | neverShow: "No mostrar de nuevo" | ||||||
|  | remindMeLater: "Recordar después" | ||||||
|  | didYouLikeMisskey: "¿Te gusta Misskey?" | ||||||
|  | pleaseDonate: "Misskey es software libre, y es usado por {host} . Por favor, ¡considera donar al proyecto principal para que podamos continuar!" | ||||||
|  | roles: "Roles" | ||||||
|  | role: "Roles" | ||||||
|  | normalUser: "Usuario normal" | ||||||
|  | undefined: "Indefinido" | ||||||
|  | assign: "Asignar" | ||||||
|  | unassign: "Quitar" | ||||||
| color: "Color" | color: "Color" | ||||||
|  | manageCustomEmojis: "Administrar emojis personalizados" | ||||||
|  | youCannotCreateAnymore: "Se alcanzó el límite de creación" | ||||||
|  | cannotPerformTemporary: "Indisponible temporalmente" | ||||||
|  | cannotPerformTemporaryDescription: "Esta acción no se puede realizar porque se excedió el límite de ejecución. Espera un poco y prueba de nuevo." | ||||||
|  | preset: "Predefinido" | ||||||
|  | selectFromPresets: "Escoger desde predefinidos" | ||||||
|  | achievements: "Logros" | ||||||
|  | gotInvalidResponseError: "Respuesta del servidor inválida" | ||||||
|  | gotInvalidResponseErrorDescription: "Puede que el servidor esté caído o en mantenimiento. Favor de intentar más tarde" | ||||||
|  | _achievements: | ||||||
|  |   earnedAt: "Desbloqueado el" | ||||||
|  |   _types: | ||||||
|  |     _notes1: | ||||||
|  |       title: "¡Hola Misskey!" | ||||||
|  |       description: "Publicaste tu primera nota" | ||||||
|  |       flavor: "¡Pasándola bien con Misskey!" | ||||||
|  |     _notes10: | ||||||
|  |       title: "Algunas notas" | ||||||
|  |       description: "10 notas publicadas" | ||||||
|  |     _notes100: | ||||||
|  |       title: "¡Muchas notas!" | ||||||
|  |       description: "100 notas publicadas" | ||||||
|  |     _notes500: | ||||||
|  |       title: "¡Cubierto de notas!" | ||||||
|  |       description: "500 notas publicadas" | ||||||
|  |     _notes1000: | ||||||
|  |       title: "¡Una montaña de notas!" | ||||||
|  |       description: "1000 notas publicadas" | ||||||
|  |     _notes5000: | ||||||
|  |       title: "¡Exceso de notas!" | ||||||
|  |       description: "5000 notas publicadas" | ||||||
|  |     _notes10000: | ||||||
|  |       title: "¡Súpernota!" | ||||||
|  |       description: "10000 notas publicadas" | ||||||
|  |     _notes20000: | ||||||
|  |       title: "Necesito... Más... ¡Notas!" | ||||||
|  |       description: "20000 notas publicadas" | ||||||
|  |     _notes30000: | ||||||
|  |       title: "¡Notas! ¡Notas! ¡Notas!" | ||||||
|  |       description: "30000 notas publicadas" | ||||||
|  |     _notes40000: | ||||||
|  |       title: "Fábrica de notas" | ||||||
|  |       description: "40000 notas publicadas" | ||||||
|  |     _notes50000: | ||||||
|  |       title: "¡Un planeta de notas!" | ||||||
|  |       description: "50000 notas publicadas" | ||||||
|  |     _notes60000: | ||||||
|  |       title: "¡Un cuásar de notas!" | ||||||
|  |       description: "60000 notas publicadas" | ||||||
|  |     _notes70000: | ||||||
|  |       title: "¡Un hoyo negro de notas!" | ||||||
|  |       description: "70000 notas publicadas" | ||||||
|  |     _notes80000: | ||||||
|  |       title: "¡Una galaxia de notas!" | ||||||
|  |       description: "80000 notas publicadas" | ||||||
|  |     _notes90000: | ||||||
|  |       title: "¡Todo un universo de notas!" | ||||||
|  |       description: "90000 notas publicadas" | ||||||
|  |     _notes100000: | ||||||
|  |       title: "ALL YOUR NOTE ARE BELONG TO US" | ||||||
|  |       description: "100000 notas publicadas" | ||||||
|  |       flavor: "¿Tienes tanto para publicar?" | ||||||
|  |     _login3: | ||||||
|  |       title: "Principiante I" | ||||||
|  |       description: "Días desde el inicio de sesión: 3" | ||||||
|  |       flavor: "Desde hoy, soy Misskero" | ||||||
|  |     _login7: | ||||||
|  |       title: "Principiante II" | ||||||
|  |       description: "Días desde el inicio de sesión: 7" | ||||||
|  |       flavor: "¿Ya te acostumbraste?" | ||||||
|  |     _login15: | ||||||
|  |       title: "Principiante III" | ||||||
|  |       description: "Días desde el inicio de sesión: 15" | ||||||
|  |     _login30: | ||||||
|  |       title: "Misskero I" | ||||||
|  |       description: "Días desde el inicio de sesión: 30" | ||||||
|  |     _login60: | ||||||
|  |       title: "Misskero II" | ||||||
|  |       description: "Días desde el inicio de sesión: 60" | ||||||
|  |     _login100: | ||||||
|  |       title: "Misskero III" | ||||||
|  |       description: "Días desde el inicio de sesión: 100" | ||||||
|  |       flavor: "Para este usuario, Misskaína" | ||||||
|  |     _login200: | ||||||
|  |       title: "Regular I" | ||||||
|  |       description: "Días desde el inicio de sesión: 200" | ||||||
|  |     _login300: | ||||||
|  |       title: "Regular II" | ||||||
|  |       description: "Días desde el inicio de sesión: 300" | ||||||
|  |     _login400: | ||||||
|  |       title: "Regular III" | ||||||
|  |       description: "Días desde el inicio de sesión: 400" | ||||||
|  |     _login500: | ||||||
|  |       title: "Veterano I" | ||||||
|  |       description: "Días desde el inicio de sesión: 500" | ||||||
|  |       flavor: "Chicos, me encantan las libretas..." | ||||||
|  |     _login600: | ||||||
|  |       title: "Veterano II" | ||||||
|  |       description: "Días desde el inicio de sesión: 600" | ||||||
|  |     _login700: | ||||||
|  |       title: "Veterano III" | ||||||
|  |       description: "Días desde el inicio de sesión: 700" | ||||||
|  |     _login800: | ||||||
|  |       title: "Maestro I" | ||||||
|  |       description: "Días desde el inicio de sesión: 800" | ||||||
|  |     _login900: | ||||||
|  |       title: "Maestro II" | ||||||
|  |       description: "Días desde el inicio de sesión: 900" | ||||||
|  |     _login1000: | ||||||
|  |       title: "Maestro III" | ||||||
|  |       description: "Días desde el inicio de sesión: 1000" | ||||||
|  |       flavor: "¡Gracias por usar Misskey!" | ||||||
|  |     _noteClipped1: | ||||||
|  |       title: "No puedo evitar clipearte..." | ||||||
|  |       description: "Hacer un clip por primera vez" | ||||||
|  |     _noteFavorited1: | ||||||
|  |       title: "Contemplando las estrellas" | ||||||
|  |       description: "Poner una nota como favorito por primera vez" | ||||||
|  |     _myNoteFavorited1: | ||||||
|  |       title: "¡Quiero una estrella!" | ||||||
|  |       description: "Tu nota ha sido marcada como favorito por primera vez" | ||||||
|  |     _profileFilled: | ||||||
|  |       title: "¡Listo!" | ||||||
|  |       description: "Perfil completado" | ||||||
|  |     _markedAsCat: | ||||||
|  |       title: "Soy un gato" | ||||||
|  |       description: "Configurar la cuenta como cuenta de un gato" | ||||||
|  |       flavor: "Aún no tengo nombre" | ||||||
|  |     _following1: | ||||||
|  |       title: "Primera vez siguiendo a alguien" | ||||||
|  |       description: "Seguir a un usuario" | ||||||
|  |     _following10: | ||||||
|  |       title: "Ahí la llevas, ahí la llevas..." | ||||||
|  |       description: "10 usuarios seguidos" | ||||||
|  |     _following50: | ||||||
|  |       title: "¡Un puñado de amigos!" | ||||||
|  |       description: "50 cuentas seguidas" | ||||||
|  |     _following100: | ||||||
|  |       title: "100 amigos" | ||||||
|  |       description: "100 cuentas seguidas" | ||||||
|  |     _following300: | ||||||
|  |       title: "¡Sobrecarga de amigos!" | ||||||
|  |       description: "300 cuentas seguidas" | ||||||
|  |     _followers1: | ||||||
|  |       title: "¡Tu primer seguidor!" | ||||||
|  |       description: "1 seguidor ganado" | ||||||
|  |     _followers10: | ||||||
|  |       title: "¡Sígueme!" | ||||||
|  |       description: "10 seguidores ganados" | ||||||
|  |     _followers50: | ||||||
|  |       title: "Viniendo en manada" | ||||||
|  |       description: "50 seguidores ganados" | ||||||
|  |     _followers100: | ||||||
|  |       title: "Popular" | ||||||
|  |       description: "100 cuentas seguidas" | ||||||
|  |     _followers300: | ||||||
|  |       title: "Por favor, hagan una fila" | ||||||
|  |       description: "300 seguidores ganados" | ||||||
|  |     _followers500: | ||||||
|  |       title: "¡Toda una torre de radio!" | ||||||
|  |       description: "500 seguidores ganados" | ||||||
|  |     _followers1000: | ||||||
|  |       title: "\"Influyente\"" | ||||||
|  |       description: "1000 seguidores gandos" | ||||||
|  |     _collectAchievements30: | ||||||
|  |       title: "Coleccionista" | ||||||
|  |       description: "30 logros ganados" | ||||||
|  |     _viewAchievements3min: | ||||||
|  |       title: "¡Te gustan los logros!" | ||||||
|  |       description: "Mirando tus logros por 3 minutos" | ||||||
|  |     _iLoveMisskey: | ||||||
|  |       title: "¡AMO Misskey!" | ||||||
|  |       description: "\"I ❤ #Misskey\" Publicado" | ||||||
|  |       flavor: "El equipo de desarrollo de Misskey, en verdad, ¡aprecia tu apoyo!" | ||||||
|  |     _foundTreasure: | ||||||
|  |       title: "Búsqueda del tesoro" | ||||||
|  |       description: "Encontraste un tesoro" | ||||||
|  |     _client30min: | ||||||
|  |       title: "Un descansito" | ||||||
|  |       description: "30 minutos dedicados a Misskey" | ||||||
|  |     _noteDeletedWithin1min: | ||||||
|  |       title: "Ah... Mejor no..." | ||||||
|  |       description: "Borrar una nota antes que de pase 1 minuto" | ||||||
|  |     _postedAtLateNight: | ||||||
|  |       title: "Nocturno" | ||||||
|  |       description: "Una nota publicada por la noche" | ||||||
|  |       flavor: "¡Ya casi es hora de dormir!" | ||||||
|  |     _postedAt0min0sec: | ||||||
|  |       title: "Reloj parlante" | ||||||
|  |       description: "Publicar una nota a las 00:00 de la madrugada" | ||||||
|  |       flavor: "Tic, tic, tic ¡TUUUUUN!" | ||||||
|  |     _selfQuote: | ||||||
|  |       title: "Autoreferencia" | ||||||
|  |       description: "Citar tu propia nota" | ||||||
|  |     _htl20npm: | ||||||
|  |       title: "Línea de tiempo fluyendo" | ||||||
|  |       description: "La velocidad de tu línea de tiempo excede las 20 npm (notas por minuto)" | ||||||
|  |     _viewInstanceChart: | ||||||
|  |       title: "Analista" | ||||||
|  |       description: "Gráficas de la instancia mostradas" | ||||||
|  |     _outputHelloWorldOnScratchpad: | ||||||
|  |       title: "¡Hola mundo!" | ||||||
|  |       description: "Escribir \"hello world\" en el compositor" | ||||||
|  |     _open3windows: | ||||||
|  |       title: "Multiventana" | ||||||
|  |       description: "Tener más de 3 ventanas al mismo tiempo" | ||||||
|  |     _driveFolderCircularReference: | ||||||
|  |       title: "Referencia circular" | ||||||
|  |       description: "Intento de crear carpetas recursivamente" | ||||||
|  |     _reactWithoutRead: | ||||||
|  |       title: "¡Sí lo leíste bien?" | ||||||
|  |       description: "Reaccionar a los 3 segundos de publicación de una nota con más de 100 caracteres" | ||||||
|  |     _clickedClickHere: | ||||||
|  |       title: "Pícale aquí" | ||||||
|  |       description: "Le picó ahí" | ||||||
|  |     _justPlainLucky: | ||||||
|  |       title: "Pura suerte" | ||||||
|  |       description: "Obtenido con una probabilidad del 0.01% cada 10 segundos" | ||||||
|  |     _setNameToSyuilo: | ||||||
|  |       title: "Complejo de superioridad" | ||||||
|  |       description: "Configurar el nombre como 'Syuilo'" | ||||||
|  |     _passedSinceAccountCreated1: | ||||||
|  |       title: "Primer aniversario" | ||||||
|  |       description: "Pasó un año desde la creación de la cuenta" | ||||||
|  |     _passedSinceAccountCreated2: | ||||||
|  |       title: "Segundo aniversario" | ||||||
|  |       description: "Pasaron dos años desde la creación de la cuenta" | ||||||
|  |     _passedSinceAccountCreated3: | ||||||
|  |       title: "Tercer aniversario" | ||||||
|  |       description: "Pasaron tres años desde la creación de la cuenta" | ||||||
|  |     _loggedInOnBirthday: | ||||||
|  |       title: "¡Feliz cumpleaños!" | ||||||
|  |       description: "En linea el día de tu cumpleaños" | ||||||
|  |     _loggedInOnNewYearsDay: | ||||||
|  |       title: "¡Feliz Año Nuevo!" | ||||||
|  |       description: "En linea en año nuevo" | ||||||
|  |       flavor: "¡Gracias por tu apoyo a la instancia durante todo este año!" | ||||||
|  |     _cookieClicked: | ||||||
|  |       title: "Un juego para picarle a una galleta" | ||||||
|  |       description: "Picaste una galleta" | ||||||
|  |       flavor: "¿Está mal este juego?" | ||||||
|  |     _brainDiver: | ||||||
|  |       title: "Brain Diver" | ||||||
|  |       description: "Publicaste un vínculo a \"Brain Diver\"" | ||||||
|  |       flavor: "Misskey-Misskey La-Tu-Ma" | ||||||
| _role: | _role: | ||||||
|  |   new: "Crear rol" | ||||||
|  |   edit: "Editar rol" | ||||||
|  |   name: "Nombre del rol" | ||||||
|  |   description: "Descripción del rol" | ||||||
|  |   permission: "Permisos del rol" | ||||||
|  |   descriptionOfPermission: "<b>Moderador</b> Te permite ejecutar acciones básicas de moderación.\n<b>Administradores</b> puede cambiar todas las configuraciones de la instancia." | ||||||
|  |   assignTarget: "Asignar objetivo" | ||||||
|  |   descriptionOfAssignTarget: "<b>Manual</b> Para cambiar manualmente lo que se incluye en este rol.\n<b>Condicional</b> configura una condición, y los usuarios que cumplan la condición serán incluídos automáticamente." | ||||||
|  |   manual: "manual" | ||||||
|  |   conditional: "condicional" | ||||||
|  |   condition: "condición" | ||||||
|  |   isConditionalRole: "Esto es un rol condicional" | ||||||
|  |   isPublic: "Publicar rol" | ||||||
|  |   descriptionOfIsPublic: "Cualquiera puede ver los usuarios asignados a este rol. También, el perfil del usuario mostrará este rol." | ||||||
|  |   options: "Opción" | ||||||
|  |   policies: "Política" | ||||||
|  |   baseRole: "Rol base" | ||||||
|  |   useBaseValue: "Usar los valores del rol base" | ||||||
|  |   chooseRoleToAssign: "Selecciona el rol para asignar" | ||||||
|  |   iconUrl: "URL del ícono" | ||||||
|  |   asBadge: "Mostrar como emblema" | ||||||
|  |   descriptionOfAsBadge: "Este ícono de rol se mostrará a lado del nombre de usuario cuando este rol se encuentre activo." | ||||||
|  |   canEditMembersByModerator: "Permitir a los moderadores editar los miembros" | ||||||
|  |   descriptionOfCanEditMembersByModerator: "Si se activa, los moderadores, al igual que los administradores, serán capaces de asignar/quitar usuarios a éste rol. Si se desactiva, sólo los administradores podrán hacerlo." | ||||||
|   priority: "Prioridad" |   priority: "Prioridad" | ||||||
|   _priority: |   _priority: | ||||||
|     low: "Baja" |     low: "Baja" | ||||||
|     middle: "Mediano" |     middle: "Mediano" | ||||||
|     high: "Alta" |     high: "Alta" | ||||||
|  |   _options: | ||||||
|  |     gtlAvailable: "Explorar la línea de tiempo global" | ||||||
|  |     ltlAvailable: "Explorar la línea de tiempo local" | ||||||
|  |     canPublicNote: "Permitir la publicación" | ||||||
|  |     canInvite: "Puede crear códigos de invitación" | ||||||
|  |     canManageCustomEmojis: "Administrar emojis personalizados" | ||||||
|  |     driveCapacity: "Capacidad de almacenamiento" | ||||||
|  |     pinMax: "Máximo de notas fijadas" | ||||||
|  |     antennaMax: "Máximo de antenas" | ||||||
|  |     wordMuteMax: "Máximo de caracteres en palabras silenciadas" | ||||||
|  |     webhookMax: "Máximo de Webhooks" | ||||||
|  |     clipMax: "Máximo de clips" | ||||||
|  |     noteEachClipsMax: "Máximo de notas con clip" | ||||||
|  |     userListMax: "Máximo de listas de usuarios" | ||||||
|  |     userEachUserListsMax: "Máximo de usuarios en una lista" | ||||||
|  |     rateLimitFactor: "Limitador" | ||||||
|  |     descriptionOfRateLimitFactor: "Límites más bajos son menos restrictivos, más altos menos restrictivos" | ||||||
|  |     canHideAds: "Puede ocultar anuncios" | ||||||
|  |   _condition: | ||||||
|  |     isLocal: "Usuario local" | ||||||
|  |     isRemote: "Usuario remoto" | ||||||
|  |     createdLessThan: "Menos de X han pasado desde la creación de la cuenta" | ||||||
|  |     createdMoreThan: "Más de X han pasado desde la creación de la cuenta" | ||||||
|  |     followersLessThanOrEq: "Tiene X o menos seguidores" | ||||||
|  |     followersMoreThanOrEq: "Tiene X o más seguidores" | ||||||
|  |     followingLessThanOrEq: "Sigue X o menos cuentas" | ||||||
|  |     followingMoreThanOrEq: "Sigue X o más cuentas" | ||||||
|  |     and: "Condicional AND" | ||||||
|  |     or: "Condicional OR" | ||||||
|  |     not: "Condicional NOT" | ||||||
| _sensitiveMediaDetection: | _sensitiveMediaDetection: | ||||||
|   description: "Reduce el esfuerzo de la moderación el el servidor a través del reconocimiento automático de contenido NSFW usando 'Machine Learning'. Esto puede incrementar ligeramente la carga en el servidor." |   description: "Reduce el esfuerzo de la moderación en el servidor a través del reconocimiento automático de contenido NSFW usando 'Machine Learning'. Esto puede incrementar ligeramente la carga en el servidor." | ||||||
|   sensitivity: "Sensibilidad de detección" |   sensitivity: "Sensibilidad de la detección" | ||||||
|   sensitivityDescription: "Reducir la sensibilidad puede acarrear a varios falsos positivos, mientras que incrementarla puede reducir las detecciones (falsos negativos)." |   sensitivityDescription: "Reducir la sensibilidad puede acarrear a varios falsos positivos, mientras que incrementarla puede reducir las detecciones (falsos negativos)." | ||||||
|   setSensitiveFlagAutomatically: "Marcar como NSFW" |   setSensitiveFlagAutomatically: "Marcar como NSFW" | ||||||
|   setSensitiveFlagAutomaticallyDescription: "Los resultados de la detección interna pueden ser retenidos incluso si la opción está desactivada." |   setSensitiveFlagAutomaticallyDescription: "Los resultados de la detección interna pueden ser retenidos incluso si la opción está desactivada." | ||||||
| @@ -1014,72 +1327,6 @@ _nsfw: | |||||||
|   respect: "Ocultar medios NSFW" |   respect: "Ocultar medios NSFW" | ||||||
|   ignore: "No esconder medios NSFW " |   ignore: "No esconder medios NSFW " | ||||||
|   force: "Ocultar todos los medios" |   force: "Ocultar todos los medios" | ||||||
| _mfm: |  | ||||||
|   cheatSheet: "Hoja de referencia de MFM" |  | ||||||
|   intro: "MFM es un lenguaje de marcado dedicado que se puede usar en varios lugares dentro de Misskey. Aquí puede ver una lista de sintaxis disponibles en MFM." |  | ||||||
|   dummy: "Misskey expande el mundo de la Fediverso" |  | ||||||
|   mention: "Menciones" |  | ||||||
|   mentionDescription: "El signo @ seguido de un nombre de usuario se puede utilizar para notificar a un usuario en particular." |  | ||||||
|   hashtag: "Hashtag" |  | ||||||
|   hashtagDescription: "Puede especificar un hashtag con un numeral y el texto." |  | ||||||
|   url: "URL" |  | ||||||
|   urlDescription: "Se pueden mostrar las URL" |  | ||||||
|   link: "Vínculo" |  | ||||||
|   linkDescription: "Se pueden asociar partes de texto a la URL" |  | ||||||
|   bold: "Negrita" |  | ||||||
|   boldDescription: "Muestra el texto con las letras más gruesas" |  | ||||||
|   small: "Pequeño" |  | ||||||
|   smallDescription: "Muestra el texto más pequeño y delgado" |  | ||||||
|   center: "Centrar" |  | ||||||
|   centerDescription: "Muestra el texto centrado" |  | ||||||
|   inlineCode: "Código (insertado)" |  | ||||||
|   inlineCodeDescription: "Muestra el código de un programa resaltando su sintaxis" |  | ||||||
|   blockCode: "Código (bloque)" |  | ||||||
|   blockCodeDescription: "Código de resaltado de sintaxis, como programas de varias líneas con bloques." |  | ||||||
|   inlineMath: "Fórmula (insertado)" |  | ||||||
|   inlineMathDescription: "Muestra fórmulas (KaTeX) insertadas" |  | ||||||
|   blockMath: "Fórmula (bloque)" |  | ||||||
|   blockMathDescription: "Muestra fórmulas (KaTeX) de varias líneas en un bloque" |  | ||||||
|   quote: "Citar" |  | ||||||
|   quoteDescription: "Muestra el contenido como una cita" |  | ||||||
|   emoji: "Emojis personalizados" |  | ||||||
|   emojiDescription: "Muestra los emojis personalizados encerrados entre dos puntos." |  | ||||||
|   search: "Buscar" |  | ||||||
|   searchDescription: "Muestra una caja de búsqueda con texto pre-escrito" |  | ||||||
|   flip: "Echar de un capirotazo" |  | ||||||
|   flipDescription: "Voltea el contenido hacia arriba / abajo o hacia la izquierda / derecha." |  | ||||||
|   jelly: "Animación (gelatina)" |  | ||||||
|   jellyDescription: "Aplica un efecto de animación tipo gelatina" |  | ||||||
|   tada: "Animación (tadá)" |  | ||||||
|   tadaDescription: "Aplica un efecto de animación al estilo \"Tadá\"" |  | ||||||
|   jump: "Animación (saltar)" |  | ||||||
|   jumpDescription: "Aplica un efecto de animación tipo salto" |  | ||||||
|   bounce: "Animación (rebotar)" |  | ||||||
|   bounceDescription: "Aplica un efecto de animación tipo rebote" |  | ||||||
|   shake: "Animación (temblor)" |  | ||||||
|   shakeDescription: "Aplica un efecto de animación tipo temblor" |  | ||||||
|   twitch: "Animación (sacudida)" |  | ||||||
|   twitchDescription: "Aplica un efecto de animación tipo sacudida" |  | ||||||
|   spin: "Animación (giro)" |  | ||||||
|   spinDescription: "Aplica un efecto de animación tipo rotación" |  | ||||||
|   x2: "Grande" |  | ||||||
|   x2Description: "Muestra el contenido más grande" |  | ||||||
|   x3: "Muy grande" |  | ||||||
|   x3Description: "Muestra el contenido mucho más grande" |  | ||||||
|   x4: "Totalmente grande" |  | ||||||
|   x4Description: "Muestra el contenido totalmente grande" |  | ||||||
|   blur: "Desenfoque" |  | ||||||
|   blurDescription: "Para desenfocar el contenido. Se muestra claramente al colocar el puntero encima." |  | ||||||
|   font: "Fuente" |  | ||||||
|   fontDescription: "Elegir la fuente del contenido" |  | ||||||
|   rainbow: "Arcoíris" |  | ||||||
|   rainbowDescription: "Muestra el contenido con los colores del arcoíris" |  | ||||||
|   sparkle: "Parpadeante" |  | ||||||
|   sparkleDescription: "Aplica un efecto de partículas parpadeantes" |  | ||||||
|   rotate: "Rotar" |  | ||||||
|   rotateDescription: "Rota el contenido a un ángulo especificado." |  | ||||||
|   plain: "Plano" |  | ||||||
|   plainDescription: "Desactiva los efectos de todo el contenido MFM con este efecto MFM." |  | ||||||
| _instanceTicker: | _instanceTicker: | ||||||
|   none: "No mostrar" |   none: "No mostrar" | ||||||
|   remote: "Mostrar a usuarios remotos" |   remote: "Mostrar a usuarios remotos" | ||||||
| @@ -1284,12 +1531,15 @@ _permissions: | |||||||
|   "read:gallery-likes": "Ver favoritos de la galería" |   "read:gallery-likes": "Ver favoritos de la galería" | ||||||
|   "write:gallery-likes": "Editar favoritos de la galería" |   "write:gallery-likes": "Editar favoritos de la galería" | ||||||
| _auth: | _auth: | ||||||
|  |   shareAccessTitle: "Permisos de la aplicación" | ||||||
|   shareAccess: "¿Desea permitir el acceso a la cuenta \"{name}\"?" |   shareAccess: "¿Desea permitir el acceso a la cuenta \"{name}\"?" | ||||||
|   shareAccessAsk: "¿Está seguro de que desea autorizar esta aplicación para acceder a su cuenta?" |   shareAccessAsk: "¿Está seguro de que desea autorizar esta aplicación para acceder a su cuenta?" | ||||||
|  |   permission: "{name} solicita los siguientes permisos" | ||||||
|   permissionAsk: "Esta aplicación requiere los siguientes permisos" |   permissionAsk: "Esta aplicación requiere los siguientes permisos" | ||||||
|   pleaseGoBack: "Por favor, vuelve a la aplicación" |   pleaseGoBack: "Por favor, vuelve a la aplicación" | ||||||
|   callback: "Volviendo a la aplicación" |   callback: "Volviendo a la aplicación" | ||||||
|   denied: "Acceso denegado" |   denied: "Acceso denegado" | ||||||
|  |   pleaseLogin: "Se requiere un inicio de sesión para darle permisos a la aplicación" | ||||||
| _antennaSources: | _antennaSources: | ||||||
|   all: "Todas las notas" |   all: "Todas las notas" | ||||||
|   homeTimeline: "Notas de los usuarios que sigues" |   homeTimeline: "Notas de los usuarios que sigues" | ||||||
| @@ -1328,10 +1578,12 @@ _widgets: | |||||||
|   jobQueue: "Cola de trabajos" |   jobQueue: "Cola de trabajos" | ||||||
|   serverMetric: "Estadísticas del servidor" |   serverMetric: "Estadísticas del servidor" | ||||||
|   aiscript: "Consola de AiScript" |   aiscript: "Consola de AiScript" | ||||||
|  |   aiscriptApp: "Aplicación AiScript" | ||||||
|   aichan: "indigo" |   aichan: "indigo" | ||||||
|   userList: "Lista de usuarios" |   userList: "Lista de usuarios" | ||||||
|   _userList: |   _userList: | ||||||
|     chooseList: "Seleccione una lista" |     chooseList: "Seleccione una lista" | ||||||
|  |   clicker: "Cliqueador" | ||||||
| _cw: | _cw: | ||||||
|   hide: "Ocultar" |   hide: "Ocultar" | ||||||
|   show: "Ver más" |   show: "Ver más" | ||||||
| @@ -1434,7 +1686,16 @@ _timelines: | |||||||
|   social: "Social" |   social: "Social" | ||||||
|   global: "Global" |   global: "Global" | ||||||
| _play: | _play: | ||||||
|  |   new: "Crear guión" | ||||||
|  |   edit: "Editar guión" | ||||||
|  |   created: "Guión creado" | ||||||
|  |   updated: "Guión editado" | ||||||
|  |   deleted: "Guión eliminado" | ||||||
|  |   pageSetting: "Configuración de guión" | ||||||
|  |   editThisPage: "Editar este guión" | ||||||
|   viewSource: "Ver la fuente" |   viewSource: "Ver la fuente" | ||||||
|  |   my: "Mis guiones" | ||||||
|  |   liked: "Guiones que te gustaron" | ||||||
|   featured: "Popular" |   featured: "Popular" | ||||||
|   title: "Título" |   title: "Título" | ||||||
|   script: "Script" |   script: "Script" | ||||||
| @@ -1507,6 +1768,7 @@ _notification: | |||||||
|   pollEnded: "Estan disponibles los resultados de la encuesta" |   pollEnded: "Estan disponibles los resultados de la encuesta" | ||||||
|   unreadAntennaNote: "Antena {name}" |   unreadAntennaNote: "Antena {name}" | ||||||
|   emptyPushNotificationMessage: "Se han actualizado las notificaciones push" |   emptyPushNotificationMessage: "Se han actualizado las notificaciones push" | ||||||
|  |   achievementEarned: "Logro desbloqueado" | ||||||
|   _types: |   _types: | ||||||
|     all: "Todo" |     all: "Todo" | ||||||
|     follow: "Siguiendo" |     follow: "Siguiendo" | ||||||
| @@ -1548,5 +1810,6 @@ _deck: | |||||||
|     tl: "Linea de tiempo" |     tl: "Linea de tiempo" | ||||||
|     antenna: "Antenas" |     antenna: "Antenas" | ||||||
|     list: "Listas" |     list: "Listas" | ||||||
|  |     channel: "Canal" | ||||||
|     mentions: "Menciones" |     mentions: "Menciones" | ||||||
|     direct: "Mensaje directo" |     direct: "Mensaje directo" | ||||||
|   | |||||||
| @@ -464,7 +464,6 @@ youHaveNoGroups: "Vous n’avez aucun groupe" | |||||||
| joinOrCreateGroup: "Vous pouvez être invité·e à rejoindre des groupes existants ou créer votre propre nouveau groupe." | joinOrCreateGroup: "Vous pouvez être invité·e à rejoindre des groupes existants ou créer votre propre nouveau groupe." | ||||||
| noHistory: "Pas d'historique" | noHistory: "Pas d'historique" | ||||||
| signinHistory: "Historique de connexion" | signinHistory: "Historique de connexion" | ||||||
| disableAnimatedMfm: "Désactiver MFM ayant des animations" |  | ||||||
| doing: "En cours..." | doing: "En cours..." | ||||||
| category: "Catégorie" | category: "Catégorie" | ||||||
| tags: "Étiquettes" | tags: "Étiquettes" | ||||||
| @@ -1011,72 +1010,6 @@ _nsfw: | |||||||
|   respect: "Cacher les médias marqués comme contenu sensible" |   respect: "Cacher les médias marqués comme contenu sensible" | ||||||
|   ignore: "Afficher les médias sensibles" |   ignore: "Afficher les médias sensibles" | ||||||
|   force: "Cacher tous les médias" |   force: "Cacher tous les médias" | ||||||
| _mfm: |  | ||||||
|   cheatSheet: "Antisèche MFM" |  | ||||||
|   intro: "MFM est un langage Markdown spécifique utilisable ici et là dans Misskey. Vous pouvez vérifier ici les structures utilisables avec MFM." |  | ||||||
|   dummy: "La Fédiverse s'agrandit avec Misskey" |  | ||||||
|   mention: "Mentionner" |  | ||||||
|   mentionDescription: "Vous pouvez afficher un utilisateur spécifique en indiquant une arobase suivie d'un nom d'utilisateur" |  | ||||||
|   hashtag: "Hashtags" |  | ||||||
|   hashtagDescription: "Vous pouvez afficher un mot-dièse en utilisant un croisillon et du texte" |  | ||||||
|   url: "URL" |  | ||||||
|   urlDescription: "L'adresse web peut être affichée." |  | ||||||
|   link: "Lien" |  | ||||||
|   linkDescription: "Une partie précise d'une phrase peut être liée à l'adresse web." |  | ||||||
|   bold: "Gras" |  | ||||||
|   boldDescription: "Il est possible de mettre le texte en exergue en le mettant en gras." |  | ||||||
|   small: "Diminuer l'emphase" |  | ||||||
|   smallDescription: "Le contenu peut être affiché en petit et fin." |  | ||||||
|   center: "Centrer" |  | ||||||
|   centerDescription: "Le contenu peut être centré" |  | ||||||
|   inlineCode: "Code (inline)" |  | ||||||
|   inlineCodeDescription: "Coloration syntaxique des lignes de code." |  | ||||||
|   blockCode: "Bloc de code" |  | ||||||
|   blockCodeDescription: "Coloration syntaxique des lignes de code pour les blocs multi-lignes." |  | ||||||
|   inlineMath: "Formule mathématique (inline)" |  | ||||||
|   inlineMathDescription: "Afficher les formules mathématiques (KaTeX)." |  | ||||||
|   blockMath: "Formule mathématique (bloc)" |  | ||||||
|   blockMathDescription: "Afficher les formules mathématiques (KaTeX) multi-lignes dans un bloc." |  | ||||||
|   quote: "Citer" |  | ||||||
|   quoteDescription: "Affiche le contenu sous forme de citation." |  | ||||||
|   emoji: "Émojis personnalisés" |  | ||||||
|   emojiDescription: "Entourez le nom de l'émoji personnalisé de deux points pour l'afficher." |  | ||||||
|   search: "Rechercher" |  | ||||||
|   searchDescription: "Affiche une boîte de recherche avec du texte pré-saisi." |  | ||||||
|   flip: "Inverser" |  | ||||||
|   flipDescription: "Rotation verticale ou horizontale du contenu" |  | ||||||
|   jelly: "Animation (Gelée)" |  | ||||||
|   jellyDescription: "Donne une animation d'étirement." |  | ||||||
|   tada: "Animation (Tada)" |  | ||||||
|   tadaDescription: "Donne une animation qui donne une impression de \"Tada !\"" |  | ||||||
|   jump: "Animation (Saut)" |  | ||||||
|   jumpDescription: "Donne une animation qui saute." |  | ||||||
|   bounce: "Animation (Rebond)" |  | ||||||
|   bounceDescription: "Donne une animation de rebondissement." |  | ||||||
|   shake: "Animation (Secousse)" |  | ||||||
|   shakeDescription: "Donne une animation tremblante." |  | ||||||
|   twitch: "Animation (Tremblement)" |  | ||||||
|   twitchDescription: "Donne une animation de tremblement intense." |  | ||||||
|   spin: "Animation (Rotation)" |  | ||||||
|   spinDescription: "Donne une animation de rotation." |  | ||||||
|   x2: "Grand" |  | ||||||
|   x2Description: "Afficher le contenu en grand." |  | ||||||
|   x3: "Très grand" |  | ||||||
|   x3Description: "Afficher le contenu en très grand." |  | ||||||
|   x4: "Plus grand" |  | ||||||
|   x4Description: "Afficher le contenu en plus grand." |  | ||||||
|   blur: "Flou" |  | ||||||
|   blurDescription: "Le contenu peut être flouté ; il sera visible en le survolant avec le curseur." |  | ||||||
|   font: "Police de caractères" |  | ||||||
|   fontDescription: "Il est possible de choisir la police." |  | ||||||
|   rainbow: "Arc-en-ciel" |  | ||||||
|   rainbowDescription: "Permet d'afficher le contenu en couleurs arc-en-ciel." |  | ||||||
|   sparkle: "Paillettes" |  | ||||||
|   sparkleDescription: "Ajoute un effet scintillant au contenu." |  | ||||||
|   rotate: "Pivoter" |  | ||||||
|   rotateDescription: "Faire pivoter à un angle spécifié." |  | ||||||
|   plain: "Vu texte non formaté" |  | ||||||
|   plainDescription: "Désactive toute la syntaxe interne." |  | ||||||
| _instanceTicker: | _instanceTicker: | ||||||
|   none: "Cacher " |   none: "Cacher " | ||||||
|   remote: "Montrer pour les utilisateur·ice·s distant·e·s" |   remote: "Montrer pour les utilisateur·ice·s distant·e·s" | ||||||
| @@ -1541,5 +1474,6 @@ _deck: | |||||||
|     tl: "Fil" |     tl: "Fil" | ||||||
|     antenna: "Antennes" |     antenna: "Antennes" | ||||||
|     list: "Listes" |     list: "Listes" | ||||||
|  |     channel: "Canaux" | ||||||
|     mentions: "Mentions" |     mentions: "Mentions" | ||||||
|     direct: "Direct" |     direct: "Direct" | ||||||
|   | |||||||
| @@ -464,7 +464,6 @@ youHaveNoGroups: "Kamu tidak memiliki grup" | |||||||
| joinOrCreateGroup: "Bergabunglah dengan grup atau kamu dapat membuat grupmu sendiri." | joinOrCreateGroup: "Bergabunglah dengan grup atau kamu dapat membuat grupmu sendiri." | ||||||
| noHistory: "Tidak ada riwayat" | noHistory: "Tidak ada riwayat" | ||||||
| signinHistory: "Riwayat masuk" | signinHistory: "Riwayat masuk" | ||||||
| disableAnimatedMfm: "Nonaktifkan MFM dengan animasi" |  | ||||||
| doing: "Sedang berkerja..." | doing: "Sedang berkerja..." | ||||||
| category: "Kategori" | category: "Kategori" | ||||||
| tags: "Tandai" | tags: "Tandai" | ||||||
| @@ -1131,70 +1130,6 @@ _nsfw: | |||||||
|   respect: "Sembunyikan media NSFW" |   respect: "Sembunyikan media NSFW" | ||||||
|   ignore: "Jangan sembunyikan media NSFW" |   ignore: "Jangan sembunyikan media NSFW" | ||||||
|   force: "Sembunyikan semua media" |   force: "Sembunyikan semua media" | ||||||
| _mfm: |  | ||||||
|   cheatSheet: "Contekan MFM" |  | ||||||
|   intro: "MFM adalah Misskey-exclusive Markup Language yang dapat digunakan di banyak tempat. Berikut kamu bisa melihat daftar dari syntax MFM yang ada." |  | ||||||
|   dummy: "Misskey membentangkan dunia Fediverse" |  | ||||||
|   mention: "Sebut" |  | ||||||
|   mentionDescription: "Kamu dapat menentukan pengguna tertentu dengan menggunakan simbol-At dan nama engguna mereka." |  | ||||||
|   hashtag: "Tagar" |  | ||||||
|   hashtagDescription: "Kamu dapat menentukan tagar dengan menggunakan angka dan teks." |  | ||||||
|   url: "URL" |  | ||||||
|   urlDescription: "URL dapat ditampilkan." |  | ||||||
|   link: "Tautan" |  | ||||||
|   linkDescription: "Bagian tertentu dari teks dapat ditampilka sebagai URL." |  | ||||||
|   bold: "Tebal" |  | ||||||
|   boldDescription: "Sorot tulisan dengan membuatnya tebal." |  | ||||||
|   small: "Kecil" |  | ||||||
|   smallDescription: "Tampilkan konten kecil dan tipis." |  | ||||||
|   center: "Tengah" |  | ||||||
|   centerDescription: "Tampilkan konten di tengah." |  | ||||||
|   inlineCode: "Kode (Dalam baris)" |  | ||||||
|   inlineCodeDescription: "Menampilkan sorotan sintaks dalam baris untuk kode(program-)." |  | ||||||
|   blockCode: "Kode (Blok)" |  | ||||||
|   blockCodeDescription: "Menampilkan sorotan sintaks untuk kode(program-) multi baris dalam sebuah blok." |  | ||||||
|   inlineMath: "Matematika (Dalam baris)" |  | ||||||
|   inlineMathDescription: "Menampilkan formula matematika (KaTeX) dalam baris." |  | ||||||
|   blockMath: "Matematika (Blok)" |  | ||||||
|   blockMathDescription: "Menampilkan formula matematika (KaTeX) multibaris dalam sebuah blok." |  | ||||||
|   quote: "Kutip" |  | ||||||
|   quoteDescription: "Menampilkan konten sebagai kutipan." |  | ||||||
|   emoji: "Emoji kustom" |  | ||||||
|   emojiDescription: "Emoji kustom dapat ditampilkan dengan mengurung nama emoji kustom menggunakan tanda titik dua." |  | ||||||
|   search: "Penelusuran" |  | ||||||
|   searchDescription: "Menampilkan kotak pencarian dengan teks yang sudah dimasukkan." |  | ||||||
|   flip: "Balik" |  | ||||||
|   flipDescription: "Balikkan konten secara horizontal atau vertikal." |  | ||||||
|   jelly: "Animasi (Jelly)" |  | ||||||
|   jellyDescription: "Menerapkan animasi seperti jelly" |  | ||||||
|   tada: "Animasi (Tada)" |  | ||||||
|   tadaDescription: "Menerapkan animasi seperti \"Kejutan!\"." |  | ||||||
|   jump: "Animasi (Loncat)" |  | ||||||
|   jumpDescription: "Menerapkan animasi melompat." |  | ||||||
|   bounce: "Animasi (Melambung)" |  | ||||||
|   bounceDescription: "Menerapkan animasi melambung." |  | ||||||
|   shake: "Animasi (Goyang)" |  | ||||||
|   shakeDescription: "Menerapkan animasi bergoyang." |  | ||||||
|   twitch: "Animasi (Cubit)" |  | ||||||
|   twitchDescription: "Terapkan animasi cubit yang kuat." |  | ||||||
|   spin: "Animasi (Putar)" |  | ||||||
|   spinDescription: "Terapkan animasi putar." |  | ||||||
|   x2: "Besar" |  | ||||||
|   x2Description: "Tampilkan konten menjadi besar." |  | ||||||
|   x3: "Lebih besar" |  | ||||||
|   x3Description: "Tampilkan konten menjadi lebih besar." |  | ||||||
|   x4: "Sangat besar" |  | ||||||
|   x4Description: "Tampilka konten menjadi sangat besar." |  | ||||||
|   blur: "Buram" |  | ||||||
|   blurDescription: "Konten dapat diburamkan dengan efek ini. Konten dapat ditampilkan dengan jelas dengan melayangkan kursor tetikus di atasnya." |  | ||||||
|   font: "Font" |  | ||||||
|   fontDescription: "Setel font yang ditampilkan untuk konten." |  | ||||||
|   rainbow: "Pelangi" |  | ||||||
|   rainbowDescription: "Membuat konten muncul dalam warna pelangi." |  | ||||||
|   sparkle: "Kelap-kelip" |  | ||||||
|   sparkleDescription: "Memberikan konten efek partikel kelap-kelip." |  | ||||||
|   rotate: "Putar" |  | ||||||
|   rotateDescription: "Putar konten sesuai sudut yang ditentukan." |  | ||||||
| _instanceTicker: | _instanceTicker: | ||||||
|   none: "Jangan tampilkan" |   none: "Jangan tampilkan" | ||||||
|   remote: "Tampilkan untuk pengguna luar" |   remote: "Tampilkan untuk pengguna luar" | ||||||
| @@ -1673,5 +1608,6 @@ _deck: | |||||||
|     tl: "Linimasa" |     tl: "Linimasa" | ||||||
|     antenna: "Antena" |     antenna: "Antena" | ||||||
|     list: "Daftar" |     list: "Daftar" | ||||||
|  |     channel: "Kanal" | ||||||
|     mentions: "Sebutan" |     mentions: "Sebutan" | ||||||
|     direct: "Langsung" |     direct: "Langsung" | ||||||
|   | |||||||
| @@ -34,6 +34,7 @@ const languages = [ | |||||||
| 	'pt-PT', | 	'pt-PT', | ||||||
| 	'ru-RU', | 	'ru-RU', | ||||||
| 	'sk-SK', | 	'sk-SK', | ||||||
|  | 	'th-TH', | ||||||
| 	'ug-CN', | 	'ug-CN', | ||||||
| 	'uk-UA', | 	'uk-UA', | ||||||
| 	'vi-VN', | 	'vi-VN', | ||||||
|   | |||||||
| @@ -464,7 +464,6 @@ youHaveNoGroups: "Nessun gruppo" | |||||||
| joinOrCreateGroup: "Puoi creare il tuo gruppo o essere invitat@ a gruppi che già esistono." | joinOrCreateGroup: "Puoi creare il tuo gruppo o essere invitat@ a gruppi che già esistono." | ||||||
| noHistory: "Nessuna cronologia" | noHistory: "Nessuna cronologia" | ||||||
| signinHistory: "Storico degli accessi al profilo" | signinHistory: "Storico degli accessi al profilo" | ||||||
| disableAnimatedMfm: "Disabilità i MFM animati" |  | ||||||
| doing: "In corso..." | doing: "In corso..." | ||||||
| category: "Categoria" | category: "Categoria" | ||||||
| tags: "Tag" | tags: "Tag" | ||||||
| @@ -1044,7 +1043,7 @@ _achievements: | |||||||
|       flavor: "Grazie per aver usato Misskey!" |       flavor: "Grazie per aver usato Misskey!" | ||||||
|     _noteClipped1: |     _noteClipped1: | ||||||
|       title: "Devo clippare!" |       title: "Devo clippare!" | ||||||
|       description: "Ho raccolto in Clip la prima Nota" |       description: "Hai raccolto la tua prima Nota in una Clip" | ||||||
|     _noteFavorited1: |     _noteFavorited1: | ||||||
|       title: "Guarda le stelle" |       title: "Guarda le stelle" | ||||||
|       description: "Aggiungi una Nota ai preferiti per la prima volta" |       description: "Aggiungi una Nota ai preferiti per la prima volta" | ||||||
| @@ -1080,7 +1079,7 @@ _achievements: | |||||||
|       title: "Follow me!" |       title: "Follow me!" | ||||||
|       description: "Hai ottenuto 10 profili Follower" |       description: "Hai ottenuto 10 profili Follower" | ||||||
|     _followers50: |     _followers50: | ||||||
|       title: "Follower a frotte" |       title: "Un gregge di Follower" | ||||||
|       description: "Hai ottenuto 50 Follower" |       description: "Hai ottenuto 50 Follower" | ||||||
|     _followers100: |     _followers100: | ||||||
|       title: "Popolare" |       title: "Popolare" | ||||||
| @@ -1108,7 +1107,7 @@ _achievements: | |||||||
|       title: "Caccia al tesoro" |       title: "Caccia al tesoro" | ||||||
|       description: "Hai trovato un tesoro nascosto" |       description: "Hai trovato un tesoro nascosto" | ||||||
|     _client30min: |     _client30min: | ||||||
|       title: "Piccola pausa" |       title: "Piccola grande pausa" | ||||||
|       description: "Hai passato più di 30 minuti su Misskey" |       description: "Hai passato più di 30 minuti su Misskey" | ||||||
|     _noteDeletedWithin1min: |     _noteDeletedWithin1min: | ||||||
|       title: "Ooops!" |       title: "Ooops!" | ||||||
| @@ -1134,7 +1133,7 @@ _achievements: | |||||||
|       title: "Hello, world!" |       title: "Hello, world!" | ||||||
|       description: "Hai scritto «Hello world» nel blocco appunti" |       description: "Hai scritto «Hello world» nel blocco appunti" | ||||||
|     _open3windows: |     _open3windows: | ||||||
|       title: "Finestrato" |       title: "Apri le finestre!" | ||||||
|       description: "Hai aperto almeno 3 finestre contemporaneamente" |       description: "Hai aperto almeno 3 finestre contemporaneamente" | ||||||
|     _driveFolderCircularReference: |     _driveFolderCircularReference: | ||||||
|       title: "Riferimento circolare" |       title: "Riferimento circolare" | ||||||
| @@ -1170,7 +1169,7 @@ _achievements: | |||||||
|     _cookieClicked: |     _cookieClicked: | ||||||
|       title: "Clicca il biscotto" |       title: "Clicca il biscotto" | ||||||
|       description: "Hai giocato a cliccare il cookie" |       description: "Hai giocato a cliccare il cookie" | ||||||
|       flavor: "Hai autorizzato i cookie?" |       flavor: "È il sito giusto?" | ||||||
|     _brainDiver: |     _brainDiver: | ||||||
|       title: "Brain Diver" |       title: "Brain Diver" | ||||||
|       description: "Pubblica un link a Brain Diver" |       description: "Pubblica un link a Brain Diver" | ||||||
| @@ -1195,6 +1194,9 @@ _role: | |||||||
|   baseRole: "Ruolo di base" |   baseRole: "Ruolo di base" | ||||||
|   useBaseValue: "Eredita dal ruolo base" |   useBaseValue: "Eredita dal ruolo base" | ||||||
|   chooseRoleToAssign: "Seleziona il ruolo da assegnare" |   chooseRoleToAssign: "Seleziona il ruolo da assegnare" | ||||||
|  |   iconUrl: "URL dell'icona" | ||||||
|  |   asBadge: "Mostra come badge" | ||||||
|  |   descriptionOfAsBadge: "Se indicato, accanto al nome utente viene visualizzata l'icona del ruolo." | ||||||
|   canEditMembersByModerator: "Anche i Moderatori assegnano profili a questo ruolo" |   canEditMembersByModerator: "Anche i Moderatori assegnano profili a questo ruolo" | ||||||
|   descriptionOfCanEditMembersByModerator: "Se disattivo, potranno farlo solamente gli Amministratori." |   descriptionOfCanEditMembersByModerator: "Se disattivo, potranno farlo solamente gli Amministratori." | ||||||
|   priority: "Priorità" |   priority: "Priorità" | ||||||
| @@ -1320,72 +1322,6 @@ _nsfw: | |||||||
|   respect: "Nascondere i media segnati come sensibli" |   respect: "Nascondere i media segnati come sensibli" | ||||||
|   ignore: "Visualizzare i media segnati come sensibili" |   ignore: "Visualizzare i media segnati come sensibili" | ||||||
|   force: "Nascondere tutti i media" |   force: "Nascondere tutti i media" | ||||||
| _mfm: |  | ||||||
|   cheatSheet: "Bigliettino MFM" |  | ||||||
|   intro: "MFM è un linguaggio Markdown particolare che si può usare in diverse parti di Misskey. Qui puoi visualizzare a colpo d'occhio tutta la sintassi MFM utile." |  | ||||||
|   dummy: "Il Fediverso si espande con Misskey" |  | ||||||
|   mention: "Menzioni" |  | ||||||
|   mentionDescription: "Si può menzionare un utente specifico digitando il suo nome utente subito dopo il segno @." |  | ||||||
|   hashtag: "Hashtag" |  | ||||||
|   hashtagDescription: "Per indicare un hashtag si può usare il segno numerico + tag." |  | ||||||
|   url: "URL" |  | ||||||
|   urlDescription: "È possibile indicare gli URL" |  | ||||||
|   link: "Link" |  | ||||||
|   linkDescription: "È possibile associare specifici intervalli di testo agli URL" |  | ||||||
|   bold: "Grassetto" |  | ||||||
|   boldDescription: "Il testo può essere grassettato per enfasi" |  | ||||||
|   small: "vistosamente" |  | ||||||
|   smallDescription: "Il contenuto può essere visualizzato più piccolo e più sottile" |  | ||||||
|   center: "centratura" |  | ||||||
|   centerDescription: "Il contenuto può essere centrato" |  | ||||||
|   inlineCode: "Codice (inline)" |  | ||||||
|   inlineCodeDescription: "Evidenziazione della sintassi in linea di programmi e altro codice" |  | ||||||
|   blockCode: "Codice (blocco)" |  | ||||||
|   blockCodeDescription: "Evidenziazione della sintassi di programmi multilinea e di altro codice in blocchi" |  | ||||||
|   inlineMath: "Espressione matematica(Immersione)" |  | ||||||
|   inlineMathDescription: "Visualizza le formule (KaTeX) in linea." |  | ||||||
|   blockMath: "Formula matematica (blocco)" |  | ||||||
|   blockMathDescription: "Visualizzazione di formule multilinea (KaTeX) in blocchi." |  | ||||||
|   quote: "Cita il nota" |  | ||||||
|   quoteDescription: "Può indicare che il contenuto è una citazione." |  | ||||||
|   emoji: "Emoji personalizzati" |  | ||||||
|   emojiDescription: "Utilizzare i due punti per racchiudere il nome di un'emoji personalizzata e visualizzarla." |  | ||||||
|   search: "Cerca" |  | ||||||
|   searchDescription: "È possibile visualizzare una casella di ricerca precompilata." |  | ||||||
|   flip: "Inverti" |  | ||||||
|   flipDescription: "Capovolgere il contenuto verso l'alto o verso il basso, a sinistra o a destra." |  | ||||||
|   jelly: "Animazione (Biyon Biyon)." |  | ||||||
|   jellyDescription: "Dà un'animazione di salto." |  | ||||||
|   tada: "Animazione (jang)." |  | ||||||
|   tadaDescription: "Ta-da! dà un'animazione che assomiglia a." |  | ||||||
|   jump: "Animazione(salto)" |  | ||||||
|   jumpDescription: "Da un animazione che salta su e giù." |  | ||||||
|   bounce: "Animazione(rimbalzo)" |  | ||||||
|   bounceDescription: "Rende il testo rimbalzante" |  | ||||||
|   shake: "rimbalzante" |  | ||||||
|   shakeDescription: "Rende il testo traballante" |  | ||||||
|   twitch: "testo" |  | ||||||
|   twitchDescription: "Fa tremare il testo" |  | ||||||
|   spin: "Animazione (rotazione)" |  | ||||||
|   spinDescription: "Fornisce un'animazione rotante." |  | ||||||
|   x2: "Più grande" |  | ||||||
|   x2Description: "Mostra il contenuto ingrandito." |  | ||||||
|   x3: "Molto più grande" |  | ||||||
|   x3Description: "Mostra il contenuto molto più ingrandito." |  | ||||||
|   x4: "Estremamente più grande" |  | ||||||
|   x4Description: "Mostra il contenuto estremamente più ingrandito." |  | ||||||
|   blur: "Sfocatura" |  | ||||||
|   blurDescription: "È possibile rendere sfocato il contenuto. Spostando il cursore su di esso tornerà visibile chiaramente." |  | ||||||
|   font: "Tipo di carattere" |  | ||||||
|   fontDescription: "Puoi scegliere il tipo di carattere per il contenuto." |  | ||||||
|   rainbow: "Arcobaleno" |  | ||||||
|   rainbowDescription: "Arcobaleno il contenuto." |  | ||||||
|   sparkle: "brillantini" |  | ||||||
|   sparkleDescription: "Aggiungere effetti particellari scintillanti." |  | ||||||
|   rotate: "Ruota" |  | ||||||
|   rotateDescription: "Ruota con un angolo specificato." |  | ||||||
|   plain: "Testo semplice" |  | ||||||
|   plainDescription: "Disattiva tutta la sintassi interna." |  | ||||||
| _instanceTicker: | _instanceTicker: | ||||||
|   none: "Nascondi" |   none: "Nascondi" | ||||||
|   remote: "Mostra solo per i profili remoti" |   remote: "Mostra solo per i profili remoti" | ||||||
| @@ -1866,5 +1802,6 @@ _deck: | |||||||
|     tl: "Timeline" |     tl: "Timeline" | ||||||
|     antenna: "Antenne" |     antenna: "Antenne" | ||||||
|     list: "Liste" |     list: "Liste" | ||||||
|  |     channel: "Canale" | ||||||
|     mentions: "Menzioni" |     mentions: "Menzioni" | ||||||
|     direct: "Diretta" |     direct: "Diretta" | ||||||
|   | |||||||
| @@ -103,6 +103,8 @@ renoted: "Renoteしました。" | |||||||
| cantRenote: "この投稿はRenoteできません。" | cantRenote: "この投稿はRenoteできません。" | ||||||
| cantReRenote: "RenoteをRenoteすることはできません。" | cantReRenote: "RenoteをRenoteすることはできません。" | ||||||
| quote: "引用" | quote: "引用" | ||||||
|  | inChannelRenote: "チャンネル内Renote" | ||||||
|  | inChannelQuote: "チャンネル内引用" | ||||||
| pinnedNote: "ピン留めされたノート" | pinnedNote: "ピン留めされたノート" | ||||||
| pinned: "ピン留め" | pinned: "ピン留め" | ||||||
| you: "あなた" | you: "あなた" | ||||||
| @@ -129,6 +131,7 @@ unblockConfirm: "ブロック解除しますか?" | |||||||
| suspendConfirm: "凍結しますか?" | suspendConfirm: "凍結しますか?" | ||||||
| unsuspendConfirm: "解凍しますか?" | unsuspendConfirm: "解凍しますか?" | ||||||
| selectList: "リストを選択" | selectList: "リストを選択" | ||||||
|  | selectChannel: "チャンネルを選択" | ||||||
| selectAntenna: "アンテナを選択" | selectAntenna: "アンテナを選択" | ||||||
| selectWidget: "ウィジェットを選択" | selectWidget: "ウィジェットを選択" | ||||||
| editWidgets: "ウィジェットを編集" | editWidgets: "ウィジェットを編集" | ||||||
| @@ -256,6 +259,8 @@ noMoreHistory: "これより過去の履歴はありません" | |||||||
| startMessaging: "チャットを開始" | startMessaging: "チャットを開始" | ||||||
| nUsersRead: "{n}人が読みました" | nUsersRead: "{n}人が読みました" | ||||||
| agreeTo: "{0}に同意" | agreeTo: "{0}に同意" | ||||||
|  | agreeBelow: "下記に同意する" | ||||||
|  | basicNotesBeforeCreateAccount: "基本的な注意事項" | ||||||
| tos: "利用規約" | tos: "利用規約" | ||||||
| start: "始める" | start: "始める" | ||||||
| home: "ホーム" | home: "ホーム" | ||||||
| @@ -464,7 +469,8 @@ youHaveNoGroups: "グループがありません" | |||||||
| joinOrCreateGroup: "既存のグループに招待してもらうか、新しくグループを作成してください。" | joinOrCreateGroup: "既存のグループに招待してもらうか、新しくグループを作成してください。" | ||||||
| noHistory: "履歴はありません" | noHistory: "履歴はありません" | ||||||
| signinHistory: "ログイン履歴" | signinHistory: "ログイン履歴" | ||||||
| disableAnimatedMfm: "動きのあるMFMを無効にする" | enableAdvancedMfm: "高度なMFMを有効にする" | ||||||
|  | enableAnimatedMfm: "動きのあるMFMを有効にする" | ||||||
| doing: "やっています" | doing: "やっています" | ||||||
| category: "カテゴリ" | category: "カテゴリ" | ||||||
| tags: "タグ" | tags: "タグ" | ||||||
| @@ -861,6 +867,8 @@ failedToFetchAccountInformation: "アカウント情報の取得に失敗しま | |||||||
| rateLimitExceeded: "レート制限を超えました" | rateLimitExceeded: "レート制限を超えました" | ||||||
| cropImage: "画像のクロップ" | cropImage: "画像のクロップ" | ||||||
| cropImageAsk: "画像をクロップしますか?" | cropImageAsk: "画像をクロップしますか?" | ||||||
|  | cropYes: "クロップする" | ||||||
|  | cropNo: "そのまま使う" | ||||||
| file: "ファイル" | file: "ファイル" | ||||||
| recentNHours: "直近{n}時間" | recentNHours: "直近{n}時間" | ||||||
| recentNDays: "直近{n}日" | recentNDays: "直近{n}日" | ||||||
| @@ -939,6 +947,16 @@ cannotPerformTemporaryDescription: "操作回数が制限を超過するため | |||||||
| preset: "プリセット" | preset: "プリセット" | ||||||
| selectFromPresets: "プリセットから選択" | selectFromPresets: "プリセットから選択" | ||||||
| achievements: "実績" | achievements: "実績" | ||||||
|  | gotInvalidResponseError: "サーバーの応答が無効です" | ||||||
|  | gotInvalidResponseErrorDescription: "サーバーがダウンまたはメンテナンスしている可能性があります。しばらくしてから再度お試しください。" | ||||||
|  | thisPostMayBeAnnoying: "この投稿は迷惑になる可能性があります。" | ||||||
|  | thisPostMayBeAnnoyingHome: "ホームに投稿" | ||||||
|  | thisPostMayBeAnnoyingCancel: "やめる" | ||||||
|  | thisPostMayBeAnnoyingIgnore: "このまま投稿" | ||||||
|  | collapseRenotes: "見たことのあるRenoteを省略して表示" | ||||||
|  | internalServerError: "サーバー内部エラー" | ||||||
|  | internalServerErrorDescription: "サーバー内部で予期しないエラーが発生しました。" | ||||||
|  | copyErrorInfo: "エラー情報をコピー" | ||||||
|  |  | ||||||
| _achievements: | _achievements: | ||||||
|   earnedAt: "獲得日時" |   earnedAt: "獲得日時" | ||||||
| @@ -1148,7 +1166,7 @@ _achievements: | |||||||
|       description: "ここをクリックした" |       description: "ここをクリックした" | ||||||
|     _justPlainLucky: |     _justPlainLucky: | ||||||
|       title: "単なるラッキー" |       title: "単なるラッキー" | ||||||
|       description: "10秒ごとに0.01%の確率で獲得" |       description: "10秒ごとに0.005%の確率で獲得" | ||||||
|     _setNameToSyuilo: |     _setNameToSyuilo: | ||||||
|       title: "神様コンプレックス" |       title: "神様コンプレックス" | ||||||
|       description: "名前を syuilo に設定した" |       description: "名前を syuilo に設定した" | ||||||
| @@ -1184,7 +1202,7 @@ _role: | |||||||
|   description: "ロールの説明" |   description: "ロールの説明" | ||||||
|   permission: "ロールの権限" |   permission: "ロールの権限" | ||||||
|   descriptionOfPermission: "<b>モデレーター</b>は基本的なモデレーションに関する操作を行えます。\n<b>管理者</b>はインスタンスの全ての設定を変更できます。" |   descriptionOfPermission: "<b>モデレーター</b>は基本的なモデレーションに関する操作を行えます。\n<b>管理者</b>はインスタンスの全ての設定を変更できます。" | ||||||
|   assignTarget: "アサインターゲット" |   assignTarget: "アサイン" | ||||||
|   descriptionOfAssignTarget: "<b>マニュアル</b>は誰がこのロールに含まれるかを手動で管理します。\n<b>コンディショナル</b>は条件を設定し、それに合致するユーザーが自動で含まれるようになります。" |   descriptionOfAssignTarget: "<b>マニュアル</b>は誰がこのロールに含まれるかを手動で管理します。\n<b>コンディショナル</b>は条件を設定し、それに合致するユーザーが自動で含まれるようになります。" | ||||||
|   manual: "マニュアル" |   manual: "マニュアル" | ||||||
|   conditional: "コンディショナル" |   conditional: "コンディショナル" | ||||||
| @@ -1197,6 +1215,9 @@ _role: | |||||||
|   baseRole: "ベースロール" |   baseRole: "ベースロール" | ||||||
|   useBaseValue: "ベースロールの値を使用" |   useBaseValue: "ベースロールの値を使用" | ||||||
|   chooseRoleToAssign: "アサインするロールを選択" |   chooseRoleToAssign: "アサインするロールを選択" | ||||||
|  |   iconUrl: "アイコン画像のURL" | ||||||
|  |   asBadge: "バッジとして表示" | ||||||
|  |   descriptionOfAsBadge: "オンにすると、ユーザー名の横にロールのアイコンが表示されます。" | ||||||
|   canEditMembersByModerator: "モデレーターのメンバー編集を許可" |   canEditMembersByModerator: "モデレーターのメンバー編集を許可" | ||||||
|   descriptionOfCanEditMembersByModerator: "オンにすると、管理者に加えてモデレーターもこのロールへユーザーをアサイン/アサイン解除できるようになります。オフにすると管理者のみが行えます。" |   descriptionOfCanEditMembersByModerator: "オンにすると、管理者に加えてモデレーターもこのロールへユーザーをアサイン/アサイン解除できるようになります。オフにすると管理者のみが行えます。" | ||||||
|   priority: "優先度" |   priority: "優先度" | ||||||
| @@ -1337,73 +1358,6 @@ _nsfw: | |||||||
|   ignore: "閲覧注意のメディアを隠さない" |   ignore: "閲覧注意のメディアを隠さない" | ||||||
|   force: "常にメディアを隠す" |   force: "常にメディアを隠す" | ||||||
|  |  | ||||||
| _mfm: |  | ||||||
|   cheatSheet: "MFMチートシート" |  | ||||||
|   intro: "MFMは、Misskey内の様々な場所で使用できる専用のマークアップ言語です。ここでは、MFMで使用可能な構文一覧が確認できます。" |  | ||||||
|   dummy: "MisskeyでFediverseの世界が広がります" |  | ||||||
|   mention: "メンション" |  | ||||||
|   mentionDescription: "アットマーク + ユーザー名で、特定のユーザーを示すことができます。" |  | ||||||
|   hashtag: "ハッシュタグ" |  | ||||||
|   hashtagDescription: "ナンバーサイン + タグで、ハッシュタグを示すことができます。" |  | ||||||
|   url: "URL" |  | ||||||
|   urlDescription: "URLを示すことができます。" |  | ||||||
|   link: "リンク" |  | ||||||
|   linkDescription: "文章の特定の範囲を、URLに紐づけることができます。" |  | ||||||
|   bold: "太字" |  | ||||||
|   boldDescription: "文字を太く表示して強調することができます。" |  | ||||||
|   small: "目立たなく" |  | ||||||
|   smallDescription: "内容を小さく・薄く表示させることができます。" |  | ||||||
|   center: "中央寄せ" |  | ||||||
|   centerDescription: "内容を中央寄せで表示させることができます。" |  | ||||||
|   inlineCode: "コード(インライン)" |  | ||||||
|   inlineCodeDescription: "プログラムなどのコードをインラインでシンタックスハイライトします。" |  | ||||||
|   blockCode: "コード(ブロック)" |  | ||||||
|   blockCodeDescription: "複数行のプログラムなどのコードをブロックでシンタックスハイライトします。" |  | ||||||
|   inlineMath: "数式(インライン)" |  | ||||||
|   inlineMathDescription: "数式(KaTeX)をインラインで表示します。" |  | ||||||
|   blockMath: "数式(ブロック)" |  | ||||||
|   blockMathDescription: "複数行の数式(KaTeX)をブロックで表示します。" |  | ||||||
|   quote: "引用" |  | ||||||
|   quoteDescription: "内容が引用であることを示すことができます。" |  | ||||||
|   emoji: "カスタム絵文字" |  | ||||||
|   emojiDescription: "コロンでカスタム絵文字名を囲むと、カスタム絵文字を表示させることができます。" |  | ||||||
|   search: "検索" |  | ||||||
|   searchDescription: "入力済み検索ボックスを表示させることができます。" |  | ||||||
|   flip: "反転" |  | ||||||
|   flipDescription: "内容を上下または左右に反転させます。" |  | ||||||
|   jelly: "アニメーション(びよんびよん)" |  | ||||||
|   jellyDescription: "びよんびよんするアニメーションを与えます。" |  | ||||||
|   tada: "アニメーション(じゃーん)" |  | ||||||
|   tadaDescription: "ジャーン!という感じのアニメーションを与えます。" |  | ||||||
|   jump: "アニメーション(ジャンプ)" |  | ||||||
|   jumpDescription: "飛び跳ねるようなアニメーションを与えます。" |  | ||||||
|   bounce: "アニメーション(バウンド)" |  | ||||||
|   bounceDescription: "ぽよんぽよん弾むようなアニメーションを与えます。" |  | ||||||
|   shake: "アニメーション(ぶるぶる)" |  | ||||||
|   shakeDescription: "ぶるぶる震えるアニメーションを与えます。" |  | ||||||
|   twitch: "アニメーション(ブレ)" |  | ||||||
|   twitchDescription: "激しくブレるアニメーションを与えます。" |  | ||||||
|   spin: "アニメーション(回転)" |  | ||||||
|   spinDescription: "回転するアニメーションを与えます。" |  | ||||||
|   x2: "大きく" |  | ||||||
|   x2Description: "内容を大きく表示します。" |  | ||||||
|   x3: "とても大きく" |  | ||||||
|   x3Description: "内容をとても大きく表示します。" |  | ||||||
|   x4: "究極に大きく" |  | ||||||
|   x4Description: "内容を究極に大きく表示します。" |  | ||||||
|   blur: "ぼかし" |  | ||||||
|   blurDescription: "内容をぼかすことができます。ポインターを上に乗せるとはっきり見えるようになります。" |  | ||||||
|   font: "フォント" |  | ||||||
|   fontDescription: "内容のフォントを指定することができます。" |  | ||||||
|   rainbow: "レインボー" |  | ||||||
|   rainbowDescription: "内容をレインボーにします。" |  | ||||||
|   sparkle: "キラキラ" |  | ||||||
|   sparkleDescription: "キラキラしたパーティクルのエフェクトを追加します。" |  | ||||||
|   rotate: "回転" |  | ||||||
|   rotateDescription: "指定した角度で回転させます。" |  | ||||||
|   plain: "プレーン" |  | ||||||
|   plainDescription: "内側の構文を全て無効にします。" |  | ||||||
|  |  | ||||||
| _instanceTicker: | _instanceTicker: | ||||||
|   none: "表示しない" |   none: "表示しない" | ||||||
|   remote: "リモートユーザーに表示" |   remote: "リモートユーザーに表示" | ||||||
| @@ -1622,12 +1576,15 @@ _permissions: | |||||||
|   "write:gallery-likes": "ギャラリーのいいねを操作する" |   "write:gallery-likes": "ギャラリーのいいねを操作する" | ||||||
|  |  | ||||||
| _auth: | _auth: | ||||||
|  |   shareAccessTitle: "アプリへのアクセス許可" | ||||||
|   shareAccess: "「{name}」がアカウントにアクセスすることを許可しますか?" |   shareAccess: "「{name}」がアカウントにアクセスすることを許可しますか?" | ||||||
|   shareAccessAsk: "アカウントへのアクセスを許可しますか?" |   shareAccessAsk: "アカウントへのアクセスを許可しますか?" | ||||||
|  |   permission: "{name}は次の権限を要求しています" | ||||||
|   permissionAsk: "このアプリは次の権限を要求しています" |   permissionAsk: "このアプリは次の権限を要求しています" | ||||||
|   pleaseGoBack: "アプリケーションに戻ってやっていってください" |   pleaseGoBack: "アプリケーションに戻ってやっていってください" | ||||||
|   callback: "アプリケーションに戻っています" |   callback: "アプリケーションに戻っています" | ||||||
|   denied: "アクセスを拒否しました" |   denied: "アクセスを拒否しました" | ||||||
|  |   pleaseLogin: "アプリケーションにアクセス許可を与えるには、ログインが必要です。" | ||||||
|  |  | ||||||
| _antennaSources: | _antennaSources: | ||||||
|   all: "全てのノート" |   all: "全てのノート" | ||||||
| @@ -1919,5 +1876,6 @@ _deck: | |||||||
|     tl: "タイムライン" |     tl: "タイムライン" | ||||||
|     antenna: "アンテナ" |     antenna: "アンテナ" | ||||||
|     list: "リスト" |     list: "リスト" | ||||||
|  |     channel: "チャンネル" | ||||||
|     mentions: "あなた宛て" |     mentions: "あなた宛て" | ||||||
|     direct: "ダイレクト" |     direct: "ダイレクト" | ||||||
|   | |||||||
| @@ -46,7 +46,7 @@ copyContent: "内容をコピー" | |||||||
| copyLink: "リンクをコピー" | copyLink: "リンクをコピー" | ||||||
| delete: "ほかす" | delete: "ほかす" | ||||||
| deleteAndEdit: "ほかして直す" | deleteAndEdit: "ほかして直す" | ||||||
| deleteAndEditConfirm: "このノートをほかして書き直すんか?このノートへのリアクション、Renote、返信も全部消えてまうで。" | deleteAndEditConfirm: "このノートをほかしてもっかい直す?このノートへのリアクション、Renote、返信も全部消えるんやけどそれでもええん?" | ||||||
| addToList: "リストに入れたる" | addToList: "リストに入れたる" | ||||||
| sendMessage: "メッセージを送る" | sendMessage: "メッセージを送る" | ||||||
| copyRSS: "RSSをコピー" | copyRSS: "RSSをコピー" | ||||||
| @@ -89,7 +89,7 @@ serverIsDead: "サーバーからの応答がないで。もうちょい待っ | |||||||
| youShouldUpgradeClient: "このページを表示するには、リロードして新しいバージョンのクライアントを使ってなー。" | youShouldUpgradeClient: "このページを表示するには、リロードして新しいバージョンのクライアントを使ってなー。" | ||||||
| enterListName: "リスト名を入れてや" | enterListName: "リスト名を入れてや" | ||||||
| privacy: "プライバシー" | privacy: "プライバシー" | ||||||
| makeFollowManuallyApprove: "自分が認めた人だけがこのアカウントをフォローできるようにする" | makeFollowManuallyApprove: "他人のフォローは許可してからや!" | ||||||
| defaultNoteVisibility: "もとからの公開範囲" | defaultNoteVisibility: "もとからの公開範囲" | ||||||
| follow: "フォロー" | follow: "フォロー" | ||||||
| followRequest: "フォローを頼む" | followRequest: "フォローを頼む" | ||||||
| @@ -129,6 +129,7 @@ unblockConfirm: "ブロックやめたるってほんまか?" | |||||||
| suspendConfirm: "凍結してしもうてええか?" | suspendConfirm: "凍結してしもうてええか?" | ||||||
| unsuspendConfirm: "解凍するけどええか?" | unsuspendConfirm: "解凍するけどええか?" | ||||||
| selectList: "リストを選ぶ" | selectList: "リストを選ぶ" | ||||||
|  | selectChannel: "チャンネルを選ぶ" | ||||||
| selectAntenna: "アンテナを選ぶ" | selectAntenna: "アンテナを選ぶ" | ||||||
| selectWidget: "ウィジェットを選ぶ" | selectWidget: "ウィジェットを選ぶ" | ||||||
| editWidgets: "ウィジェットをいじる" | editWidgets: "ウィジェットをいじる" | ||||||
| @@ -256,6 +257,8 @@ noMoreHistory: "これより過去の履歴はあらへんで" | |||||||
| startMessaging: "チャットやるで" | startMessaging: "チャットやるで" | ||||||
| nUsersRead: "{n}人が読んでもうた" | nUsersRead: "{n}人が読んでもうた" | ||||||
| agreeTo: "{0}に同意したで" | agreeTo: "{0}に同意したで" | ||||||
|  | agreeBelow: "下記に同意したる" | ||||||
|  | basicNotesBeforeCreateAccount: "よう読んでやってや" | ||||||
| tos: "利用規約" | tos: "利用規約" | ||||||
| start: "始める" | start: "始める" | ||||||
| home: "ホーム" | home: "ホーム" | ||||||
| @@ -300,7 +303,7 @@ avatar: "アイコン" | |||||||
| banner: "バナー" | banner: "バナー" | ||||||
| nsfw: "閲覧注意" | nsfw: "閲覧注意" | ||||||
| whenServerDisconnected: "サーバーとの接続が切れたとき" | whenServerDisconnected: "サーバーとの接続が切れたとき" | ||||||
| disconnectedFromServer: "サーバーとの通信が切れたで" | disconnectedFromServer: "サーバーが機嫌悪いねん" | ||||||
| reload: "リロード" | reload: "リロード" | ||||||
| doNothing: "何もせんとく" | doNothing: "何もせんとく" | ||||||
| reloadConfirm: "リロードしてええか?" | reloadConfirm: "リロードしてええか?" | ||||||
| @@ -464,7 +467,8 @@ youHaveNoGroups: "グループがあらへんねぇ。" | |||||||
| joinOrCreateGroup: "既存のグループに招待してもらうか、新しくグループ作ってからやってな" | joinOrCreateGroup: "既存のグループに招待してもらうか、新しくグループ作ってからやってな" | ||||||
| noHistory: "履歴はあらへんねぇ。" | noHistory: "履歴はあらへんねぇ。" | ||||||
| signinHistory: "ログイン履歴" | signinHistory: "ログイン履歴" | ||||||
| disableAnimatedMfm: "動きがやかましいMFMを止める" | enableAdvancedMfm: "ややこしいMFMもありにする" | ||||||
|  | enableAnimatedMfm: "動きがやかましいMFMも許したる" | ||||||
| doing: "やっとるがな" | doing: "やっとるがな" | ||||||
| category: "カテゴリ" | category: "カテゴリ" | ||||||
| tags: "タグ" | tags: "タグ" | ||||||
| @@ -673,8 +677,8 @@ sentReactionsCount: "リアクションした数やで" | |||||||
| receivedReactionsCount: "リアクションされた数" | receivedReactionsCount: "リアクションされた数" | ||||||
| pollVotesCount: "アンケートに投票した数" | pollVotesCount: "アンケートに投票した数" | ||||||
| pollVotedCount: "アンケートに投票された数" | pollVotedCount: "アンケートに投票された数" | ||||||
| yes: "はい" | yes: "ええで" | ||||||
| no: "いいえ" | no: "あかんで" | ||||||
| driveFilesCount: "ドライブのファイル数" | driveFilesCount: "ドライブのファイル数" | ||||||
| driveUsage: "ドライブ使用量やで" | driveUsage: "ドライブ使用量やで" | ||||||
| noCrawle: "クローラーによるインデックスを拒否するで" | noCrawle: "クローラーによるインデックスを拒否するで" | ||||||
| @@ -861,6 +865,8 @@ failedToFetchAccountInformation: "アカウントの取得に失敗したみた | |||||||
| rateLimitExceeded: "レート制限が超えたみたいやで" | rateLimitExceeded: "レート制限が超えたみたいやで" | ||||||
| cropImage: "画像のクロップ" | cropImage: "画像のクロップ" | ||||||
| cropImageAsk: "画像をクロップしたってええか?" | cropImageAsk: "画像をクロップしたってええか?" | ||||||
|  | cropYes: "切り抜いたる" | ||||||
|  | cropNo: "切り抜かへん" | ||||||
| file: "ファイル" | file: "ファイル" | ||||||
| recentNHours: "直近{n}時間" | recentNHours: "直近{n}時間" | ||||||
| recentNDays: "直近{n}日" | recentNDays: "直近{n}日" | ||||||
| @@ -938,6 +944,39 @@ cannotPerformTemporary: "一時的に利用できへんで" | |||||||
| cannotPerformTemporaryDescription: "操作回数が制限を超えたから一時的に利用できへんくなったで。ちょっと時間置いてからもう一回やってやー。" | cannotPerformTemporaryDescription: "操作回数が制限を超えたから一時的に利用できへんくなったで。ちょっと時間置いてからもう一回やってやー。" | ||||||
| preset: "プリセット" | preset: "プリセット" | ||||||
| selectFromPresets: "プリセットから選ぶ" | selectFromPresets: "プリセットから選ぶ" | ||||||
|  | achievements: "実績" | ||||||
|  | gotInvalidResponseError: "サーバー黙っとるわ、知らんけど" | ||||||
|  | gotInvalidResponseErrorDescription: "サーバーいま日曜日。またきて月曜日。" | ||||||
|  | thisPostMayBeAnnoying: "この投稿は迷惑かもしらんで。" | ||||||
|  | collapseRenotes: "見たことあるRenoteは省略やで" | ||||||
|  | _achievements: | ||||||
|  |   earnedAt: "貰った日ぃ" | ||||||
|  |   _types: | ||||||
|  |     _notes1: | ||||||
|  |       title: "まいど!" | ||||||
|  |       description: "初めてノート投稿したった" | ||||||
|  |     _notes10: | ||||||
|  |       title: "ノートの天保山" | ||||||
|  |     _notes100: | ||||||
|  |       title: "ノートの真田山" | ||||||
|  |     _notes500: | ||||||
|  |       title: "ノートの生駒山" | ||||||
|  |     _notes5000: | ||||||
|  |       title: "箕面の滝からノート" | ||||||
|  |     _login3: | ||||||
|  |       flavor: "今日からワシはミスキストやで" | ||||||
|  |     _iLoveMisskey: | ||||||
|  |       title: "Misskey好きやねん" | ||||||
|  |     _foundTreasure: | ||||||
|  |       title: "なんでも鑑定団" | ||||||
|  |     _client30min: | ||||||
|  |       title: "ねんね" | ||||||
|  |     _noteDeletedWithin1min: | ||||||
|  |       title: "*おおっと*" | ||||||
|  |     _open3windows: | ||||||
|  |       title: "マド開けすぎ" | ||||||
|  |     _driveFolderCircularReference: | ||||||
|  |       title: "環状線" | ||||||
| _role: | _role: | ||||||
|   new: "ロールの作成" |   new: "ロールの作成" | ||||||
|   edit: "ロールの編集" |   edit: "ロールの編集" | ||||||
| @@ -1083,72 +1122,6 @@ _nsfw: | |||||||
|   respect: "閲覧注意のメディアは隠すで" |   respect: "閲覧注意のメディアは隠すで" | ||||||
|   ignore: "閲覧注意のメディアは隠さへんで" |   ignore: "閲覧注意のメディアは隠さへんで" | ||||||
|   force: "常にメディアを隠すで" |   force: "常にメディアを隠すで" | ||||||
| _mfm: |  | ||||||
|   cheatSheet: "MFMチートシート" |  | ||||||
|   intro: "MFMは、Misskey内の色んな所で使える専用のマークアップ言語やで。このページでMFMで使える構文一覧が確認できるで。" |  | ||||||
|   dummy: "MisskeyでFediverseの世界が広がります" |  | ||||||
|   mention: "メンション" |  | ||||||
|   mentionDescription: "アットマーク + ユーザー名で、特定のユーザーを示すことができるで。" |  | ||||||
|   hashtag: "ハッシュタグ" |  | ||||||
|   hashtagDescription: "ナンバーサイン + タグで、ハッシュタグを示すことができるで。" |  | ||||||
|   url: "URL" |  | ||||||
|   urlDescription: "URLを示すことができるで。" |  | ||||||
|   link: "リンク" |  | ||||||
|   linkDescription: "文章の特定の範囲をURLに紐づけることができるで" |  | ||||||
|   bold: "太字" |  | ||||||
|   boldDescription: "文字を太く表示して強調することができるで" |  | ||||||
|   small: "目立たなく" |  | ||||||
|   smallDescription: "内容を小さく・薄く表示することができるで" |  | ||||||
|   center: "中央寄せ" |  | ||||||
|   centerDescription: "内容を中央寄せで表示することができるで" |  | ||||||
|   inlineCode: "コード(インライン)" |  | ||||||
|   inlineCodeDescription: "プログラムとかのコードをインラインでシンタックスハイライトするで" |  | ||||||
|   blockCode: "コード(ブロック)" |  | ||||||
|   blockCodeDescription: "複数行のプログラムとかのコードをブロックでシンタックスハイライトするで" |  | ||||||
|   inlineMath: "数式(インライン)" |  | ||||||
|   inlineMathDescription: "数式(KaTeX)をインラインで表示するで" |  | ||||||
|   blockMath: "数式(ブロック)" |  | ||||||
|   blockMathDescription: "複数行の数式(KaTeX)をブロックで表示するで" |  | ||||||
|   quote: "引用" |  | ||||||
|   quoteDescription: "内容が引用ってことを示すことができるで" |  | ||||||
|   emoji: "カスタム絵文字" |  | ||||||
|   emojiDescription: "コロンでカスタム絵文字名を囲んだると、カスタム絵文字を表示させることができるで" |  | ||||||
|   search: "探す" |  | ||||||
|   searchDescription: "入力済み検索ボックスを表示することができるで" |  | ||||||
|   flip: "反転" |  | ||||||
|   flipDescription: "内容を上下または左右に反転するで" |  | ||||||
|   jelly: "アニメーション(びよんびよん)" |  | ||||||
|   jellyDescription: "びよんびよんするアニメーションやな。" |  | ||||||
|   tada: "アニメーション(じゃーん)" |  | ||||||
|   tadaDescription: "ジャーン!ってな感じのアニメーションやな。" |  | ||||||
|   jump: "アニメーション(ジャンプ)" |  | ||||||
|   jumpDescription: "飛び跳ねるようなアニメーションやな。" |  | ||||||
|   bounce: "アニメーション(バウンド)" |  | ||||||
|   bounceDescription: "ぽよんぽよん弾むようなアニメーションやな。" |  | ||||||
|   shake: "アニメーション(ぶるぶる)" |  | ||||||
|   shakeDescription: "ぶるぶる震えるアニメーションやな。" |  | ||||||
|   twitch: "アニメーション(ブレ)" |  | ||||||
|   twitchDescription: "激しくブレるアニメーションやな。" |  | ||||||
|   spin: "アニメーション(回転)" |  | ||||||
|   spinDescription: "回転するアニメーションやな。" |  | ||||||
|   x2: "大きく" |  | ||||||
|   x2Description: "内容を大きく表示するで" |  | ||||||
|   x3: "とても大きく" |  | ||||||
|   x3Description: "内容をとても大きく表示するで" |  | ||||||
|   x4: "究極に大きく" |  | ||||||
|   x4Description: "内容を究極に大きく表示するで" |  | ||||||
|   blur: "ぼかし" |  | ||||||
|   blurDescription: "内容をぼかすことができるで。ポインターを上に乗せるとはっきり見えるようになるで" |  | ||||||
|   font: "フォント" |  | ||||||
|   fontDescription: "内容のフォントを指定することができるで" |  | ||||||
|   rainbow: "レインボー" |  | ||||||
|   rainbowDescription: "内容をレインボーにするで" |  | ||||||
|   sparkle: "キラキラ" |  | ||||||
|   sparkleDescription: "キラキラしたバーティ来るのエフェクトを追加するで" |  | ||||||
|   rotate: "回転" |  | ||||||
|   rotateDescription: "指定した角度で回転させるで" |  | ||||||
|   plain: "プレーン" |  | ||||||
|   plainDescription: "内側の構文を全部無効にするで" |  | ||||||
| _instanceTicker: | _instanceTicker: | ||||||
|   none: "表示せん" |   none: "表示せん" | ||||||
|   remote: "リモートユーザーに表示" |   remote: "リモートユーザーに表示" | ||||||
| @@ -1355,10 +1328,12 @@ _permissions: | |||||||
| _auth: | _auth: | ||||||
|   shareAccess: "「{name}」がアカウントにアクセスすることを許可してええか?" |   shareAccess: "「{name}」がアカウントにアクセスすることを許可してええか?" | ||||||
|   shareAccessAsk: "アカウントのアクセスを許可してもええか?" |   shareAccessAsk: "アカウントのアクセスを許可してもええか?" | ||||||
|  |   permission: "{name}に次の権限つけたってやって" | ||||||
|   permissionAsk: "このアプリは次の権限を要求しとるで" |   permissionAsk: "このアプリは次の権限を要求しとるで" | ||||||
|   pleaseGoBack: "アプリケーションに戻ってええよ" |   pleaseGoBack: "アプリケーションに戻ってええよ" | ||||||
|   callback: "アプリケーションに戻っとるで" |   callback: "アプリケーションに戻っとるで" | ||||||
|   denied: "アクセスを拒否ったで" |   denied: "アクセスを拒否ったで" | ||||||
|  |   pleaseLogin: "アプリにアクセスさせるんやったら、ログインしてや。" | ||||||
| _antennaSources: | _antennaSources: | ||||||
|   all: "みんなのノート" |   all: "みんなのノート" | ||||||
|   homeTimeline: "フォローしとるユーザーのノート" |   homeTimeline: "フォローしとるユーザーのノート" | ||||||
| @@ -1587,6 +1562,7 @@ _notification: | |||||||
|   pollEnded: "アンケートの結果が出たみたいや" |   pollEnded: "アンケートの結果が出たみたいや" | ||||||
|   unreadAntennaNote: "アンテナ {name}" |   unreadAntennaNote: "アンテナ {name}" | ||||||
|   emptyPushNotificationMessage: "プッシュ通知の更新をしといたで" |   emptyPushNotificationMessage: "プッシュ通知の更新をしといたで" | ||||||
|  |   achievementEarned: "実績を獲得しとるで" | ||||||
|   _types: |   _types: | ||||||
|     all: "すべて" |     all: "すべて" | ||||||
|     follow: "フォロー" |     follow: "フォロー" | ||||||
| @@ -1628,5 +1604,6 @@ _deck: | |||||||
|     tl: "タイムライン" |     tl: "タイムライン" | ||||||
|     antenna: "アンテナ" |     antenna: "アンテナ" | ||||||
|     list: "リスト" |     list: "リスト" | ||||||
|  |     channel: "チャンネル" | ||||||
|     mentions: "あんた宛て" |     mentions: "あんた宛て" | ||||||
|     direct: "ダイレクト" |     direct: "ダイレクト" | ||||||
|   | |||||||
| @@ -61,10 +61,6 @@ account: "Imiḍan" | |||||||
| _email: | _email: | ||||||
|   _follow: |   _follow: | ||||||
|     title: "Yeṭṭafaṛ-ik·em-id" |     title: "Yeṭṭafaṛ-ik·em-id" | ||||||
| _mfm: |  | ||||||
|   mention: "Bder" |  | ||||||
|   search: "Nadi" |  | ||||||
|   font: "Tasefsit" |  | ||||||
| _theme: | _theme: | ||||||
|   keys: |   keys: | ||||||
|     mention: "Bder" |     mention: "Bder" | ||||||
|   | |||||||
| @@ -64,8 +64,6 @@ file: "ಕಡತಗಳು" | |||||||
| _email: | _email: | ||||||
|   _follow: |   _follow: | ||||||
|     title: "ಹಿಂಬಾಲಿಸಿದರು" |     title: "ಹಿಂಬಾಲಿಸಿದರು" | ||||||
| _mfm: |  | ||||||
|   search: "ಹುಡುಕು" |  | ||||||
| _sfx: | _sfx: | ||||||
|   notification: "ಅಧಿಸೂಚನೆಗಳು" |   notification: "ಅಧಿಸೂಚನೆಗಳು" | ||||||
| _widgets: | _widgets: | ||||||
|   | |||||||
| @@ -129,6 +129,7 @@ unblockConfirm: "이 계정의 차단을 해제하시겠습니까?" | |||||||
| suspendConfirm: "이 계정을 정지하시겠습니까?" | suspendConfirm: "이 계정을 정지하시겠습니까?" | ||||||
| unsuspendConfirm: "이 계정의 정지를 해제하시겠습니까?" | unsuspendConfirm: "이 계정의 정지를 해제하시겠습니까?" | ||||||
| selectList: "리스트 선택" | selectList: "리스트 선택" | ||||||
|  | selectChannel: "채널 선택" | ||||||
| selectAntenna: "안테나 선택" | selectAntenna: "안테나 선택" | ||||||
| selectWidget: "위젯 선택" | selectWidget: "위젯 선택" | ||||||
| editWidgets: "위젯 편집" | editWidgets: "위젯 편집" | ||||||
| @@ -256,6 +257,8 @@ noMoreHistory: "이것보다 과거의 기록이 없습니다" | |||||||
| startMessaging: "대화 시작하기" | startMessaging: "대화 시작하기" | ||||||
| nUsersRead: "{n}명이 읽음" | nUsersRead: "{n}명이 읽음" | ||||||
| agreeTo: "{0}에 동의" | agreeTo: "{0}에 동의" | ||||||
|  | agreeBelow: "아래 내용에 동의합니다" | ||||||
|  | basicNotesBeforeCreateAccount: "기본적인 주의사항" | ||||||
| tos: "이용 약관" | tos: "이용 약관" | ||||||
| start: "시작하기" | start: "시작하기" | ||||||
| home: "홈" | home: "홈" | ||||||
| @@ -464,7 +467,8 @@ youHaveNoGroups: "그룹이 없습니다" | |||||||
| joinOrCreateGroup: "다른 그룹의 초대를 받거나, 직접 새 그룹을 만들어 보세요." | joinOrCreateGroup: "다른 그룹의 초대를 받거나, 직접 새 그룹을 만들어 보세요." | ||||||
| noHistory: "기록이 없습니다" | noHistory: "기록이 없습니다" | ||||||
| signinHistory: "로그인 기록" | signinHistory: "로그인 기록" | ||||||
| disableAnimatedMfm: "움직임이 있는 MFM을 비활성화" | enableAdvancedMfm: "고급 MFM을 활성화" | ||||||
|  | enableAnimatedMfm: "움직임이 있는 MFM을 활성화" | ||||||
| doing: "잠시만요" | doing: "잠시만요" | ||||||
| category: "카테고리" | category: "카테고리" | ||||||
| tags: "태그" | tags: "태그" | ||||||
| @@ -861,6 +865,8 @@ failedToFetchAccountInformation: "계정 정보를 가져오지 못했습니다" | |||||||
| rateLimitExceeded: "요청 제한 횟수를 초과하였습니다" | rateLimitExceeded: "요청 제한 횟수를 초과하였습니다" | ||||||
| cropImage: "이미지 자르기" | cropImage: "이미지 자르기" | ||||||
| cropImageAsk: "이미지를 자르시겠습니까?" | cropImageAsk: "이미지를 자르시겠습니까?" | ||||||
|  | cropYes: "잘라내기" | ||||||
|  | cropNo: "그대로 사용" | ||||||
| file: "파일" | file: "파일" | ||||||
| recentNHours: "최근 {n}시간" | recentNHours: "최근 {n}시간" | ||||||
| recentNDays: "최근 {n}일" | recentNDays: "최근 {n}일" | ||||||
| @@ -939,6 +945,12 @@ cannotPerformTemporaryDescription: "조작 횟수 제한을 초과하여 일시 | |||||||
| preset: "프리셋" | preset: "프리셋" | ||||||
| selectFromPresets: "프리셋에서 선택" | selectFromPresets: "프리셋에서 선택" | ||||||
| achievements: "도전 과제" | achievements: "도전 과제" | ||||||
|  | gotInvalidResponseError: "서버의 응답이 올바르지 않습니다" | ||||||
|  | gotInvalidResponseErrorDescription: " 서버가 다운되었거나 점검중일 가능성이 있습니다. 잠시후에 다시 시도해 주십시오." | ||||||
|  | thisPostMayBeAnnoying: "이 게시물은 다른 유저에게 피해를 줄 가능성이 있습니다." | ||||||
|  | thisPostMayBeAnnoyingHome: "홈에 게시" | ||||||
|  | thisPostMayBeAnnoyingCancel: "그만두기" | ||||||
|  | thisPostMayBeAnnoyingIgnore: "이대로 게시" | ||||||
| _achievements: | _achievements: | ||||||
|   earnedAt: "달성 일시" |   earnedAt: "달성 일시" | ||||||
|   _types: |   _types: | ||||||
| @@ -1195,6 +1207,9 @@ _role: | |||||||
|   baseRole: "기본 역할" |   baseRole: "기본 역할" | ||||||
|   useBaseValue: "기본값 사용" |   useBaseValue: "기본값 사용" | ||||||
|   chooseRoleToAssign: "할당할 역할 선택" |   chooseRoleToAssign: "할당할 역할 선택" | ||||||
|  |   iconUrl: "아이콘 URL" | ||||||
|  |   asBadge: "뱃지로 표시" | ||||||
|  |   descriptionOfAsBadge: "활성화하면 유저명 옆에 역할의 아이콘이 표시됩니다." | ||||||
|   canEditMembersByModerator: "모더레이터의 역할 수정 허용" |   canEditMembersByModerator: "모더레이터의 역할 수정 허용" | ||||||
|   descriptionOfCanEditMembersByModerator: "이 옵션을 켜면 모더레이터도 이 역할에 사용자를 할당하거나 삭제할 수 있습니다. 꺼져 있으면 관리자만 할당이 가능합니다." |   descriptionOfCanEditMembersByModerator: "이 옵션을 켜면 모더레이터도 이 역할에 사용자를 할당하거나 삭제할 수 있습니다. 꺼져 있으면 관리자만 할당이 가능합니다." | ||||||
|   priority: "우선순위" |   priority: "우선순위" | ||||||
| @@ -1320,72 +1335,6 @@ _nsfw: | |||||||
|   respect: "열람주의로 설정된 미디어 숨기기" |   respect: "열람주의로 설정된 미디어 숨기기" | ||||||
|   ignore: "열람 주의 미디어 항상 표시" |   ignore: "열람 주의 미디어 항상 표시" | ||||||
|   force: "미디어 항상 숨기기" |   force: "미디어 항상 숨기기" | ||||||
| _mfm: |  | ||||||
|   cheatSheet: "MFM 도움말" |  | ||||||
|   intro: "MFM는 Misskey의 다양한 곳에서 사용할 수 있는 전용 마크업 언어입니다. 여기에서는 MFM에서 사용할 수 있는 구문을 확인할 수 있습니다." |  | ||||||
|   dummy: "Misskey로 연합우주의 세계가 펼쳐집니다" |  | ||||||
|   mention: "멘션" |  | ||||||
|   mentionDescription: "골뱅이표(@) 뒤에 사용자명을 넣어 특정 유저를 나타낼 수 있습니다." |  | ||||||
|   hashtag: "해시태그" |  | ||||||
|   hashtagDescription: "샵 또는 우물정자(#)를 앞에 붙여서 해시태그를 나타낼 수 있습니다." |  | ||||||
|   url: "URL" |  | ||||||
|   urlDescription: "URL을 나타낼 수 있습니다." |  | ||||||
|   link: "링크" |  | ||||||
|   linkDescription: "문장의 특정 범위를 URL로 표시합니다." |  | ||||||
|   bold: "굵음/볼드체" |  | ||||||
|   boldDescription: "문자를 굵게 강조합니다." |  | ||||||
|   small: "눈에 띄지 않음" |  | ||||||
|   smallDescription: "내용을 작고 연하게 보이게 합니다." |  | ||||||
|   center: "가운데 정렬" |  | ||||||
|   centerDescription: "내용을 가운데 정렬로 보이게 합니다." |  | ||||||
|   inlineCode: "코드(인라인)" |  | ||||||
|   inlineCodeDescription: "여러 행의 코드를 문법 강조를 적용하여 인라인으로 표시합니다." |  | ||||||
|   blockCode: "코드(블록)" |  | ||||||
|   blockCodeDescription: "여러 행의 코드를 문법 강조를 적용하여 블록으로 표시합니다." |  | ||||||
|   inlineMath: "수식(인라인)" |  | ||||||
|   inlineMathDescription: "수식(KaTeX)를 인라인으로 보이게 합니다." |  | ||||||
|   blockMath: "수식(블록)" |  | ||||||
|   blockMathDescription: "여러 줄의 수식(KaTeX)를 블록으로 보이게 합니다." |  | ||||||
|   quote: "인용" |  | ||||||
|   quoteDescription: "내용을 인용문으로 표시합니다." |  | ||||||
|   emoji: "커스텀 이모지" |  | ||||||
|   emojiDescription: "커스텀 이모지의 이름을 쌍점(:)으로 감싸서 커스텀 이모지를 사용합니다." |  | ||||||
|   search: "검색" |  | ||||||
|   searchDescription: "주어진 키워드가 입력된 검색창을 보이게 합니다." |  | ||||||
|   flip: "플립" |  | ||||||
|   flipDescription: "내용을 상하 또는 좌우로 반전시킵니다." |  | ||||||
|   jelly: "애니메이션 (젤리)" |  | ||||||
|   jellyDescription: "젤리처럼 탱글탱글한 느낌의 효과를 줍니다." |  | ||||||
|   tada: "애니메이션 (짠!)" |  | ||||||
|   tadaDescription: "짠! 하는 느낌의 효과를 줍니다." |  | ||||||
|   jump: "애니메이션(점프)" |  | ||||||
|   jumpDescription: "펄쩍 뛸 듯한 느낌의 효과를 줍니다." |  | ||||||
|   bounce: "애니메이션 (바운스)" |  | ||||||
|   bounceDescription: "통통 튀는 느낌의 효과를 줍니다." |  | ||||||
|   shake: "애니메이션 (부들부들)" |  | ||||||
|   shakeDescription: "부들부들 떠는 느낌의 효과를 줍니다." |  | ||||||
|   twitch: "애니메이션 (경련)" |  | ||||||
|   twitchDescription: "격하게 흔들리는 느낌의 효과를 줍니다." |  | ||||||
|   spin: "애니메이션 (회전)" |  | ||||||
|   spinDescription: "회전 효과를 줍니다." |  | ||||||
|   x2: "크게" |  | ||||||
|   x2Description: "내용을 크게 표시합니다." |  | ||||||
|   x3: "더 크게" |  | ||||||
|   x3Description: "내용을 더 크게 표시합니다." |  | ||||||
|   x4: "매우 크게" |  | ||||||
|   x4Description: "내용을 매우 크게 표시합니다." |  | ||||||
|   blur: "흐림" |  | ||||||
|   blurDescription: "내용이 흐리게 보입니다. 마우스를 위에 올려두면 내용이 보입니다." |  | ||||||
|   font: "폰트" |  | ||||||
|   fontDescription: "내용의 글꼴을 지정할 수 있습니다." |  | ||||||
|   rainbow: "무지개" |  | ||||||
|   rainbowDescription: "내용을 무지개로 표시합니다." |  | ||||||
|   sparkle: "반짝반짝" |  | ||||||
|   sparkleDescription: "반짝이는 파티클 효과를 추가합니다." |  | ||||||
|   rotate: "회전" |  | ||||||
|   rotateDescription: "지정한 각도로 회전시킵니다." |  | ||||||
|   plain: "평문" |  | ||||||
|   plainDescription: "안에 있는 MFM 구문을 모두 무시하고 평문으로 표시합니다." |  | ||||||
| _instanceTicker: | _instanceTicker: | ||||||
|   none: "보이지 않음" |   none: "보이지 않음" | ||||||
|   remote: "리모트 유저에게만 보이기" |   remote: "리모트 유저에게만 보이기" | ||||||
| @@ -1590,12 +1539,15 @@ _permissions: | |||||||
|   "read:gallery-likes": "갤러리의 좋아요를 확인합니다" |   "read:gallery-likes": "갤러리의 좋아요를 확인합니다" | ||||||
|   "write:gallery-likes": "갤러리에 좋아요를 추가하거나 취소합니다" |   "write:gallery-likes": "갤러리에 좋아요를 추가하거나 취소합니다" | ||||||
| _auth: | _auth: | ||||||
|  |   shareAccessTitle: "어플리케이션의 접근 허가" | ||||||
|   shareAccess: "\"{name}\" 이 계정에 접근하는 것을 허용하시겠습니까?" |   shareAccess: "\"{name}\" 이 계정에 접근하는 것을 허용하시겠습니까?" | ||||||
|   shareAccessAsk: "이 애플리케이션이 계정에 접근하는 것을 허용하시겠습니까?" |   shareAccessAsk: "이 애플리케이션이 계정에 접근하는 것을 허용하시겠습니까?" | ||||||
|  |   permission: "{name}에서 다음 권한을 요청하였습니다" | ||||||
|   permissionAsk: "이 앱은 다음의 권한을 요청합니다" |   permissionAsk: "이 앱은 다음의 권한을 요청합니다" | ||||||
|   pleaseGoBack: "앱으로 돌아가서 시도해 주세요" |   pleaseGoBack: "앱으로 돌아가서 시도해 주세요" | ||||||
|   callback: "앱으로 돌아갑니다" |   callback: "앱으로 돌아갑니다" | ||||||
|   denied: "접근이 거부되었습니다" |   denied: "접근이 거부되었습니다" | ||||||
|  |   pleaseLogin: "어플리케이션의 접근을 허가하려면 로그인하십시오." | ||||||
| _antennaSources: | _antennaSources: | ||||||
|   all: "모든 노트" |   all: "모든 노트" | ||||||
|   homeTimeline: "팔로우중인 유저의 노트" |   homeTimeline: "팔로우중인 유저의 노트" | ||||||
| @@ -1866,5 +1818,6 @@ _deck: | |||||||
|     tl: "타임라인" |     tl: "타임라인" | ||||||
|     antenna: "안테나" |     antenna: "안테나" | ||||||
|     list: "리스트" |     list: "리스트" | ||||||
|  |     channel: "채널" | ||||||
|     mentions: "받은 멘션" |     mentions: "받은 멘션" | ||||||
|     direct: "다이렉트" |     direct: "다이렉트" | ||||||
|   | |||||||
							
								
								
									
										239
									
								
								locales/lo-LA.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										239
									
								
								locales/lo-LA.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,239 @@ | |||||||
|  | --- | ||||||
|  | _lang_: "ພາສາລາວ" | ||||||
|  | headlineMisskey: "ເຊື່ອມຕໍ່ເຄືອຂ່າຍໂດຍຫມາຍເຫດ" | ||||||
|  | introMisskey: "ຍິນດີຕ້ອນຮັບ! Misskey ເປັນແຫຼ່ງເປີດ, ການບໍລິການ microblogging ກະຈາຍ\nສ້າງ \"ບັນທຶກ\" ເພື່ອແບ່ງປັນຄວາມຄິດຂອງທ່ານກັບທຸກໆຄົນທີ່ຢູ່ອ້ອມຮອບທ່ານ 📡\nດ້ວຍ \"ປະຕິກິລິຍາ\", ທ່ານຍັງສາມາດສະແດງຄວາມຮູ້ສຶກຂອງທ່ານຢ່າງໄວວາກ່ຽວກັບບັນທຶກຂອງທຸກໆຄົນ 👍\nມາສຳຫຼວດໂລກໃໝ່! 🚀" | ||||||
|  | poweredByMisskeyDescription: "{name} ແມ່ນສ່ວນໜຶ່ງຂອງການບໍລິການທີ່ຂັບເຄື່ອນໂດຍແພລດຟອມ open source. <b>Misskey</b> (ເອີ້ນວ່າ \"Misskey instance\")" | ||||||
|  | monthAndDay: "{ເດືອນ}/{ມື້}" | ||||||
|  | search: "ຄົ້ນຫາ" | ||||||
|  | notifications: "ການແຈ້ງເຕືອນ" | ||||||
|  | username: "ຊື່ຜູ້ໃຊ້" | ||||||
|  | password: "ລະຫັດຜ່ານ" | ||||||
|  | forgotPassword: "ລືມລະຫັດຜ່ານ" | ||||||
|  | fetchingAsApObject: "ກຳລັງດຶງຂໍ້ມູນຈາກ fediverse..." | ||||||
|  | ok: "ຕົກລົງ" | ||||||
|  | gotIt: "ເຂົ້າໃຈແລ້ວ!" | ||||||
|  | cancel: "ຍົກເລີກ" | ||||||
|  | noThankYou: "ບໍ່ແມ່ນຕອນນີ້" | ||||||
|  | enterUsername: "ປ້ອນຊື່ຜູ້ໃຊ້" | ||||||
|  | renotedBy: "Renoted ໂດຍ {ຜູ້ໃຊ້}" | ||||||
|  | noNotes: "ບໍ່ມີຫມາຍເຫດ" | ||||||
|  | noNotifications: "ບໍ່ມີການແຈ້ງເຕືອນ" | ||||||
|  | instance: "ອີນສະແຕນ" | ||||||
|  | settings: "ກຳນົດຄ່າ" | ||||||
|  | basicSettings: "ການຕັ້ງຄ່າພື້ນຖານ" | ||||||
|  | otherSettings: "ການຕັ້ງຄ່າອື່ນໆ" | ||||||
|  | openInWindow: "ເປີດຢູ່ໃນປ່ອງຢ້ຽມ" | ||||||
|  | profile: "ໂພຼຟາຍ" | ||||||
|  | timeline: "ເສັ້ນກຳນົດເວລາ" | ||||||
|  | noAccountDescription: "ຜູ້ໃຊ້ນີ້ຍັງບໍ່ໄດ້ຂຽນໃນຊີວະປະຫວັດຂອງເຂົາເຈົ້າເທື່ອ" | ||||||
|  | login: "ເຂົ້າສູ່ລະບົບ" | ||||||
|  | loggingIn: "ກຳລັງເຂົ້າສູ່ລະບົບ..." | ||||||
|  | logout: "ອອກຈາກລະບົບ" | ||||||
|  | signup: "ລົງທະບຽນ" | ||||||
|  | uploading: "ການອັບໂຫຼດ..." | ||||||
|  | save: "ບັນທຶກ" | ||||||
|  | users: "ຜູ້ໃຊ້ຕ່າງໆ" | ||||||
|  | addUser: "ເພີ່ມຜູ້ໃຊ້" | ||||||
|  | favorite: "ເພີ່ມໃສ່ລາຍການທີ່ມັກ" | ||||||
|  | favorites: "ລາຍການທີ່ມັກ" | ||||||
|  | unfavorite: "ລຶບອອກຈາກລາຍການທີ່ມັກ" | ||||||
|  | favorited: "ເພີ່ມໃສ່ລາຍການທີ່ມັກແລ້ວ" | ||||||
|  | alreadyFavorited: "ເພີ່ມເຂົ້າໃນລາຍການທີ່ມັກແລ້ວ." | ||||||
|  | cantFavorite: "ບໍ່ສາມາດເພີ່ມໃສ່ລາຍການທີ່ມັກໄດ້." | ||||||
|  | pin: "ປັກໝຸດໄປຫາໂປຣໄຟລ໌" | ||||||
|  | unpin: "ຖອດປັກໝຸດອອກຈາກໂປຣໄຟລ໌" | ||||||
|  | copyContent: "ຄັດລອກເນື້ອຫາ" | ||||||
|  | copyLink: "ສຳເນົາລິ້ງ" | ||||||
|  | delete: "ລຶບ" | ||||||
|  | deleteAndEdit: "ລົບແລະແກ້ໄຂ" | ||||||
|  | deleteAndEditConfirm: "ເຈົ້າແນ່ໃຈບໍ່? ທີ່ທ່ານຕ້ອງການທີ່ຈະລຶບບັນທຶກນີ້ແລະແກ້ໄຂມັນ ທ່ານອາດຈະສູນເສຍການໂຕ້ຕອບ, ບັນທຶກ, ແລະການຕອບກັບທັງໝົດ" | ||||||
|  | addToList: "ເພີ່ມໃສ່ລາຍຊື່" | ||||||
|  | sendMessage: "ສົ່ງຂໍ້ຄວາມ" | ||||||
|  | copyRSS: "ສຳເນົາ RSS" | ||||||
|  | copyUsername: "ສຳເນົາຊື່ຜູ້ໃຊ້" | ||||||
|  | searchUser: "ຄົ້ນຫາຜູ້ໃຊ້" | ||||||
|  | reply: "ຕອບໄປທີ" | ||||||
|  | loadMore: "ໂຫຼດເພີ່ມເຕີມ" | ||||||
|  | showMore: "ໂຫຼດເພີ່ມເຕີມ" | ||||||
|  | showLess: "ປິດ" | ||||||
|  | youGotNewFollower: "ໄດ້ຕິດຕາມທ່ານ" | ||||||
|  | receiveFollowRequest: "ປະຕິບັດຕາມຄໍາຮ້ອງຂໍທີ່ໄດ້ຮັບ" | ||||||
|  | followRequestAccepted: "ຜູ້ຕິດຕາມໄດ້ຍອມຮັບຄໍາຮ້ອງຂໍຂອງທ່ານ" | ||||||
|  | mention: "ໄດ້ກ່າວມາ" | ||||||
|  | mentions: "ກ່າວເຖິງ" | ||||||
|  | directNotes: "ໂດຍກົງຫມາຍເຫດ" | ||||||
|  | importAndExport: "ນໍາເຂົ້າ / ສົ່ງອອກ" | ||||||
|  | import: "ນຳເຂົ້າ" | ||||||
|  | export: "ນຳອອກ" | ||||||
|  | files: "ໄຟລ໌" | ||||||
|  | download: "ດາວໂຫລດ" | ||||||
|  | driveFileDeleteConfirm: "ທ່ານແນ່ໃຈບໍ່ວ່າຕ້ອງການລຶບໄຟລ໌ \"{name}\"? ບັນທຶກທີ່ມີໄຟລ໌ແນບນີ້ຈະຖືກລຶບຖິ້ມ" | ||||||
|  | unfollowConfirm: "ທ່ານແນ່ໃຈບໍ່ວ່າຕ້ອງການເຊົາຕິດຕາມ {name}?" | ||||||
|  | exportRequested: "ໃນເວລາທີ່ທ່ານໄດ້ຮ້ອງຂໍການສົ່ງອອກ ມັນອາດຈະໃຊ້ເວລາບາງເວລາ ແລະມັນຈະຖືກເພີ່ມໃສ່ drive ຂອງທ່ານເມື່ອມັນສຳເລັດແລ້ວ" | ||||||
|  | importRequested: "ໃນເວລາທີ່ທ່ານໄດ້ຮ້ອງຂໍການນໍາເຂົ້າ ມັນອາດຈະໃຊ້ເວລາບາງເວລາ" | ||||||
|  | lists: "ລາຍການ" | ||||||
|  | noLists: "ທ່ານບໍ່ມີລາຍການໃດໆ" | ||||||
|  | note: "ບັນທຶກ" | ||||||
|  | notes: "ບັນທຶກ" | ||||||
|  | following: "ກຳລັງຕິດຕາມ" | ||||||
|  | followers: "ຜູ້ຕິດຕາມ" | ||||||
|  | followsYou: "ຕິດຕາມເຈົ້າ" | ||||||
|  | createList: "ສ້າງລາຍຊື່" | ||||||
|  | manageLists: "ການບໍລິຫານບັນຊີລາຍການ" | ||||||
|  | error: "ຂໍ້ຜິດພາດ" | ||||||
|  | somethingHappened: "ອຸຍ, ມີບາງຢ່າງຜິດພາດ" | ||||||
|  | retry: "ລອງໃຫມ່" | ||||||
|  | pageLoadError: "ເກີດຄວາມຜິດພາດໃນການໂຫລດໜ້ານີ້" | ||||||
|  | pageLoadErrorDescription: "ປົກກະຕິແລ້ວມັນເກີດຈາກຄວາມຜິດພາດເຄືອຂ່າຍ ຫຼື cache ຂອງຕົວທ່ອງເວັບ ລອງລຶບລ້າງແຄດແລ້ວລອງໃໝ່ພາຍຫຼັງສອງສາມນາທີ" | ||||||
|  | serverIsDead: "ເຊີບເວີນີ້ບໍ່ຕອບສະໜອງ ກະລຸນາລໍຖ້າຈັກໜ່ອຍແລ້ວລອງໃໝ່ອີກຄັ້ງ" | ||||||
|  | youShouldUpgradeClient: "ເພື່ອເບິ່ງໜ້ານີ້, ກະລຸນາໂຫຼດຂໍ້ມູນຄືນໃໝ່ເພື່ອອັບເດດລູກຄ້າຂອງທ່ານ" | ||||||
|  | enterListName: "ໃສ່ຊື່ສຳລັບລາຍຊື່" | ||||||
|  | privacy: "ຄວາມເປັນສ່ວນຕົວ" | ||||||
|  | makeFollowManuallyApprove: "ປະຕິບັດຕາມການຮ້ອງຂໍຮຽກຮ້ອງໃຫ້ມີການອະນຸມັດ" | ||||||
|  | defaultNoteVisibility: "ເປັນຄ່າເລີ່ມຕົ້ນ" | ||||||
|  | follow: "ກຳລັງຕິດຕາມ" | ||||||
|  | followRequest: "ສົ່ງການຮ້ອງຂໍປະຕິບຕາມ" | ||||||
|  | followRequests: "ປະຕິບັດຕາມຄໍາຮ້ອງຂໍ" | ||||||
|  | unfollow: "ເຊົາຕິດຕາມ" | ||||||
|  | followRequestPending: "ປະຕິບັດຕາມຄໍາຮ້ອງຂໍທີ່ລໍຖ້າຢູ່" | ||||||
|  | enterEmoji: "ປ້ອນອີໂມຈິ" | ||||||
|  | renote: "Renote" | ||||||
|  | unrenote: "ເລີກ Renote" | ||||||
|  | renoted: "ເກັບບັນທຶກໄວ້" | ||||||
|  | quote: "ລວມຂໍ້ຄວາມອ້າງອີງ" | ||||||
|  | pinnedNote: "ບັນທຶກທີ່ປັກໝຸດໄວ້" | ||||||
|  | pinned: "ປັກໝຸດໄປຫາໂປຣໄຟລ໌" | ||||||
|  | you: "ເຈົ້າ" | ||||||
|  | clickToShow: "ກົດເພື່ອສະແດງໃຫ້ເຫັນ" | ||||||
|  | sensitive: "NSFW" | ||||||
|  | add: "ເພີ່ມ" | ||||||
|  | reaction: "ປະຕິກິລິຍາ" | ||||||
|  | reactions: "ປະຕິກິລິຍາ" | ||||||
|  | mute: "ປີດສຽງ" | ||||||
|  | unmute: "ເປີດສຽງ" | ||||||
|  | block: "ບ໋ອກ" | ||||||
|  | unblock: "ຍົກເລີກກາຮົບລັອກ" | ||||||
|  | suspend: "ລະງັບ" | ||||||
|  | unsuspend: "ເຊົາລະງັບ" | ||||||
|  | selectList: "ເລືອກບັນຊີລາຍການ" | ||||||
|  | selectWidget: "ເລືອກວິກເຈັດ" | ||||||
|  | editWidgets: "ແກ້ໄຂ Widget" | ||||||
|  | editWidgetsExit: "ສຳເລັດແລ້ວ" | ||||||
|  | customEmojis: "ອີໂມຈິແບບກຳນົດເອງ" | ||||||
|  | emoji: "ອີໂມຈິ" | ||||||
|  | emojis: "ອີໂມຈິ" | ||||||
|  | emojiName: "ຊື່ Emoji" | ||||||
|  | emojiUrl: "URL ອີໂມຈິ" | ||||||
|  | addEmoji: "ຕື່ມອີໂມຈິ" | ||||||
|  | flagAsBot: "ໝາຍບັນຊີນີ້ເປັນບັອດ" | ||||||
|  | flagAsCat: "ໝາຍບັນຊີນີ້ເປັນແມວ" | ||||||
|  | flagAsCatDescription: "ເປີດໃຊ້ຕົວເລືອກນີ້ເພື່ອໝາຍບັນຊີນີ້ເປັນແມວ" | ||||||
|  | flagShowTimelineReplies: "ສະແດງການຕອບກັບໃນທາມລາຍ" | ||||||
|  | flagShowTimelineRepliesDescription: "ສະແດງການຕອບກັບຂອງຜູ້ໃຊ້ຕໍ່ກັບບັນທຶກຂອງຜູ້ໃຊ້ອື່ນໃນທາມລາຍຖ້າເປີດໃຊ້ງານ" | ||||||
|  | autoAcceptFollowed: "ອະນຸມັດອັດຕະໂນມັດຕາມຄຳຮ້ອງຂໍຈາກຜູ້ໃຊ້ທີ່ທ່ານກຳລັງຕິດຕາມຢູ່" | ||||||
|  | addAccount: "ເພີ່ມບັນຊີ" | ||||||
|  | loginFailed: "ການເຂົ້າສູ່ລະບົບບໍ່ສຳເລັດ" | ||||||
|  | general: "ທົ່ວໄປ" | ||||||
|  | wallpaper: "ພາບພື້ນຫລັງ" | ||||||
|  | setWallpaper: "ຕັ້ງເປັນພາບພື້ນຫຼັງ" | ||||||
|  | instances: "ອີນສະແຕນ" | ||||||
|  | instanceInfo: "ອີນສະແຕນ" | ||||||
|  | statistics: "ສະຖິຕິ" | ||||||
|  | clearQueue: "ລ້າງຄິວ" | ||||||
|  | clearCachedFiles: "ລຶບລ້າງແຄສ" | ||||||
|  | editProfile: "ແກ້ໄຂໂປຣໄຟລ໌" | ||||||
|  | done: "ສຳເລັດ" | ||||||
|  | processing: "ກຳລັງປະມວນຜົນ" | ||||||
|  | preview: "ສະແດງເປັນຕົວຢ່າງ" | ||||||
|  | default: "ຄ່າເລີ່ມຕົ້ນ" | ||||||
|  | blocked: "ບລັອກແລ້ວ " | ||||||
|  | all: "ທັງໝົດ" | ||||||
|  | subscribing: "ສະໝັກສະມາຊິກແລັວ" | ||||||
|  | publishing: "ການພິມເຜີຍແຜ່" | ||||||
|  | notResponding: "ບໍ່ຕອບສະໜອງ" | ||||||
|  | instanceFollowing: "ກຳລັງຕິດຕາມສຸດຕົວຢ່າງ" | ||||||
|  | instanceFollowers: "ຜູ້ຕິດຕາມຕົວຢ່າງ" | ||||||
|  | instanceUsers: "ຜູ້ຊົມໃຊ້ຂອງຕົວຢ່າງນີ້" | ||||||
|  | changePassword: "ປ່ຽນລະຫັດຜ່ານ" | ||||||
|  | featured: "ໄຮໄລທ໌" | ||||||
|  | announcements: "ປະກາດ" | ||||||
|  | remove: "ລຶບ" | ||||||
|  | messaging: "ແຊ໋ດ" | ||||||
|  | tos: "ເງື່ອນໄຂການໃຫ້ບໍລິການ" | ||||||
|  | start: "ເລີ່ມຕົ້ນນຳໃຊ້ເລີຍ" | ||||||
|  | home: "ໜ້າຫຼັກ" | ||||||
|  | images: "ຮູບພາບ" | ||||||
|  | birthday: "ວັນເກີດ" | ||||||
|  | registeredDate: "ວັນທີ່ເປັນສະມາຊິກ" | ||||||
|  | location: "ທີ່ຕັ້ງ" | ||||||
|  | theme: "ແທ໋ມ" | ||||||
|  | light: "ສະຫວ່າງ" | ||||||
|  | dark: "ມືດ" | ||||||
|  | lightThemes: "ຊຸດຮູບແບບສະຫວ່າງ" | ||||||
|  | darkThemes: "ຮູບແບບສີສັນມືດ" | ||||||
|  | fileName: "ຊື່ໄຟລ໌" | ||||||
|  | selectFile: "ເລືອກໄຟລ໌" | ||||||
|  | selectFiles: "ເລືອກໄຟລ໌" | ||||||
|  | nsfw: "NSFW" | ||||||
|  | accept: "ອະນຸຍາດ" | ||||||
|  | pinnedNotes: "ບັນທຶກທີ່ປັກໝຸດໄວ້" | ||||||
|  | userList: "ລາຍການ" | ||||||
|  | smtpUser: "ຊື່ຜູ້ໃຊ້" | ||||||
|  | smtpPass: "ລະຫັດຜ່ານ" | ||||||
|  | clearCache: "ລຶບລ້າງແຄສ" | ||||||
|  | user: "ຜູ້ໃຊ້ຕ່າງໆ" | ||||||
|  | searchByGoogle: "ຄົ້ນຫາ" | ||||||
|  | file: "ໄຟລ໌" | ||||||
|  | _email: | ||||||
|  |   _follow: | ||||||
|  |     title: "ໄດ້ຕິດຕາມທ່ານ" | ||||||
|  | _theme: | ||||||
|  |   keys: | ||||||
|  |     mention: "ໄດ້ກ່າວມາ" | ||||||
|  |     renote: "Renote" | ||||||
|  | _sfx: | ||||||
|  |   note: "ບັນທຶກ" | ||||||
|  |   notification: "ການແຈ້ງເຕືອນ" | ||||||
|  |   chat: "ແຊ໋ດ" | ||||||
|  | _widgets: | ||||||
|  |   profile: "ໂພຼຟາຍ" | ||||||
|  |   instanceInfo: "ອີນສະແຕນ" | ||||||
|  |   notifications: "ການແຈ້ງເຕືອນ" | ||||||
|  |   timeline: "ເສັ້ນກຳນົດເວລາ" | ||||||
|  |   _userList: | ||||||
|  |     chooseList: "ເລືອກບັນຊີລາຍການ" | ||||||
|  | _cw: | ||||||
|  |   show: "ໂຫຼດເພີ່ມເຕີມ" | ||||||
|  | _visibility: | ||||||
|  |   home: "ໜ້າຫຼັກ" | ||||||
|  |   followers: "ຜູ້ຕິດຕາມ" | ||||||
|  | _profile: | ||||||
|  |   username: "ຊື່ຜູ້ໃຊ້" | ||||||
|  | _exportOrImport: | ||||||
|  |   followingList: "ກຳລັງຕິດຕາມ" | ||||||
|  |   muteList: "ປີດສຽງ" | ||||||
|  |   blockingList: "ບ໋ອກ" | ||||||
|  |   userLists: "ລາຍການ" | ||||||
|  | _timelines: | ||||||
|  |   home: "ໜ້າຫຼັກ" | ||||||
|  | _pages: | ||||||
|  |   blocks: | ||||||
|  |     image: "ຮູບພາບ" | ||||||
|  | _notification: | ||||||
|  |   youWereFollowed: "ໄດ້ຕິດຕາມທ່ານ" | ||||||
|  |   _types: | ||||||
|  |     follow: "ກຳລັງຕິດຕາມ" | ||||||
|  |     mention: "ໄດ້ກ່າວມາ" | ||||||
|  |     renote: "Renote" | ||||||
|  |     quote: "ລວມຂໍ້ຄວາມອ້າງອີງ" | ||||||
|  |     reaction: "ປະຕິກິລິຍາ" | ||||||
|  |   _actions: | ||||||
|  |     reply: "ຕອບໄປທີ" | ||||||
|  |     renote: "Renote" | ||||||
|  | _deck: | ||||||
|  |   _columns: | ||||||
|  |     notifications: "ການແຈ້ງເຕືອນ" | ||||||
|  |     tl: "ເສັ້ນກຳນົດເວລາ" | ||||||
|  |     list: "ລາຍການ" | ||||||
|  |     channel: "ຊ່ອງ" | ||||||
|  |     mentions: "ກ່າວເຖິງ" | ||||||
| @@ -427,11 +427,6 @@ loggedInAsBot: "Momenteel als bot ingelogd" | |||||||
| _email: | _email: | ||||||
|   _follow: |   _follow: | ||||||
|     title: "volgde jou" |     title: "volgde jou" | ||||||
| _mfm: |  | ||||||
|   mention: "Vermelding" |  | ||||||
|   quote: "Quote" |  | ||||||
|   emoji: "Maatwerk emoji" |  | ||||||
|   search: "Zoeken" |  | ||||||
| _theme: | _theme: | ||||||
|   keys: |   keys: | ||||||
|     mention: "Vermelding" |     mention: "Vermelding" | ||||||
|   | |||||||
| @@ -461,7 +461,6 @@ youHaveNoGroups: "Nie masz żadnych grup" | |||||||
| joinOrCreateGroup: "Uzyskaj zaproszenie do dołączenia do grupy lub utwórz własną grupę." | joinOrCreateGroup: "Uzyskaj zaproszenie do dołączenia do grupy lub utwórz własną grupę." | ||||||
| noHistory: "Brak historii" | noHistory: "Brak historii" | ||||||
| signinHistory: "Historia logowania" | signinHistory: "Historia logowania" | ||||||
| disableAnimatedMfm: "Wyłącz MFM z animacją" |  | ||||||
| doing: "Przetwarzanie..." | doing: "Przetwarzanie..." | ||||||
| category: "Kategoria" | category: "Kategoria" | ||||||
| tags: "Tagi" | tags: "Tagi" | ||||||
| @@ -958,68 +957,6 @@ _nsfw: | |||||||
|   respect: "Ukrywaj media NSFW" |   respect: "Ukrywaj media NSFW" | ||||||
|   ignore: "Nie ukrywaj mediów NSFW" |   ignore: "Nie ukrywaj mediów NSFW" | ||||||
|   force: "Ukrywaj wszystkie media" |   force: "Ukrywaj wszystkie media" | ||||||
| _mfm: |  | ||||||
|   cheatSheet: "Ściąga MFM" |  | ||||||
|   intro: "MFM to język składniowy wyjątkowy dla Misskey, który może być użyty w wielu miejscach. Tu znajdziesz listę wszystkich możliwych elementów składni MFM." |  | ||||||
|   dummy: "Misskey rozszerza świat Fediwersum" |  | ||||||
|   mention: "Wspomnij" |  | ||||||
|   mentionDescription: "Używając znaku @ i nazwy użytkownika, możesz określić danego użytkownika." |  | ||||||
|   hashtag: "Hashtag" |  | ||||||
|   hashtagDescription: "Używając kratki i tekstu, możesz określić hashtag." |  | ||||||
|   url: "Adres URL" |  | ||||||
|   urlDescription: "Adresy URL mogą być wyświetlane" |  | ||||||
|   link: "Odnośnik" |  | ||||||
|   linkDescription: "Określone części tekstu mogą być wyświetlane jako adres URL." |  | ||||||
|   bold: "Pogrubienie" |  | ||||||
|   boldDescription: "Wyróżnia litery pogrubiając je." |  | ||||||
|   small: "Małe" |  | ||||||
|   smallDescription: "Wyświetla treść jako małą i cienką." |  | ||||||
|   center: "Wyśrodkowanie" |  | ||||||
|   centerDescription: "Wyśrodkowuje zawartość." |  | ||||||
|   inlineCode: "Kod (w wierszu)" |  | ||||||
|   blockCode: "Kod (blok)" |  | ||||||
|   blockCodeDescription: "Wyświetla kod z podświetlaną składnią składający się z wielu linii." |  | ||||||
|   blockMath: "Matematyka (Blok)" |  | ||||||
|   quote: "Cytuj" |  | ||||||
|   quoteDescription: "Wyświetla treść jako cytat." |  | ||||||
|   emoji: "Niestandardowe emoji" |  | ||||||
|   emojiDescription: "Otaczając nazwę niestandardowego emoji dwukropkami, możesz użyć niestandardowego emoji." |  | ||||||
|   search: "Szukaj" |  | ||||||
|   searchDescription: "Wyświetla pole wyszukiwania z wcześniej wpisanym tekstem." |  | ||||||
|   flip: "Odwróć" |  | ||||||
|   flipDescription: "Przerzuca treść poziomo lub pionowo." |  | ||||||
|   jelly: "Animacja (Galaretka)" |  | ||||||
|   jellyDescription: "Nadaje treści galaretowatą animację." |  | ||||||
|   tada: "Animation (Tada)" |  | ||||||
|   tadaDescription: "Nadaje treści animację podobną do \"Tada!\"." |  | ||||||
|   jump: "Animacja (Skok)" |  | ||||||
|   jumpDescription: "Nadaje treści animację skakania." |  | ||||||
|   bounce: "Animacja (Odbijanie)" |  | ||||||
|   bounceDescription: "Nadaje treści animację odbijania się." |  | ||||||
|   shake: "Animacja (Wstrząsanie)" |  | ||||||
|   shakeDescription: "Nadaje treści animację wstrząsania." |  | ||||||
|   twitch: "Animacja (Drganie)" |  | ||||||
|   twitchDescription: "Nadaje treści mocno drgającą animację." |  | ||||||
|   spin: "Animacja (Obrót)" |  | ||||||
|   spinDescription: "Nadaje treści animację obracania." |  | ||||||
|   x2: "Duże" |  | ||||||
|   x2Description: "Czyni treść większą." |  | ||||||
|   x3: "Bardzo duże" |  | ||||||
|   x3Description: "Czyni treść jeszcze większą." |  | ||||||
|   x4: "Ogromne" |  | ||||||
|   x4Description: "Czyni treść jeszcze większą niż jeszcze większa." |  | ||||||
|   blur: "Rozmycie" |  | ||||||
|   blurDescription: "Rozmywa treść. Zostanie wyraźnie wyświetlona po najechaniu." |  | ||||||
|   font: "Czcionka" |  | ||||||
|   fontDescription: "Wybiera czcionkę do wyświetlania treści." |  | ||||||
|   rainbow: "Tęcza" |  | ||||||
|   rainbowDescription: "Sprawia, że zawartość pojawia się w kolorach tęczy." |  | ||||||
|   sparkle: "Blask" |  | ||||||
|   sparkleDescription: "Nadaje zawartości efekt lśniącego brokatu." |  | ||||||
|   rotate: "Obróć" |  | ||||||
|   rotateDescription: "Obraca zawartość o określony kąt." |  | ||||||
|   plain: "Zwyczajny" |  | ||||||
|   plainDescription: "Wyłącza efekty wszystkich MFM zawartych w tym efekcie MFM." |  | ||||||
| _instanceTicker: | _instanceTicker: | ||||||
|   none: "Nigdy nie pokazuj" |   none: "Nigdy nie pokazuj" | ||||||
|   remote: "Pokaż dla zdalnych użytkowników" |   remote: "Pokaż dla zdalnych użytkowników" | ||||||
| @@ -1438,5 +1375,6 @@ _deck: | |||||||
|     tl: "Oś czasu" |     tl: "Oś czasu" | ||||||
|     antenna: "Anteny" |     antenna: "Anteny" | ||||||
|     list: "Listy" |     list: "Listy" | ||||||
|  |     channel: "Kanały" | ||||||
|     mentions: "Wspomnienia" |     mentions: "Wspomnienia" | ||||||
|     direct: "Bezpośredni" |     direct: "Bezpośredni" | ||||||
|   | |||||||
| @@ -475,11 +475,6 @@ file: "Ficheiros" | |||||||
| _email: | _email: | ||||||
|   _follow: |   _follow: | ||||||
|     title: "Você tem um novo seguidor" |     title: "Você tem um novo seguidor" | ||||||
| _mfm: |  | ||||||
|   mention: "Menção" |  | ||||||
|   quote: "Citar" |  | ||||||
|   emoji: "Emoji personalizado" |  | ||||||
|   search: "Buscar" |  | ||||||
| _theme: | _theme: | ||||||
|   keys: |   keys: | ||||||
|     mention: "Menção" |     mention: "Menção" | ||||||
|   | |||||||
| @@ -455,7 +455,6 @@ youHaveNoGroups: "Nu ai niciun grup" | |||||||
| joinOrCreateGroup: "Primește o invitație într-un grup sau creează unul nou." | joinOrCreateGroup: "Primește o invitație într-un grup sau creează unul nou." | ||||||
| noHistory: "Nu există istoric" | noHistory: "Nu există istoric" | ||||||
| signinHistory: "Istoric autentificări" | signinHistory: "Istoric autentificări" | ||||||
| disableAnimatedMfm: "Dezactivează MFM cu animații" |  | ||||||
| doing: "Se procesează..." | doing: "Se procesează..." | ||||||
| category: "Categorie" | category: "Categorie" | ||||||
| tags: "Etichete" | tags: "Etichete" | ||||||
| @@ -655,11 +654,6 @@ _role: | |||||||
| _email: | _email: | ||||||
|   _follow: |   _follow: | ||||||
|     title: "te-a urmărit" |     title: "te-a urmărit" | ||||||
| _mfm: |  | ||||||
|   mention: "Mențiune" |  | ||||||
|   quote: "Citează" |  | ||||||
|   emoji: "Emoji personalizat" |  | ||||||
|   search: "Caută" |  | ||||||
| _theme: | _theme: | ||||||
|   description: "Descriere" |   description: "Descriere" | ||||||
|   keys: |   keys: | ||||||
| @@ -721,4 +715,5 @@ _deck: | |||||||
|     tl: "Cronologie" |     tl: "Cronologie" | ||||||
|     antenna: "Antene" |     antenna: "Antene" | ||||||
|     list: "Liste" |     list: "Liste" | ||||||
|  |     channel: "Canale" | ||||||
|     mentions: "Mențiuni" |     mentions: "Mențiuni" | ||||||
|   | |||||||
| @@ -462,7 +462,6 @@ youHaveNoGroups: "У вас нет ни одной группы" | |||||||
| joinOrCreateGroup: "Получайте приглашения в группы или создавайте свои собственные" | joinOrCreateGroup: "Получайте приглашения в группы или создавайте свои собственные" | ||||||
| noHistory: "История пока пуста" | noHistory: "История пока пуста" | ||||||
| signinHistory: "Журнал посещений" | signinHistory: "Журнал посещений" | ||||||
| disableAnimatedMfm: "Отключение анимированной разметки MFM" |  | ||||||
| doing: "В процессе" | doing: "В процессе" | ||||||
| category: "Категория" | category: "Категория" | ||||||
| tags: "Метки" | tags: "Метки" | ||||||
| @@ -1309,72 +1308,6 @@ _nsfw: | |||||||
|   respect: "Скрывать содержимое не для всех" |   respect: "Скрывать содержимое не для всех" | ||||||
|   ignore: "Показывать содержимое не для всех" |   ignore: "Показывать содержимое не для всех" | ||||||
|   force: "Скрывать вообще все файлы" |   force: "Скрывать вообще все файлы" | ||||||
| _mfm: |  | ||||||
|   cheatSheet: "Подсказка по разметке MFM" |  | ||||||
|   intro: "MFM — язык оформления текста, который придуман специально для Misskey и готов для применения во многих местах. На этой странице собраны и кратко изложены способы его использовать." |  | ||||||
|   dummy: "Misskey расширяет границы Федиверса." |  | ||||||
|   mention: "Упоминание" |  | ||||||
|   mentionDescription: "При помощи знака «собака» перед именем можно упомянуть какого-нибудь пользователя." |  | ||||||
|   hashtag: "Хэштег" |  | ||||||
|   hashtagDescription: "При помощи знака «решётка» перед словом задаётся хэштег." |  | ||||||
|   url: "Простая ссылка (URL)" |  | ||||||
|   urlDescription: "Ссылки могут отображаться непосредственно." |  | ||||||
|   link: "Ссылка с пояснением" |  | ||||||
|   linkDescription: "Можно ссылку оформить в виде произвольного текста." |  | ||||||
|   bold: "Жирный шрифт" |  | ||||||
|   boldDescription: "Выделяет текст, делая буквы жирнее." |  | ||||||
|   small: "Мелкий шрифт" |  | ||||||
|   smallDescription: "Делает текст маленьким и незаметным." |  | ||||||
|   center: "Выровнять элементы по центру" |  | ||||||
|   centerDescription: "Так можно выровнять что-то по центру." |  | ||||||
|   inlineCode: "Программа (в тексте)" |  | ||||||
|   inlineCodeDescription: "Подсвечивает фрагмент программы внутри сплошного текста." |  | ||||||
|   blockCode: "Программа (блок)" |  | ||||||
|   blockCodeDescription: "Оформляет текст программы в виде отдельного блокоа. Он может состоять из множества строк." |  | ||||||
|   inlineMath: "Математическое выражение (в тексте)" |  | ||||||
|   inlineMathDescription: "Позволяет вставлять математические выражения внутрь текста при помощи языка KaTeX." |  | ||||||
|   blockMath: "Математическое выражение (блок)" |  | ||||||
|   blockMathDescription: "Оформляет математическое выражение (KaTeX) на отдельной строке." |  | ||||||
|   quote: "Цитата" |  | ||||||
|   quoteDescription: "Так можно процитировать чей-то текст." |  | ||||||
|   emoji: "Собственные эмодзи" |  | ||||||
|   emojiDescription: "Можно вставить эмодзи в текст, окружив название двоеточиями." |  | ||||||
|   search: "Поиск" |  | ||||||
|   searchDescription: "Можно добавить форму для поиска, сразу задав, что искать." |  | ||||||
|   flip: "Переворот" |  | ||||||
|   flipDescription: "Позволяет отразить текст зеркально по вертикали или горизонтали." |  | ||||||
|   jelly: "Анимация желе (шлёп-плёп)" |  | ||||||
|   jellyDescription: "Напоминает горку джема, дёргающуюся от шлепков." |  | ||||||
|   tada: "Анимация (та-дам!)" |  | ||||||
|   tadaDescription: "Получается нечто выпрыгивающее, как бы крича: «а вот и я!»" |  | ||||||
|   jump: "Анимация прыжков (прыг-скок)" |  | ||||||
|   jumpDescription: "Побуждает радостно подпрыгивать." |  | ||||||
|   bounce: "Анимация отскоков (бум-бум)" |  | ||||||
|   bounceDescription: "Это будет скакать как мяч." |  | ||||||
|   shake: "Анимация дрожи (б-р-р-р)" |  | ||||||
|   shakeDescription: "Такое дрожит, словно от холода. Или от страха." |  | ||||||
|   twitch: "Анимация тряски" |  | ||||||
|   twitchDescription: "Заставляет трястись как одержимого" |  | ||||||
|   spin: "Вращение" |  | ||||||
|   spinDescription: "Так можно крутить содержимое в разных направлениях." |  | ||||||
|   x2: "Крупный шрифт" |  | ||||||
|   x2Description: "Увеличивает содержимое." |  | ||||||
|   x3: "Ещё крупнее" |  | ||||||
|   x3Description: "Сильнее увеличивает содержимое." |  | ||||||
|   x4: "Совсем крупно" |  | ||||||
|   x4Description: "Увеличивает содержимое совсем сильно." |  | ||||||
|   blur: "Размытие" |  | ||||||
|   blurDescription: "Размывает текст до нечитаемости, будто его поместили за матовое стекло. Наведение указателя мыши на размытый текст возвращает чёткость." |  | ||||||
|   font: "Шрифт" |  | ||||||
|   fontDescription: "Так можно писать произвольным шрифтом." |  | ||||||
|   rainbow: "Радуга" |  | ||||||
|   rainbowDescription: "Заставлять содержимое отображаться в цветах радуги." |  | ||||||
|   sparkle: "Искры" |  | ||||||
|   sparkleDescription: "Добавляет эффект искрящихся частиц." |  | ||||||
|   rotate: "Повернуть" |  | ||||||
|   rotateDescription: "Поворачивает на заданный угол." |  | ||||||
|   plain: "Буквально" |  | ||||||
|   plainDescription: "MFM внутри отключается, и текст отображается как есть" |  | ||||||
| _instanceTicker: | _instanceTicker: | ||||||
|   none: "Не показывать" |   none: "Не показывать" | ||||||
|   remote: "Только для других сайтов" |   remote: "Только для других сайтов" | ||||||
| @@ -1845,5 +1778,6 @@ _deck: | |||||||
|     tl: "Лента" |     tl: "Лента" | ||||||
|     antenna: "Антенны" |     antenna: "Антенны" | ||||||
|     list: "Списки" |     list: "Списки" | ||||||
|  |     channel: "Каналы" | ||||||
|     mentions: "Упоминания" |     mentions: "Упоминания" | ||||||
|     direct: "Личное" |     direct: "Личное" | ||||||
|   | |||||||
| @@ -464,7 +464,6 @@ youHaveNoGroups: "Nemáte žiadne skupiny" | |||||||
| joinOrCreateGroup: "Požiadajte o pozvanie do existujúcej skupiny alebo vytvorte novú." | joinOrCreateGroup: "Požiadajte o pozvanie do existujúcej skupiny alebo vytvorte novú." | ||||||
| noHistory: "Žiadna história" | noHistory: "Žiadna história" | ||||||
| signinHistory: "História prihlásení" | signinHistory: "História prihlásení" | ||||||
| disableAnimatedMfm: "Vypnúť MFM s animáciou" |  | ||||||
| doing: "Pracujem..." | doing: "Pracujem..." | ||||||
| category: "Kategórie" | category: "Kategórie" | ||||||
| tags: "Značky" | tags: "Značky" | ||||||
| @@ -1013,72 +1012,6 @@ _nsfw: | |||||||
|   respect: "Skryť NSFW médiá" |   respect: "Skryť NSFW médiá" | ||||||
|   ignore: "Neskrývať NSFW médiá" |   ignore: "Neskrývať NSFW médiá" | ||||||
|   force: "Skryť všetky médiá" |   force: "Skryť všetky médiá" | ||||||
| _mfm: |  | ||||||
|   cheatSheet: "MFM Cheatsheet" |  | ||||||
|   intro: "MFM je Misskey exkluzívny značkovací jazyk, ktorý sa dá používať na viacerých miestach. Tu môžete vidieť zoznam všetkej dostupnej MFM syntaxe." |  | ||||||
|   dummy: "Misskey rozširuje svet Fediverza" |  | ||||||
|   mention: "Zmienka" |  | ||||||
|   mentionDescription: "Používateľa spomeniete použítím zavináča a mena používateľa" |  | ||||||
|   hashtag: "Hashtag" |  | ||||||
|   hashtagDescription: "Môžete zadať hashtag použitím mriežky a textu" |  | ||||||
|   url: "URL" |  | ||||||
|   urlDescription: "URL sa dajú zobraziť." |  | ||||||
|   link: "Odkaz" |  | ||||||
|   linkDescription: "Jednotlivé časti texty sa dajú zobraziť ako URL." |  | ||||||
|   bold: "Tučné" |  | ||||||
|   boldDescription: "Zvýrazní písmená tým, že budú tučnejšie." |  | ||||||
|   small: "Malé" |  | ||||||
|   smallDescription: "Zobrazí obsah malý a tenký." |  | ||||||
|   center: "Vystrediť prvky" |  | ||||||
|   centerDescription: "Zobrazí obsah v strede" |  | ||||||
|   inlineCode: "Kód (inline)" |  | ||||||
|   inlineCodeDescription: "Zobrazí kód so zvýraznením syntaxe." |  | ||||||
|   blockCode: "Kód (blok)" |  | ||||||
|   blockCodeDescription: "Zobrazí viacriadkový kód so zvýraznením syntaxe v bloku." |  | ||||||
|   inlineMath: "Vzorec (inline)" |  | ||||||
|   inlineMathDescription: "Zobrazí matematický vzorec (KaTeX) v riadku." |  | ||||||
|   blockMath: "Vzorec (blok)" |  | ||||||
|   blockMathDescription: "Zobrazí viacriadkový matematický vzorec (KaTeX) v bloku" |  | ||||||
|   quote: "Citovať" |  | ||||||
|   quoteDescription: "Zobrazí obsah ako citát." |  | ||||||
|   emoji: "Vlastné emoji" |  | ||||||
|   emojiDescription: "Pridaním dvojbodiek pred a za názov vlastnej emoji, sa dá zobraziť vlastná emoji." |  | ||||||
|   search: "Hľadať" |  | ||||||
|   searchDescription: "Zobrazí vyhľadávacie pole so zadaným textom." |  | ||||||
|   flip: "Preklopiť" |  | ||||||
|   flipDescription: "Preklopí obsah horizontálne alebo vertikálne" |  | ||||||
|   jelly: "Animácia (želé)" |  | ||||||
|   jellyDescription: "Obsah sa bude hýbať ako želé." |  | ||||||
|   tada: "Animácia (tadá)" |  | ||||||
|   tadaDescription: "Obsah sa bude hýbať ako Tada!" |  | ||||||
|   jump: "Animácia (skok)" |  | ||||||
|   jumpDescription: "Obsah skočí." |  | ||||||
|   bounce: "Animácia (odraz)" |  | ||||||
|   bounceDescription: "Obsah sa bude odrážať." |  | ||||||
|   shake: "Animácia (trasenie)" |  | ||||||
|   shakeDescription: "Obsah sa bude triasť." |  | ||||||
|   twitch: "Animácia (myknutie)" |  | ||||||
|   twitchDescription: "Obsahu dá animáciu silného trasenia." |  | ||||||
|   spin: "Animácia (rotácia)" |  | ||||||
|   spinDescription: "Obsahu pridá otáčajúcu animáciu." |  | ||||||
|   x2: "Veľký" |  | ||||||
|   x2Description: "Zobrazí obsah väčší." |  | ||||||
|   x3: "Veľmi veľký" |  | ||||||
|   x3Description: "Zobrazí obsah ešte väčší." |  | ||||||
|   x4: "Neuveriteľne veľký" |  | ||||||
|   x4Description: "Zobrazí obsah ešte viac veľký než veľmi veľký." |  | ||||||
|   blur: "Rozmazanie" |  | ||||||
|   blurDescription: "Týmto efektom môže byť obsah rozmazaný. Zaostrí sa keď ned neho príde kurzor." |  | ||||||
|   font: "Písmo" |  | ||||||
|   fontDescription: "Nastaví písmo, ktorým sa zobrazí text." |  | ||||||
|   rainbow: "Dúha" |  | ||||||
|   rainbowDescription: "Zobrazí obsah vo farbách dúhy." |  | ||||||
|   sparkle: "Trblietky" |  | ||||||
|   sparkleDescription: "Obsahu dodá trblietajúci efekt." |  | ||||||
|   rotate: "Otáčať" |  | ||||||
|   rotateDescription: "Otočí obsah o určitý uhol." |  | ||||||
|   plain: "Obyčajné" |  | ||||||
|   plainDescription: "Bez akejkoľvej syntaxe" |  | ||||||
| _instanceTicker: | _instanceTicker: | ||||||
|   none: "Nikdy nezobrazovať" |   none: "Nikdy nezobrazovať" | ||||||
|   remote: "Zobraziť pre vzdialených používateľov" |   remote: "Zobraziť pre vzdialených používateľov" | ||||||
| @@ -1545,5 +1478,6 @@ _deck: | |||||||
|     tl: "Časová os" |     tl: "Časová os" | ||||||
|     antenna: "Antény" |     antenna: "Antény" | ||||||
|     list: "Zoznam" |     list: "Zoznam" | ||||||
|  |     channel: "Kanály" | ||||||
|     mentions: "Zmienky" |     mentions: "Zmienky" | ||||||
|     direct: "Priame poznámky" |     direct: "Priame poznámky" | ||||||
|   | |||||||
| @@ -371,11 +371,6 @@ pushNotificationNotSupported: "Din webbläsare eller instans har inte stöd för | |||||||
| _email: | _email: | ||||||
|   _follow: |   _follow: | ||||||
|     title: "följde dig" |     title: "följde dig" | ||||||
| _mfm: |  | ||||||
|   mention: "Nämn" |  | ||||||
|   quote: "Citat" |  | ||||||
|   emoji: "Anpassa emoji" |  | ||||||
|   search: "Sök" |  | ||||||
| _channel: | _channel: | ||||||
|   setBanner: "Välj banner" |   setBanner: "Välj banner" | ||||||
|   removeBanner: "Ta bort banner" |   removeBanner: "Ta bort banner" | ||||||
|   | |||||||
| @@ -129,6 +129,7 @@ unblockConfirm: "คุณแน่ใจแล้วเหรอ? ว่าต | |||||||
| suspendConfirm: "นายแน่ใจแล้วเหรอว่าต้องการระงับบัญชีนี้อ่ะ?" | suspendConfirm: "นายแน่ใจแล้วเหรอว่าต้องการระงับบัญชีนี้อ่ะ?" | ||||||
| unsuspendConfirm: "นายแน่ใจแล้วหรอ? ว่าต้องการยกเลิกการระงับบัญชีนี้" | unsuspendConfirm: "นายแน่ใจแล้วหรอ? ว่าต้องการยกเลิกการระงับบัญชีนี้" | ||||||
| selectList: "เลือกรายการ" | selectList: "เลือกรายการ" | ||||||
|  | selectChannel: "เลือกแชนแนล" | ||||||
| selectAntenna: "เลือกเสาอากาศ" | selectAntenna: "เลือกเสาอากาศ" | ||||||
| selectWidget: "เลือกวิดเจ็ต" | selectWidget: "เลือกวิดเจ็ต" | ||||||
| editWidgets: "แก้ไขวิดเจ็ต" | editWidgets: "แก้ไขวิดเจ็ต" | ||||||
| @@ -464,7 +465,6 @@ youHaveNoGroups: "คุณยังไม่มีกลุ่ม" | |||||||
| joinOrCreateGroup: "รับเชิญเข้าร่วมกลุ่มหรือสร้างกลุ่มของคุณเองเลยนะ" | joinOrCreateGroup: "รับเชิญเข้าร่วมกลุ่มหรือสร้างกลุ่มของคุณเองเลยนะ" | ||||||
| noHistory: "ไม่มีรายการ" | noHistory: "ไม่มีรายการ" | ||||||
| signinHistory: "ประวัติการเข้าสู่ระบบ" | signinHistory: "ประวัติการเข้าสู่ระบบ" | ||||||
| disableAnimatedMfm: "ปิดการใช้งาน MFM ด้วยแอนิเมชั่น" |  | ||||||
| doing: "กำลังประมวลผล......" | doing: "กำลังประมวลผล......" | ||||||
| category: "หมวดหมู่" | category: "หมวดหมู่" | ||||||
| tags: "แท็ก" | tags: "แท็ก" | ||||||
| @@ -939,6 +939,8 @@ cannotPerformTemporaryDescription: "การดําเนินการน | |||||||
| preset: "พรีเซ็ต" | preset: "พรีเซ็ต" | ||||||
| selectFromPresets: "เลือกจากการพรีเซ็ต" | selectFromPresets: "เลือกจากการพรีเซ็ต" | ||||||
| achievements: "ความสำเร็จ" | achievements: "ความสำเร็จ" | ||||||
|  | gotInvalidResponseError: "การตอบสนองเซิร์ฟเวอร์ไม่ถูกต้อง" | ||||||
|  | gotInvalidResponseErrorDescription: "เซิร์ฟเวอร์อาจไม่สามารถเข้าถึงได้หรืออาจจะกำลังอยู่ในระหว่างปรับปรุง กรุณาลองใหม่อีกครั้งในภายหลังนะคะ" | ||||||
| _achievements: | _achievements: | ||||||
|   earnedAt: "ได้รับเมื่อ" |   earnedAt: "ได้รับเมื่อ" | ||||||
|   _types: |   _types: | ||||||
| @@ -1147,7 +1149,7 @@ _achievements: | |||||||
|       description: "คุณได้คลิกที่นี่" |       description: "คุณได้คลิกที่นี่" | ||||||
|     _justPlainLucky: |     _justPlainLucky: | ||||||
|       title: "แค่ลัคกี้ธรรมดา" |       title: "แค่ลัคกี้ธรรมดา" | ||||||
|       description: "มีโอกาสที่จะได้รับด้วยความน่าจะเป็นไปได้ 0.01% ทุก ๆ 10 วินาที" |       description: "มีโอกาสที่จะได้รับด้วยความน่าจะเป็นไปได้ 0.005% ทุก ๆ 10 วินาที" | ||||||
|     _setNameToSyuilo: |     _setNameToSyuilo: | ||||||
|       title: "พระเจ้าคอมเพล็กซ์" |       title: "พระเจ้าคอมเพล็กซ์" | ||||||
|       description: "ตั้งชื่อของคุณเป็น \"syuilo\"" |       description: "ตั้งชื่อของคุณเป็น \"syuilo\"" | ||||||
| @@ -1182,7 +1184,7 @@ _role: | |||||||
|   description: "คำอธิบายบทบาท" |   description: "คำอธิบายบทบาท" | ||||||
|   permission: "สิทธิ์ตามบทบาท" |   permission: "สิทธิ์ตามบทบาท" | ||||||
|   descriptionOfPermission: "<b>ผู้ดูแลกลั่นกรองเนื้อหา</b> สามารถดำเนินการดูแลขั้นพื้นฐานได้นะ\n<b>ผู้ดูแลระบบ</b> สามารถเปลี่ยนการตั้งค่าทั้งหมดของอินสแตนซ์ได้นะ" |   descriptionOfPermission: "<b>ผู้ดูแลกลั่นกรองเนื้อหา</b> สามารถดำเนินการดูแลขั้นพื้นฐานได้นะ\n<b>ผู้ดูแลระบบ</b> สามารถเปลี่ยนการตั้งค่าทั้งหมดของอินสแตนซ์ได้นะ" | ||||||
|   assignTarget: "กำหนดเป้าหมาย" |   assignTarget: "มอบหมาย" | ||||||
|   descriptionOfAssignTarget: "<b>แมนนวล</b> เพื่อเปลี่ยนผู้ที่เป็นส่วนหนึ่งของบทบาทนี้และใครที่ไม่ใช่ด้วยตนเอง\n<b>เงื่อนไข</b> เพื่อให้ผู้ใช้ได้รับการกำหนดและนำออกจากบทบาทนี้โดยอัตโนมัติตามเงื่อนไขชุดหนึ่ง" |   descriptionOfAssignTarget: "<b>แมนนวล</b> เพื่อเปลี่ยนผู้ที่เป็นส่วนหนึ่งของบทบาทนี้และใครที่ไม่ใช่ด้วยตนเอง\n<b>เงื่อนไข</b> เพื่อให้ผู้ใช้ได้รับการกำหนดและนำออกจากบทบาทนี้โดยอัตโนมัติตามเงื่อนไขชุดหนึ่ง" | ||||||
|   manual: "ปรับเอง" |   manual: "ปรับเอง" | ||||||
|   conditional: "มีเงื่อนไข" |   conditional: "มีเงื่อนไข" | ||||||
| @@ -1195,6 +1197,9 @@ _role: | |||||||
|   baseRole: "บทบาทพื้นฐาน" |   baseRole: "บทบาทพื้นฐาน" | ||||||
|   useBaseValue: "ใช้บทบาทพื้นฐานเริ่มต้น" |   useBaseValue: "ใช้บทบาทพื้นฐานเริ่มต้น" | ||||||
|   chooseRoleToAssign: "เลือกบทบาทที่ต้องการกำหนด" |   chooseRoleToAssign: "เลือกบทบาทที่ต้องการกำหนด" | ||||||
|  |   iconUrl: "ไอคอน URL" | ||||||
|  |   asBadge: "แสดงเป็นตรา" | ||||||
|  |   descriptionOfAsBadge: "ไอคอนของบทบาทนี้จะปรากฏถัดจากชื่อผู้ใช้ของผู้ใช้งานด้วยบทบาทนี้ถ้าหากเปิดใช้งาน" | ||||||
|   canEditMembersByModerator: "อนุญาตให้ผู้ดูแลแก้ไขสมาชิก" |   canEditMembersByModerator: "อนุญาตให้ผู้ดูแลแก้ไขสมาชิก" | ||||||
|   descriptionOfCanEditMembersByModerator: "เมื่อเปิดใช้ ผู้ดูแลนอกเหนือจากผู้ดูแลระบบแล้ว จะสามารถกำหนดและยกเลิกการมอบหมายบทบาทนี้ให้กับผู้ใช้ได้ เมื่อปิด เฉพาะผู้ดูแลระบบเท่านั้นที่จะสามารถกำหนดผู้ใช้ได้นะ" |   descriptionOfCanEditMembersByModerator: "เมื่อเปิดใช้ ผู้ดูแลนอกเหนือจากผู้ดูแลระบบแล้ว จะสามารถกำหนดและยกเลิกการมอบหมายบทบาทนี้ให้กับผู้ใช้ได้ เมื่อปิด เฉพาะผู้ดูแลระบบเท่านั้นที่จะสามารถกำหนดผู้ใช้ได้นะ" | ||||||
|   priority: "ลำดับความสำคัญ" |   priority: "ลำดับความสำคัญ" | ||||||
| @@ -1320,72 +1325,6 @@ _nsfw: | |||||||
|   respect: "ซ่อนสื่อ NSFW" |   respect: "ซ่อนสื่อ NSFW" | ||||||
|   ignore: "อย่าซ่อนสื่อ NSFW" |   ignore: "อย่าซ่อนสื่อ NSFW" | ||||||
|   force: "ซ่อนสื่อทั้งหมด" |   force: "ซ่อนสื่อทั้งหมด" | ||||||
| _mfm: |  | ||||||
|   cheatSheet: "โค้ด MFM Cheat Sheet" |  | ||||||
|   intro: "MFM เป็นภาษามาร์กอัปพิเศษเฉพาะของ Misskey ที่สามารถใช้ได้ในหลายที่ คุณยังสามารถดูรายการไวยากรณ์ MFM ที่มีอยู่ทั้งหมดได้ที่นี่นะ" |  | ||||||
|   dummy: "Misskey ขยายโลกของ Fediverse" |  | ||||||
|   mention: "กล่าวถึง" |  | ||||||
|   mentionDescription: "คุณสามารถระบุผู้ใช้โดยใช้ At-Symbol และชื่อผู้ใช้ได้นะ" |  | ||||||
|   hashtag: "แฮชแท็ก" |  | ||||||
|   hashtagDescription: "คุณสามารถระบุชื่อแฮชแท็กได้โดยใช้เครื่องหมายตัวเลขและข้อความได้นะ" |  | ||||||
|   url: "URL" |  | ||||||
|   urlDescription: "สามารถแสดง URL ได้นะ" |  | ||||||
|   link: "ลิงก์" |  | ||||||
|   linkDescription: "เจาะจงเฉพาะ ส่วนของข้อความที่สามารถแสดงเป็น URL ได้" |  | ||||||
|   bold: "ตัวหนา" |  | ||||||
|   boldDescription: "ไฮไลท์ตัวอักษรโดยทำให้หนาขึ้น" |  | ||||||
|   small: "ขนาดเล็ก" |  | ||||||
|   smallDescription: "แสดงผลเนื้อหาขนาดเล็กและบาง" |  | ||||||
|   center: "เซ็นเตอร์" |  | ||||||
|   centerDescription: "แสดงผลเนื้อหาเป็นศูนย์กลาง" |  | ||||||
|   inlineCode: "โค้ด (อินไลน์)" |  | ||||||
|   inlineCodeDescription: "แสดงผลการเน้นไวยากรณ์แบบอินไลน์สำหรับโค้ด (โปรแกรม)" |  | ||||||
|   blockCode: "โค้ด (บล็อก)" |  | ||||||
|   blockCodeDescription: "แสดงผลการเน้นไวยากรณ์สำหรับโค้ดหลายบรรทัด (โปรแกรม) ในบล็อก" |  | ||||||
|   inlineMath: "คณิต (อินไลน์)" |  | ||||||
|   inlineMathDescription: "แสดงผลสูตรคณิต (KaTeX) ในบรรทัด" |  | ||||||
|   blockMath: "คณิต (บล็อก)" |  | ||||||
|   blockMathDescription: "แสดงผลสูตรคณิตหลายบรรทัด (KaTeX) ในบล็อก" |  | ||||||
|   quote: "อ้างคำพูด" |  | ||||||
|   quoteDescription: "แสดงผลเนื้อหาเป็นใบเสนอราคา" |  | ||||||
|   emoji: "กำหนดอีโมจิเอง" |  | ||||||
|   emojiDescription: "โดยล้อมรอบชื่ออีโมจิที่กำหนดเองด้วยเครื่องหมายทวิภาค จะสามารถแสดงผลอีโมจิที่กำหนดเองได้" |  | ||||||
|   search: "ค้นหา" |  | ||||||
|   searchDescription: "แสดงผลกล่องค้นหาพร้อมกับข้อความที่ป้อนไว้ล่วงหน้า" |  | ||||||
|   flip: "พลิก" |  | ||||||
|   flipDescription: "พลิกเนื้อหาในแนวนอนหรือแนวตั้ง" |  | ||||||
|   jelly: "แอนิเมชั่น (เยลลี่)" |  | ||||||
|   jellyDescription: "ให้เนื้อหาเป็นแอนิเมชั่นเหมือนเยลลี่" |  | ||||||
|   tada: "แอนิเมชั่น (ธาดา)" |  | ||||||
|   tadaDescription: "ให้เนื้อหาเป็นแอนิเมชั่นเหมือน \"ทาด้า!\"" |  | ||||||
|   jump: "อนิเมชั่น (กระโดด)" |  | ||||||
|   jumpDescription: "ให้เนื้อหามีภาพเคลื่อนไหวแบบกระโดด" |  | ||||||
|   bounce: "อนิเมชั่น (เด้ง)" |  | ||||||
|   bounceDescription: "ให้เนื้อหามีอนิเมชั่นเด้ง" |  | ||||||
|   shake: "อนิเมชั่น (เขย่า)" |  | ||||||
|   shakeDescription: "ให้เนื้อหามีภาพเคลื่อนไหวสั่น" |  | ||||||
|   twitch: "แอนิเมชั่น (Twitch)" |  | ||||||
|   twitchDescription: "ให้เนื้อหามีแอนิเมชั่นกระตุกอย่างแรง" |  | ||||||
|   spin: "แอนิเมชั่น (สปิน)" |  | ||||||
|   spinDescription: "ให้เนื้อหาเป็นภาพเคลื่อนไหวแบบหมุน" |  | ||||||
|   x2: "ขนาดใหญ่" |  | ||||||
|   x2Description: "แสดงเนื้อหาที่ใหญ่ขึ้น" |  | ||||||
|   x3: "ใหญ่มาก" |  | ||||||
|   x3Description: "แสดงเนื้อหาอีเว้นท์ที่ใหญ่ขึ้น" |  | ||||||
|   x4: "ใหญ่อย่างไม่น่าเชื่อ" |  | ||||||
|   x4Description: "แสดงผลเนื้อหาที่ใหญ่กว่าใหญ่กว่าขนาดใหญ่" |  | ||||||
|   blur: "เบลอ" |  | ||||||
|   blurDescription: "เบลอเนื้อหา จะแสดงผลอย่างชัดเจนต่อเมื่อวางเมาส์เหนือ" |  | ||||||
|   font: "ตัวอักษร" |  | ||||||
|   fontDescription: "ตั้งค่าตัวอักษรเพื่อแสดงเนื้อหาใน" |  | ||||||
|   rainbow: "สายรุ้ง" |  | ||||||
|   rainbowDescription: "ทำให้เนื้อหานั้นปรากฏเป็นสีรุ้ง" |  | ||||||
|   sparkle: "กลิตเตอร์" |  | ||||||
|   sparkleDescription: "ให้เนื้อหานั้นมีเอฟเฟกต์แบบอนุภาคประกาย" |  | ||||||
|   rotate: "หมุนหน้าจอ" |  | ||||||
|   rotateDescription: "เปลี่ยนเนื้อหาตามด้วยมุมที่ระบุไว้" |  | ||||||
|   plain: "เรียบง่าย" |  | ||||||
|   plainDescription: "ปิดการใช้งานเอฟเฟกต์ของ MFM ทั้งหมดที่มีอยู่ในเอฟเฟกต์ MFM นี้" |  | ||||||
| _instanceTicker: | _instanceTicker: | ||||||
|   none: "ไม่ต้องแสดง" |   none: "ไม่ต้องแสดง" | ||||||
|   remote: "แสดงสำหรับผู้ใช้ระยะไกล" |   remote: "แสดงสำหรับผู้ใช้ระยะไกล" | ||||||
| @@ -1590,12 +1529,15 @@ _permissions: | |||||||
|   "read:gallery-likes": "ดูรายการโพสต์ในแกลเลอรีที่ชอบของคุณ" |   "read:gallery-likes": "ดูรายการโพสต์ในแกลเลอรีที่ชอบของคุณ" | ||||||
|   "write:gallery-likes": "แก้ไขรายการโพสต์ในแกลเลอรีที่ชอบของคุณ" |   "write:gallery-likes": "แก้ไขรายการโพสต์ในแกลเลอรีที่ชอบของคุณ" | ||||||
| _auth: | _auth: | ||||||
|  |   shareAccessTitle: "การให้สิทธิ์แอปพลิเคชัน" | ||||||
|   shareAccess: "คุณต้องการอนุญาตให้ \"{name}\" เข้าถึงบัญชีนี้เลยมั้ย?" |   shareAccess: "คุณต้องการอนุญาตให้ \"{name}\" เข้าถึงบัญชีนี้เลยมั้ย?" | ||||||
|   shareAccessAsk: "คุณแน่ใจแล้วจริงๆหรอว่าต้องการอนุญาตให้แอปพลิเคชันนี้เข้าถึงบัญชีของคุณแน่ใจแล้วหรอ?" |   shareAccessAsk: "คุณแน่ใจแล้วจริงๆหรอว่าต้องการอนุญาตให้แอปพลิเคชันนี้เข้าถึงบัญชีของคุณแน่ใจแล้วหรอ?" | ||||||
|  |   permission: "{name} ได้ขอสิทธิ์การเข้าถึงดังต่อไปนี้" | ||||||
|   permissionAsk: "แอปพลิเคชันนี้ขอสิทธิ์ดังต่อไปนี้" |   permissionAsk: "แอปพลิเคชันนี้ขอสิทธิ์ดังต่อไปนี้" | ||||||
|   pleaseGoBack: "กรุณากลับไปที่แอปพลิเคชัน" |   pleaseGoBack: "กรุณากลับไปที่แอปพลิเคชัน" | ||||||
|   callback: "กำลังกลับไปที่แอปพลิเคชัน" |   callback: "กำลังกลับไปที่แอปพลิเคชัน" | ||||||
|   denied: "ปฏิเสธการเข้าใช้" |   denied: "ปฏิเสธการเข้าใช้" | ||||||
|  |   pleaseLogin: "กรุณาเข้าสู่ระบบเพื่ออนุมัติแอปพลิเคชัน" | ||||||
| _antennaSources: | _antennaSources: | ||||||
|   all: "โน้ตทั้งหมด" |   all: "โน้ตทั้งหมด" | ||||||
|   homeTimeline: "โน้ตจากผู้ใช้ที่ติดตาม" |   homeTimeline: "โน้ตจากผู้ใช้ที่ติดตาม" | ||||||
| @@ -1866,5 +1808,6 @@ _deck: | |||||||
|     tl: "ไทม์ไลน์" |     tl: "ไทม์ไลน์" | ||||||
|     antenna: "เสาอากาศ" |     antenna: "เสาอากาศ" | ||||||
|     list: "รายการ" |     list: "รายการ" | ||||||
|  |     channel: "แชนแนล" | ||||||
|     mentions: "พูดถึง" |     mentions: "พูดถึง" | ||||||
|     direct: "ไดเร็ค" |     direct: "ไดเร็ค" | ||||||
|   | |||||||
| @@ -48,8 +48,6 @@ smtpUser: "Kullanıcı Adı" | |||||||
| smtpPass: "Şifre" | smtpPass: "Şifre" | ||||||
| user: "Kullanıcı" | user: "Kullanıcı" | ||||||
| searchByGoogle: "Arama" | searchByGoogle: "Arama" | ||||||
| _mfm: |  | ||||||
|   search: "Arama" |  | ||||||
| _sfx: | _sfx: | ||||||
|   notification: "Bildirim" |   notification: "Bildirim" | ||||||
| _widgets: | _widgets: | ||||||
|   | |||||||
| @@ -2,5 +2,3 @@ | |||||||
| _lang_: "ياپونچە" | _lang_: "ياپونچە" | ||||||
| search: "ئىزدەش" | search: "ئىزدەش" | ||||||
| searchByGoogle: "ئىزدەش" | searchByGoogle: "ئىزدەش" | ||||||
| _mfm: |  | ||||||
|   search: "ئىزدەش" |  | ||||||
|   | |||||||
| @@ -166,7 +166,7 @@ recipient: "Отримувач" | |||||||
| annotation: "Коментарі" | annotation: "Коментарі" | ||||||
| federation: "Федіверс" | federation: "Федіверс" | ||||||
| instances: "Інстанс" | instances: "Інстанс" | ||||||
| registeredAt: "Приєднався(лась)" | registeredAt: "Реєстрація" | ||||||
| latestRequestReceivedAt: "Останній запит прийнято" | latestRequestReceivedAt: "Останній запит прийнято" | ||||||
| latestStatus: "Останній статус" | latestStatus: "Останній статус" | ||||||
| storageUsage: "Використання простору" | storageUsage: "Використання простору" | ||||||
| @@ -263,7 +263,7 @@ activity: "Активність" | |||||||
| images: "Зображення" | images: "Зображення" | ||||||
| birthday: "День народження" | birthday: "День народження" | ||||||
| yearsOld: "{age} років" | yearsOld: "{age} років" | ||||||
| registeredDate: "Приєднався(лась)" | registeredDate: "Приєднання" | ||||||
| location: "Локація" | location: "Локація" | ||||||
| theme: "Тема" | theme: "Тема" | ||||||
| themeForLightMode: "Світла тема" | themeForLightMode: "Світла тема" | ||||||
| @@ -461,7 +461,6 @@ youHaveNoGroups: "Немає груп" | |||||||
| joinOrCreateGroup: "Отримуйте запрошення до груп або створюйте свої власні групи." | joinOrCreateGroup: "Отримуйте запрошення до груп або створюйте свої власні групи." | ||||||
| noHistory: "Історія порожня" | noHistory: "Історія порожня" | ||||||
| signinHistory: "Історія входів" | signinHistory: "Історія входів" | ||||||
| disableAnimatedMfm: "Відключити анімації MFM" |  | ||||||
| doing: "Виконується" | doing: "Виконується" | ||||||
| category: "Категорія" | category: "Категорія" | ||||||
| tags: "Теги" | tags: "Теги" | ||||||
| @@ -529,7 +528,7 @@ state: "Стан" | |||||||
| sort: "Сортування" | sort: "Сортування" | ||||||
| ascendingOrder: "За зростанням" | ascendingOrder: "За зростанням" | ||||||
| descendingOrder: "За спаданням" | descendingOrder: "За спаданням" | ||||||
| scratchpad: "Чернетка" | scratchpad: "Scratchpad" | ||||||
| scratchpadDescription: "Scratchpad надає середовище для експериментів з AiScript. Ви можете писати, виконувати його і тестувати взаємодію з Misskey." | scratchpadDescription: "Scratchpad надає середовище для експериментів з AiScript. Ви можете писати, виконувати його і тестувати взаємодію з Misskey." | ||||||
| output: "Вихід" | output: "Вихід" | ||||||
| script: "Скрипт" | script: "Скрипт" | ||||||
| @@ -1084,22 +1083,35 @@ _achievements: | |||||||
|       description: "Перевищити швидкість домашньої стрічки 20npm (нотаток на хвилину)" |       description: "Перевищити швидкість домашньої стрічки 20npm (нотаток на хвилину)" | ||||||
|     _viewInstanceChart: |     _viewInstanceChart: | ||||||
|       title: "Аналітик" |       title: "Аналітик" | ||||||
|  |     _outputHelloWorldOnScratchpad: | ||||||
|  |       title: "Hello, world!" | ||||||
|  |       description: "Вивести \"hello world\" у Скретчпаді" | ||||||
|  |     _reactWithoutRead: | ||||||
|  |       title: "Прочитали як слід?" | ||||||
|  |       description: "Реакція на нотатку, що містить понад 100 символів, протягом 3 секунд після її публікації" | ||||||
|     _clickedClickHere: |     _clickedClickHere: | ||||||
|       title: "Натисніть тут" |       title: "Натисніть тут" | ||||||
|       description: "Натиснуто тут" |       description: "Натиснуто тут" | ||||||
|  |     _justPlainLucky: | ||||||
|  |       title: "Просто вдача" | ||||||
|  |       description: "Можна отримати з ймовірністю 0,01% кожні 10 секунд" | ||||||
|     _setNameToSyuilo: |     _setNameToSyuilo: | ||||||
|       title: "Комплекс бога" |       title: "Комплекс бога" | ||||||
|       description: "Встановлено ім'я \"syuilo\"" |       description: "Встановлено ім'я \"syuilo\"" | ||||||
|     _passedSinceAccountCreated1: |     _passedSinceAccountCreated1: | ||||||
|       title: "Перша річниця" |       title: "Перша річниця" | ||||||
|  |       description: "Минув рік з моменту створення акаунта" | ||||||
|     _passedSinceAccountCreated2: |     _passedSinceAccountCreated2: | ||||||
|       title: "Друга річниця" |       title: "Друга річниця" | ||||||
|  |       description: "Минуло 2 роки з моменту створення акаунта" | ||||||
|     _passedSinceAccountCreated3: |     _passedSinceAccountCreated3: | ||||||
|       title: "Третя річниця" |       title: "Третя річниця" | ||||||
|       description: "Минуло 3 роки з моменту створення акаунта" |       description: "Минуло 3 роки з моменту створення акаунта" | ||||||
|     _loggedInOnBirthday: |     _loggedInOnBirthday: | ||||||
|       title: "З Днем народження!" |       title: "З Днем народження!" | ||||||
|  |       description: "Увійти у свій день народження" | ||||||
|     _loggedInOnNewYearsDay: |     _loggedInOnNewYearsDay: | ||||||
|  |       title: "З Новим роком!" | ||||||
|       description: "Увійшли в перший день року" |       description: "Увійшли в перший день року" | ||||||
|     _brainDiver: |     _brainDiver: | ||||||
|       title: "Brain Diver" |       title: "Brain Diver" | ||||||
| @@ -1181,65 +1193,6 @@ _nsfw: | |||||||
|   respect: "Приховувати NSFW медіа" |   respect: "Приховувати NSFW медіа" | ||||||
|   ignore: "Не приховувати NSFW медіа" |   ignore: "Не приховувати NSFW медіа" | ||||||
|   force: "Приховувати всі медіа файли" |   force: "Приховувати всі медіа файли" | ||||||
| _mfm: |  | ||||||
|   cheatSheet: " Довідка MFM" |  | ||||||
|   intro: "MFM це ексклюзивна мова розмітки тексту в Misskey, яку можна використовувати в багатьох місцях. Тут ви можете переглянути приклади її синтаксису." |  | ||||||
|   dummy: "Misskey розширює світ Федіверсу" |  | ||||||
|   mention: "Згадка" |  | ||||||
|   mentionDescription: "За допомогою знака \"@\" перед ім'ям можна згадати конкретного користувача." |  | ||||||
|   hashtag: "Хештеґ" |  | ||||||
|   hashtagDescription: "За допомогою знака \"решітка\" перед словом задається хештег." |  | ||||||
|   url: "URL" |  | ||||||
|   urlDescription: "Відображаються URL-адреси." |  | ||||||
|   link: "Посилання" |  | ||||||
|   linkDescription: "Окремі частини тексту можуть містити посилання" |  | ||||||
|   bold: "Жирний шрифт" |  | ||||||
|   boldDescription: "Виділяє літери, роблячи їх товще" |  | ||||||
|   small: "Дрібний шрифт" |  | ||||||
|   smallDescription: "Робить текст маленьким і тонким" |  | ||||||
|   center: "По центру" |  | ||||||
|   centerDescription: "Показує вміст у центрі" |  | ||||||
|   inlineCode: "Код (у рядку)" |  | ||||||
|   inlineCodeDescription: "Показує фрагмент тексту у рядку як програмний код" |  | ||||||
|   blockCode: "Код (блок)" |  | ||||||
|   blockCodeDescription: "Показує кілька рядків тексту як блок програмного кода" |  | ||||||
|   inlineMath: "Формула (у рядку)" |  | ||||||
|   inlineMathDescription: "Відображення математичних формул (KaTeX) у рядку" |  | ||||||
|   blockMath: "Формули (блок)" |  | ||||||
|   blockMathDescription: "Відображати багаторядкові формули (KaTeX) блоками" |  | ||||||
|   quote: "Цитата" |  | ||||||
|   quoteDescription: "Відображає зміст як цитату." |  | ||||||
|   emoji: "Кастомні емоджі" |  | ||||||
|   emojiDescription: "Щоб показати нетиповий емоджі, потрібно ввести його назву в двокрапках." |  | ||||||
|   search: "Пошук" |  | ||||||
|   searchDescription: "Відображає вікно пошуку з попередньо введеним текстом" |  | ||||||
|   flip: "Перевернути" |  | ||||||
|   flipDescription: "Віддзеркалює вміст по горизонталі або вертикалі" |  | ||||||
|   jelly: "Анімація (желе)" |  | ||||||
|   jellyDescription: "Створює желеподібну анімацію" |  | ||||||
|   tada: "Анімація (Тада!)" |  | ||||||
|   tadaDescription: "Створює анімацію з відчуттям \"Тада!\"" |  | ||||||
|   jump: "Анімація (стрибки)" |  | ||||||
|   jumpDescription: "Показує стрибаючу анімацію" |  | ||||||
|   bounce: "Анімація (пружина)" |  | ||||||
|   bounceDescription: "Надає вмісту стрибаючу анімацію." |  | ||||||
|   shake: "Анімація (Shake)" |  | ||||||
|   shakeDescription: "Надає вмісту тремтливу анімацію." |  | ||||||
|   twitch: "Анімація (Twitch)" |  | ||||||
|   spin: "Анімація (Spin)" |  | ||||||
|   x2: "Великий" |  | ||||||
|   x2Description: "Показує контент збільшеним." |  | ||||||
|   x3: "Дуже великий" |  | ||||||
|   x3Description: "Показує контент ще більшим." |  | ||||||
|   x4: "Надзвичайно великий" |  | ||||||
|   x4Description: "Показує контент надзвичайно великим." |  | ||||||
|   blur: "Розмиття" |  | ||||||
|   blurDescription: "Цей ефект зробить контент розмитим. Контент можна зробити чітким, якщо навести на нього вказівник миші." |  | ||||||
|   font: "Шрифт" |  | ||||||
|   fontDescription: "Встановлює шрифт для контенту." |  | ||||||
|   rotate: "Обертати" |  | ||||||
|   plain: "Звичайний" |  | ||||||
|   plainDescription: "Деактивує всі ефекти MFM, що містяться в цьому ефекті MFM." |  | ||||||
| _instanceTicker: | _instanceTicker: | ||||||
|   none: "Не відображати" |   none: "Не відображати" | ||||||
|   remote: "Відображати для віддалених користувачів" |   remote: "Відображати для віддалених користувачів" | ||||||
| @@ -1372,8 +1325,8 @@ _tutorial: | |||||||
|   step1_1: "Ласкаво просимо!" |   step1_1: "Ласкаво просимо!" | ||||||
|   step1_2: "Ця сторінка має назву \"стрічка подій\". На ній з'являються записи користувачів на яких ви підписані." |   step1_2: "Ця сторінка має назву \"стрічка подій\". На ній з'являються записи користувачів на яких ви підписані." | ||||||
|   step1_3: "Наразі ваша стрічка порожня, оскільки ви ще не написали жодної нотатки і не підписані на інших." |   step1_3: "Наразі ваша стрічка порожня, оскільки ви ще не написали жодної нотатки і не підписані на інших." | ||||||
|   step2_1: "Перш ніж зробити запис або підписатись на когось, спочатку заповніть свій обліковий запис." |   step2_1: "Перш ніж зробити запис або підписатись на когось, заповніть свій профіль." | ||||||
|   step2_2: "Надання деякої інформації про себе дозволить іншим користувачам підписатись на вас." |   step2_2: "Надання деякої інформації про себе допоможе іншим користувачам вирішити підписатись на вас." | ||||||
|   step3_1: "Ви успішно налаштували свій обліковий запис?" |   step3_1: "Ви успішно налаштували свій обліковий запис?" | ||||||
|   step3_2: "Наступним кроком є написання нотатки. Це можна зробити, натиснувши зображення олівця на екрані." |   step3_2: "Наступним кроком є написання нотатки. Це можна зробити, натиснувши зображення олівця на екрані." | ||||||
|   step3_3: "Після написання вмісту ви можете опублікувати його, натиснувши кнопку у верхньому правому куті форми." |   step3_3: "Після написання вмісту ви можете опублікувати його, натиснувши кнопку у верхньому правому куті форми." | ||||||
| @@ -1679,5 +1632,6 @@ _deck: | |||||||
|     tl: "Стрічка" |     tl: "Стрічка" | ||||||
|     antenna: "Антени" |     antenna: "Антени" | ||||||
|     list: "Списки" |     list: "Списки" | ||||||
|  |     channel: "Канали" | ||||||
|     mentions: "Згадки" |     mentions: "Згадки" | ||||||
|     direct: "Особисте" |     direct: "Особисте" | ||||||
|   | |||||||
| @@ -457,7 +457,6 @@ youHaveNoGroups: "Không có nhóm nào" | |||||||
| joinOrCreateGroup: "Tham gia hoặc tạo một nhóm mới." | joinOrCreateGroup: "Tham gia hoặc tạo một nhóm mới." | ||||||
| noHistory: "Không có dữ liệu" | noHistory: "Không có dữ liệu" | ||||||
| signinHistory: "Lịch sử đăng nhập" | signinHistory: "Lịch sử đăng nhập" | ||||||
| disableAnimatedMfm: "Tắt MFM với chuyển động" |  | ||||||
| doing: "Đang xử lý..." | doing: "Đang xử lý..." | ||||||
| category: "Phân loại" | category: "Phân loại" | ||||||
| tags: "Thẻ" | tags: "Thẻ" | ||||||
| @@ -992,72 +991,6 @@ _nsfw: | |||||||
|   respect: "Ẩn nội dung NSFW" |   respect: "Ẩn nội dung NSFW" | ||||||
|   ignore: "Hiện nội dung NSFW" |   ignore: "Hiện nội dung NSFW" | ||||||
|   force: "Ẩn mọi media" |   force: "Ẩn mọi media" | ||||||
| _mfm: |  | ||||||
|   cheatSheet: "MFM Cheatsheet" |  | ||||||
|   intro: "MFM là ngôn ngữ phát triển độc quyền của Misskey có thể được sử dụng ở nhiều nơi. Tại đây bạn có thể xem danh sách tất cả các cú pháp MFM có sẵn." |  | ||||||
|   dummy: "Misskey mở rộng thế giới Fediverse" |  | ||||||
|   mention: "Nhắc đến" |  | ||||||
|   mentionDescription: "Bạn có thể nhắc đến ai đó bằng cách sử dụng @tên người dùng." |  | ||||||
|   hashtag: "Hashtag" |  | ||||||
|   hashtagDescription: "Bạn có thể tạo một hashtag bằng #chữ hoặc #số." |  | ||||||
|   url: "URL" |  | ||||||
|   urlDescription: "Những URL có thể hiển thị." |  | ||||||
|   link: "Đường dẫn" |  | ||||||
|   linkDescription: "Các phần cụ thể của văn bản có thể được hiển thị dưới dạng URL." |  | ||||||
|   bold: "In đậm" |  | ||||||
|   boldDescription: "Nổi bật các chữ cái bằng cách làm chúng dày hơn." |  | ||||||
|   small: "Nhỏ" |  | ||||||
|   smallDescription: "Hiển thị nội dung nhỏ và mỏng." |  | ||||||
|   center: "Giữa" |  | ||||||
|   centerDescription: "Hiển thị nội dung căn giữa." |  | ||||||
|   inlineCode: "Mã (Trong dòng)" |  | ||||||
|   inlineCodeDescription: "Hiển thị tô sáng cú pháp trong dòng cho mã (chương trình)." |  | ||||||
|   blockCode: "Mã (Khối)" |  | ||||||
|   blockCodeDescription: "Hiển thị tô sáng cú pháp cho mã nhiều dòng (chương trình) trong một khối." |  | ||||||
|   inlineMath: "Toán học (Trong dòng)" |  | ||||||
|   inlineMathDescription: "Hiển thị công thức toán (KaTeX) trong dòng" |  | ||||||
|   blockMath: "Toán học (Khối)" |  | ||||||
|   blockMathDescription: "Hiển thị công thức toán học nhiều dòng (KaTeX) trong một khối" |  | ||||||
|   quote: "Trích dẫn" |  | ||||||
|   quoteDescription: "Hiển thị nội dung dạng lời trích dạng." |  | ||||||
|   emoji: "Tùy chỉnh emoji" |  | ||||||
|   emojiDescription: "Hiển thị emoji với cú pháp :tên emoji:" |  | ||||||
|   search: "Tìm kiếm" |  | ||||||
|   searchDescription: "Hiển thị hộp tìm kiếm với văn bản được nhập trước." |  | ||||||
|   flip: "Lật" |  | ||||||
|   flipDescription: "Lật nội dung theo chiều ngang hoặc chiều dọc." |  | ||||||
|   jelly: "Chuyển động (Thạch rau câu)" |  | ||||||
|   jellyDescription: "Cho phép nội dung chuyển động giống như thạch rau câu." |  | ||||||
|   tada: "Chuyển động (Tada)" |  | ||||||
|   tadaDescription: "Cho phép nội dung chuyển động kiểu \"Tada!\"." |  | ||||||
|   jump: "Chuyển động (Nhảy múa)" |  | ||||||
|   jumpDescription: "Cho phép nội dung chuyển động nhảy nhót." |  | ||||||
|   bounce: "Chuyển động (Cà tưng)" |  | ||||||
|   bounceDescription: "Cho phép nội dung chuyển động cà tưng." |  | ||||||
|   shake: "Chuyển động (Rung)" |  | ||||||
|   shakeDescription: "Cho phép nội dung chuyển động rung lắc." |  | ||||||
|   twitch: "Chuyển động (Co rút)" |  | ||||||
|   twitchDescription: "Cho phép nội dung chuyển động co rút." |  | ||||||
|   spin: "Chuyển động (Xoay tít)" |  | ||||||
|   spinDescription: "Cho phép nội dung chuyển động xoay tít." |  | ||||||
|   x2: "Lớn" |  | ||||||
|   x2Description: "Hiển thị nội dung cỡ lớn hơn." |  | ||||||
|   x3: "Rất lớn" |  | ||||||
|   x3Description: "Hiển thị nội dung cỡ lớn hơn nữa." |  | ||||||
|   x4: "Khổng lồ" |  | ||||||
|   x4Description: "Hiển thị nội dung cỡ khổng lồ." |  | ||||||
|   blur: "Làm mờ" |  | ||||||
|   blurDescription: "Làm mờ nội dung. Nó sẽ được hiển thị rõ ràng khi di chuột qua." |  | ||||||
|   font: "Phông chữ" |  | ||||||
|   fontDescription: "Chọn phông chữ để hiển thị nội dung." |  | ||||||
|   rainbow: "Cầu vồng" |  | ||||||
|   rainbowDescription: "Làm cho nội dung hiển thị với màu sắc cầu vồng." |  | ||||||
|   sparkle: "Lấp lánh" |  | ||||||
|   sparkleDescription: "Làm cho nội dung hiệu ứng hạt lấp lánh." |  | ||||||
|   rotate: "Xoay" |  | ||||||
|   rotateDescription: "Xoay nội dung theo một góc cụ thể." |  | ||||||
|   plain: "Đơn giản" |  | ||||||
|   plainDescription: "Vô hiệu hóa mọi hiệu ứng MFM chứa trong hiệu ứng MFM này." |  | ||||||
| _instanceTicker: | _instanceTicker: | ||||||
|   none: "Không hiển thị" |   none: "Không hiển thị" | ||||||
|   remote: "Hiện cho người dùng từ máy chủ khác" |   remote: "Hiện cho người dùng từ máy chủ khác" | ||||||
| @@ -1520,5 +1453,6 @@ _deck: | |||||||
|     tl: "Bảng tin" |     tl: "Bảng tin" | ||||||
|     antenna: "Trạm phát sóng" |     antenna: "Trạm phát sóng" | ||||||
|     list: "Danh sách" |     list: "Danh sách" | ||||||
|  |     channel: "Kênh" | ||||||
|     mentions: "Lượt nhắc" |     mentions: "Lượt nhắc" | ||||||
|     direct: "Nhắn riêng" |     direct: "Nhắn riêng" | ||||||
|   | |||||||
| @@ -129,6 +129,7 @@ unblockConfirm: "确定要解除拉黑吗?" | |||||||
| suspendConfirm: "要冻结吗?" | suspendConfirm: "要冻结吗?" | ||||||
| unsuspendConfirm: "要解除冻结吗?" | unsuspendConfirm: "要解除冻结吗?" | ||||||
| selectList: "选择列表" | selectList: "选择列表" | ||||||
|  | selectChannel: "选择频道" | ||||||
| selectAntenna: "选择天线" | selectAntenna: "选择天线" | ||||||
| selectWidget: "选择小工具" | selectWidget: "选择小工具" | ||||||
| editWidgets: "编辑部件" | editWidgets: "编辑部件" | ||||||
| @@ -256,6 +257,8 @@ noMoreHistory: "没有更多的历史记录" | |||||||
| startMessaging: "添加聊天" | startMessaging: "添加聊天" | ||||||
| nUsersRead: "{n}人已读" | nUsersRead: "{n}人已读" | ||||||
| agreeTo: "勾选则表示已阅读并同意{0}" | agreeTo: "勾选则表示已阅读并同意{0}" | ||||||
|  | agreeBelow: "同意以下观点" | ||||||
|  | basicNotesBeforeCreateAccount: "基本注意事项" | ||||||
| tos: "服务条款" | tos: "服务条款" | ||||||
| start: "开始" | start: "开始" | ||||||
| home: "首页" | home: "首页" | ||||||
| @@ -464,7 +467,8 @@ youHaveNoGroups: "没有群组" | |||||||
| joinOrCreateGroup: "请加入一个现有的群组,或者创建新群组。" | joinOrCreateGroup: "请加入一个现有的群组,或者创建新群组。" | ||||||
| noHistory: "没有历史记录" | noHistory: "没有历史记录" | ||||||
| signinHistory: "登录历史" | signinHistory: "登录历史" | ||||||
| disableAnimatedMfm: "禁用MFM动画" | enableAdvancedMfm: "启用扩展MFM" | ||||||
|  | enableAnimatedMfm: "启用MFM动画" | ||||||
| doing: "正在进行" | doing: "正在进行" | ||||||
| category: "类别" | category: "类别" | ||||||
| tags: "标签" | tags: "标签" | ||||||
| @@ -861,6 +865,8 @@ failedToFetchAccountInformation: "获取账户信息失败" | |||||||
| rateLimitExceeded: "已超過速率限制" | rateLimitExceeded: "已超過速率限制" | ||||||
| cropImage: "剪裁图像" | cropImage: "剪裁图像" | ||||||
| cropImageAsk: "是否要裁剪图像?" | cropImageAsk: "是否要裁剪图像?" | ||||||
|  | cropYes: "已裁剪" | ||||||
|  | cropNo: "就这样吧!" | ||||||
| file: "文件" | file: "文件" | ||||||
| recentNHours: "最近{n}小时" | recentNHours: "最近{n}小时" | ||||||
| recentNDays: "最近{n}天" | recentNDays: "最近{n}天" | ||||||
| @@ -939,6 +945,12 @@ cannotPerformTemporaryDescription: "因操作过于频繁,暂时不可用, | |||||||
| preset: "預設值" | preset: "預設值" | ||||||
| selectFromPresets: "從預設值中選擇" | selectFromPresets: "從預設值中選擇" | ||||||
| achievements: "成就" | achievements: "成就" | ||||||
|  | gotInvalidResponseError: "服务器无应答" | ||||||
|  | gotInvalidResponseErrorDescription: "您的网络连接可能出现了问题, 或是远程服务器暂时不可用. 请稍后重试。" | ||||||
|  | thisPostMayBeAnnoying: "这个帖子可能会让其他人感到困扰。" | ||||||
|  | thisPostMayBeAnnoyingHome: "发到首页" | ||||||
|  | thisPostMayBeAnnoyingCancel: "取消" | ||||||
|  | thisPostMayBeAnnoyingIgnore: "就这样发布" | ||||||
| _achievements: | _achievements: | ||||||
|   earnedAt: "达成时间" |   earnedAt: "达成时间" | ||||||
|   _types: |   _types: | ||||||
| @@ -1023,17 +1035,23 @@ _achievements: | |||||||
|       title: "定期联系Ⅲ" |       title: "定期联系Ⅲ" | ||||||
|       description: "总登录天数400天" |       description: "总登录天数400天" | ||||||
|     _login500: |     _login500: | ||||||
|  |       title: "老熟人Ⅰ" | ||||||
|       description: "总登录天数500天" |       description: "总登录天数500天" | ||||||
|       flavor: "诸君,我喜欢贴文" |       flavor: "诸君,我喜欢贴文" | ||||||
|     _login600: |     _login600: | ||||||
|  |       title: "老熟人Ⅱ" | ||||||
|       description: "总登录天数600天" |       description: "总登录天数600天" | ||||||
|     _login700: |     _login700: | ||||||
|  |       title: "老熟人Ⅲ" | ||||||
|       description: "总登录天数700天" |       description: "总登录天数700天" | ||||||
|     _login800: |     _login800: | ||||||
|  |       title: "帖子大师Ⅰ" | ||||||
|       description: "总登录天数800天" |       description: "总登录天数800天" | ||||||
|     _login900: |     _login900: | ||||||
|  |       title: "帖子大师Ⅱ" | ||||||
|       description: "总登录天数900天" |       description: "总登录天数900天" | ||||||
|     _login1000: |     _login1000: | ||||||
|  |       title: "帖子大师Ⅲ" | ||||||
|       description: "总登录天数1000天" |       description: "总登录天数1000天" | ||||||
|       flavor: "感谢您使用Misskey!" |       flavor: "感谢您使用Misskey!" | ||||||
|     _noteClipped1: |     _noteClipped1: | ||||||
| @@ -1072,19 +1090,22 @@ _achievements: | |||||||
|       description: "第一次被关注" |       description: "第一次被关注" | ||||||
|     _followers10: |     _followers10: | ||||||
|       title: "关注我吧!" |       title: "关注我吧!" | ||||||
|       description: "关注者超过10人" |       description: "拥有超过10名关注者" | ||||||
|     _followers50: |     _followers50: | ||||||
|       title: "三五成群" |       title: "三五成群" | ||||||
|       description: "关注者超过50人" |       description: "拥有超过50名关注者" | ||||||
|     _followers100: |     _followers100: | ||||||
|       title: "胜友如云" |       title: "胜友如云" | ||||||
|       description: "关注者超过100人" |       description: "拥有超过100名关注者" | ||||||
|     _followers300: |     _followers300: | ||||||
|       title: "排列成行" |       title: "排列成行" | ||||||
|       description: "关注者超过300人" |       description: "拥有超过300名关注者" | ||||||
|     _followers500: |     _followers500: | ||||||
|       title: "风向标" |       title: "信号塔" | ||||||
|       description: "关注者超过500人" |       description: "拥有超过500名关注者" | ||||||
|  |     _followers1000: | ||||||
|  |       title: "大影响家" | ||||||
|  |       description: "拥有超过1000名关注者" | ||||||
|     _collectAchievements30: |     _collectAchievements30: | ||||||
|       title: "成就收藏家" |       title: "成就收藏家" | ||||||
|       description: "获得超过30个成就" |       description: "获得超过30个成就" | ||||||
| @@ -1096,6 +1117,7 @@ _achievements: | |||||||
|       description: "发布\"I ❤ #Misskey\"帖子" |       description: "发布\"I ❤ #Misskey\"帖子" | ||||||
|       flavor: "感谢您使用 Misskey ! by 开发团队" |       flavor: "感谢您使用 Misskey ! by 开发团队" | ||||||
|     _foundTreasure: |     _foundTreasure: | ||||||
|  |       title: "寻宝" | ||||||
|       description: "发现了隐藏的宝藏" |       description: "发现了隐藏的宝藏" | ||||||
|     _client30min: |     _client30min: | ||||||
|       title: "休息一下!" |       title: "休息一下!" | ||||||
| @@ -1104,7 +1126,7 @@ _achievements: | |||||||
|       title: "无话可说" |       title: "无话可说" | ||||||
|       description: "发帖后一分钟内就将其删除" |       description: "发帖后一分钟内就将其删除" | ||||||
|     _postedAtLateNight: |     _postedAtLateNight: | ||||||
|       title: "夜行者" |       title: "夜猫子" | ||||||
|       description: "深夜发布帖子" |       description: "深夜发布帖子" | ||||||
|       flavor: "差不多该去睡了喔。" |       flavor: "差不多该去睡了喔。" | ||||||
|     _postedAt0min0sec: |     _postedAt0min0sec: | ||||||
| @@ -1112,15 +1134,23 @@ _achievements: | |||||||
|       description: "在0点发布一篇帖子" |       description: "在0点发布一篇帖子" | ||||||
|       flavor: "嘣 嘣 嘣 Biu——!" |       flavor: "嘣 嘣 嘣 Biu——!" | ||||||
|     _selfQuote: |     _selfQuote: | ||||||
|       title: "自我提及" |       title: "自我引用" | ||||||
|       description: "引用了自己的帖子" |       description: "引用了自己的帖子" | ||||||
|  |     _htl20npm: | ||||||
|  |       title: "流动的时间线" | ||||||
|  |       description: "在首页时间线的流速超过20npm" | ||||||
|  |     _viewInstanceChart: | ||||||
|  |       title: "分析师" | ||||||
|  |       description: "查看了实例信息中的图表" | ||||||
|     _outputHelloWorldOnScratchpad: |     _outputHelloWorldOnScratchpad: | ||||||
|       title: "Hello, world!" |       title: "Hello, world!" | ||||||
|  |       description: "在AiScript控制台中输出 hello world" | ||||||
|     _open3windows: |     _open3windows: | ||||||
|       title: "多窗口" |       title: "多窗口" | ||||||
|       description: "打开了三个或更多的窗口" |       description: "打开了三个或更多的窗口" | ||||||
|     _driveFolderCircularReference: |     _driveFolderCircularReference: | ||||||
|       title: "循环引用" |       title: "循环引用" | ||||||
|  |       description: "试图对网盘中的文件夹进行循环嵌套" | ||||||
|     _reactWithoutRead: |     _reactWithoutRead: | ||||||
|       title: "有好好读过吗?" |       title: "有好好读过吗?" | ||||||
|       description: "在含有100字以上的帖子被发出三秒内做出回应" |       description: "在含有100字以上的帖子被发出三秒内做出回应" | ||||||
| @@ -1129,7 +1159,7 @@ _achievements: | |||||||
|       description: "点了这里" |       description: "点了这里" | ||||||
|     _justPlainLucky: |     _justPlainLucky: | ||||||
|       title: "超高校级的幸运" |       title: "超高校级的幸运" | ||||||
|       description: "每10秒有0.01的概率获得" |       description: "每10秒有0.01的概率自动获得" | ||||||
|     _setNameToSyuilo: |     _setNameToSyuilo: | ||||||
|       title: "像神一样呐" |       title: "像神一样呐" | ||||||
|       description: "将名称设定为syuilo" |       description: "将名称设定为syuilo" | ||||||
| @@ -1177,6 +1207,9 @@ _role: | |||||||
|   baseRole: "基本角色" |   baseRole: "基本角色" | ||||||
|   useBaseValue: "使用基本角色的值" |   useBaseValue: "使用基本角色的值" | ||||||
|   chooseRoleToAssign: "选择要分配的角色" |   chooseRoleToAssign: "选择要分配的角色" | ||||||
|  |   iconUrl: "图标URL" | ||||||
|  |   asBadge: "作为徽章显示" | ||||||
|  |   descriptionOfAsBadge: "开启后,用户名旁边将会出现角色图标。" | ||||||
|   canEditMembersByModerator: "允许监察者编辑成员" |   canEditMembersByModerator: "允许监察者编辑成员" | ||||||
|   descriptionOfCanEditMembersByModerator: "如果选中,监察者和管理员都能够为用户分配/取消分配角色。如果未选中,则只有管理员可以执行此操作。" |   descriptionOfCanEditMembersByModerator: "如果选中,监察者和管理员都能够为用户分配/取消分配角色。如果未选中,则只有管理员可以执行此操作。" | ||||||
|   priority: "优先级" |   priority: "优先级" | ||||||
| @@ -1302,72 +1335,6 @@ _nsfw: | |||||||
|   respect: "隐藏敏感内容" |   respect: "隐藏敏感内容" | ||||||
|   ignore: "不隐藏敏感内容" |   ignore: "不隐藏敏感内容" | ||||||
|   force: "总是隐藏内容" |   force: "总是隐藏内容" | ||||||
| _mfm: |  | ||||||
|   cheatSheet: "MFM代码速查表" |  | ||||||
|   intro: "MFM是一种在Misskey中的各个位置使用的专用标记语言。在这里您可以看到MFM中可用的语法列表。" |  | ||||||
|   dummy: "通过Misskey扩展联邦宇宙的世界" |  | ||||||
|   mention: "提及" |  | ||||||
|   mentionDescription: "可以使用 @+用户名 来指示特定用户" |  | ||||||
|   hashtag: "话题标签" |  | ||||||
|   hashtagDescription: "可以使用井号+文字来表示话题标签。" |  | ||||||
|   url: "URL" |  | ||||||
|   urlDescription: "可以表示URL地址。" |  | ||||||
|   link: "链接" |  | ||||||
|   linkDescription: "可以将部分文字和URL关联起来。" |  | ||||||
|   bold: "粗体" |  | ||||||
|   boldDescription: "可以将文字显示为粗体来表示强调。" |  | ||||||
|   small: "缩小" |  | ||||||
|   smallDescription: "可以使内容文字变小、变淡。" |  | ||||||
|   center: "居中" |  | ||||||
|   centerDescription: "可以将内容居中显示。" |  | ||||||
|   inlineCode: "代码(内嵌)" |  | ||||||
|   inlineCodeDescription: "将文字中的程序代码语法高亮显示。" |  | ||||||
|   blockCode: "代码(块)" |  | ||||||
|   blockCodeDescription: "语法高亮显示整块程序代码。" |  | ||||||
|   inlineMath: "数学公式(内嵌)" |  | ||||||
|   inlineMathDescription: "显示内嵌的KaTex公式。" |  | ||||||
|   blockMath: "数学公式(块)" |  | ||||||
|   blockMathDescription: "显示整块的多行KaTex数学公式。" |  | ||||||
|   quote: "引用" |  | ||||||
|   quoteDescription: "可以用来表示引用的内容。" |  | ||||||
|   emoji: "自定义表情符号" |  | ||||||
|   emojiDescription: "可以将自定义表情符号使用冒号括起来,就可以显示自定义表情符号了。" |  | ||||||
|   search: "搜索" |  | ||||||
|   searchDescription: "显示含有搜索内容示例的搜索框。" |  | ||||||
|   flip: "翻转" |  | ||||||
|   flipDescription: "将内容上下或左右翻转。" |  | ||||||
|   jelly: "动画(果冻)" |  | ||||||
|   jellyDescription: "显示果冻一样的动画效果。" |  | ||||||
|   tada: "动画(锵锵)" |  | ||||||
|   tadaDescription: "显示\"锵锵!\"的动画效果。" |  | ||||||
|   jump: "动画(跳动)" |  | ||||||
|   jumpDescription: "显示跳动的动画效果。" |  | ||||||
|   bounce: "动画(弹性)" |  | ||||||
|   bounceDescription: "显示弹性一样的动画效果。" |  | ||||||
|   shake: "动画(摇晃)" |  | ||||||
|   shakeDescription: "显示摇晃的动画效果。" |  | ||||||
|   twitch: "动画(颤抖)" |  | ||||||
|   twitchDescription: "显示强烈颤抖的动画效果。" |  | ||||||
|   spin: "动画(旋转)" |  | ||||||
|   spinDescription: "显示旋转的动画效果。" |  | ||||||
|   x2: "大" |  | ||||||
|   x2Description: "以大尺寸显示内容。" |  | ||||||
|   x3: "非常大" |  | ||||||
|   x3Description: "以更大尺寸显示内容。" |  | ||||||
|   x4: "最大" |  | ||||||
|   x4Description: "以最大尺寸显示内容。" |  | ||||||
|   blur: "模糊" |  | ||||||
|   blurDescription: "产生模糊效果。将鼠标指针放在上面即可将内容显示出来。" |  | ||||||
|   font: "字体" |  | ||||||
|   fontDescription: "可以设置内容所使用的字体。" |  | ||||||
|   rainbow: "彩虹" |  | ||||||
|   rainbowDescription: "用彩虹色来显示内容。" |  | ||||||
|   sparkle: "闪光" |  | ||||||
|   sparkleDescription: "添加发光粒子效果。" |  | ||||||
|   rotate: "旋转" |  | ||||||
|   rotateDescription: "旋转指定的角度。" |  | ||||||
|   plain: "简洁" |  | ||||||
|   plainDescription: "禁用所有内部语法。" |  | ||||||
| _instanceTicker: | _instanceTicker: | ||||||
|   none: "不显示" |   none: "不显示" | ||||||
|   remote: "仅远程用户" |   remote: "仅远程用户" | ||||||
| @@ -1572,12 +1539,15 @@ _permissions: | |||||||
|   "read:gallery-likes": "读取喜欢的图片" |   "read:gallery-likes": "读取喜欢的图片" | ||||||
|   "write:gallery-likes": "操作喜欢的图片" |   "write:gallery-likes": "操作喜欢的图片" | ||||||
| _auth: | _auth: | ||||||
|  |   shareAccessTitle: "应用程序授权许可" | ||||||
|   shareAccess: "您要授权允许“{name}”访问您的帐户吗?" |   shareAccess: "您要授权允许“{name}”访问您的帐户吗?" | ||||||
|   shareAccessAsk: "您确定要授权此应用程序访问您的帐户吗?" |   shareAccessAsk: "您确定要授权此应用程序访问您的帐户吗?" | ||||||
|  |   permission: "{name}需要以下权限" | ||||||
|   permissionAsk: "这个应用程序需要以下权限" |   permissionAsk: "这个应用程序需要以下权限" | ||||||
|   pleaseGoBack: "请返回到应用程序" |   pleaseGoBack: "请返回到应用程序" | ||||||
|   callback: "回到应用程序" |   callback: "回到应用程序" | ||||||
|   denied: "拒绝访问" |   denied: "拒绝访问" | ||||||
|  |   pleaseLogin: "在对应用进行授权许可之前,请先登录" | ||||||
| _antennaSources: | _antennaSources: | ||||||
|   all: "所有帖子" |   all: "所有帖子" | ||||||
|   homeTimeline: "已关注用户的帖子" |   homeTimeline: "已关注用户的帖子" | ||||||
| @@ -1848,5 +1818,6 @@ _deck: | |||||||
|     tl: "时间线" |     tl: "时间线" | ||||||
|     antenna: "天线" |     antenna: "天线" | ||||||
|     list: "列表" |     list: "列表" | ||||||
|  |     channel: "频道" | ||||||
|     mentions: "提及" |     mentions: "提及" | ||||||
|     direct: "指定用户" |     direct: "指定用户" | ||||||
|   | |||||||
| @@ -129,6 +129,7 @@ unblockConfirm: "確定解除封鎖此用戶?" | |||||||
| suspendConfirm: "確定凍結此帳號?" | suspendConfirm: "確定凍結此帳號?" | ||||||
| unsuspendConfirm: "確定解凍此帳號?" | unsuspendConfirm: "確定解凍此帳號?" | ||||||
| selectList: "選擇清單" | selectList: "選擇清單" | ||||||
|  | selectChannel: "選擇頻道" | ||||||
| selectAntenna: "選擇天線" | selectAntenna: "選擇天線" | ||||||
| selectWidget: "選擇小工具" | selectWidget: "選擇小工具" | ||||||
| editWidgets: "編輯小工具" | editWidgets: "編輯小工具" | ||||||
| @@ -256,6 +257,8 @@ noMoreHistory: "沒有更多歷史紀錄" | |||||||
| startMessaging: "開始聊天" | startMessaging: "開始聊天" | ||||||
| nUsersRead: "{n}人已讀" | nUsersRead: "{n}人已讀" | ||||||
| agreeTo: "我同意{0}" | agreeTo: "我同意{0}" | ||||||
|  | agreeBelow: "同意以下內容" | ||||||
|  | basicNotesBeforeCreateAccount: "基本注意事項" | ||||||
| tos: "使用條款" | tos: "使用條款" | ||||||
| start: "開始" | start: "開始" | ||||||
| home: "首頁" | home: "首頁" | ||||||
| @@ -326,7 +329,7 @@ connectService: "己連結" | |||||||
| disconnectService: "己斷開 " | disconnectService: "己斷開 " | ||||||
| enableLocalTimeline: "開啟本地時間軸" | enableLocalTimeline: "開啟本地時間軸" | ||||||
| enableGlobalTimeline: "啟用全域時間軸" | enableGlobalTimeline: "啟用全域時間軸" | ||||||
| disablingTimelinesInfo: "為了方便,即使您關閉了時間線功能,管理員和審核員仍可以繼續使用。" | disablingTimelinesInfo: "為了方便,即使您關閉了時間線功能,管理員和審查員仍可以繼續使用。" | ||||||
| registration: "註冊" | registration: "註冊" | ||||||
| enableRegistration: "開啟新使用者註冊" | enableRegistration: "開啟新使用者註冊" | ||||||
| invite: "邀請" | invite: "邀請" | ||||||
| @@ -389,8 +392,8 @@ aboutMisskey: "關於 Misskey" | |||||||
| administrator: "管理員" | administrator: "管理員" | ||||||
| token: "權杖" | token: "權杖" | ||||||
| twoStepAuthentication: "兩階段驗證" | twoStepAuthentication: "兩階段驗證" | ||||||
| moderator: "審核員" | moderator: "審查員" | ||||||
| moderation: "監察" | moderation: "審查" | ||||||
| nUsersMentioned: "提到了{n}" | nUsersMentioned: "提到了{n}" | ||||||
| securityKey: "安全金鑰" | securityKey: "安全金鑰" | ||||||
| securityKeyName: "金鑰名稱" | securityKeyName: "金鑰名稱" | ||||||
| @@ -464,7 +467,8 @@ youHaveNoGroups: "找不到群組" | |||||||
| joinOrCreateGroup: "請加入現有群組,或創建新群組。" | joinOrCreateGroup: "請加入現有群組,或創建新群組。" | ||||||
| noHistory: "沒有歷史紀錄" | noHistory: "沒有歷史紀錄" | ||||||
| signinHistory: "登入歷史" | signinHistory: "登入歷史" | ||||||
| disableAnimatedMfm: "禁用MFM動畫" | enableAdvancedMfm: "啟用高級MFM" | ||||||
|  | enableAnimatedMfm: "啟用MFM動畫" | ||||||
| doing: "正在進行" | doing: "正在進行" | ||||||
| category: "類別" | category: "類別" | ||||||
| tags: "標籤" | tags: "標籤" | ||||||
| @@ -607,7 +611,7 @@ testEmail: "測試郵件發送" | |||||||
| wordMute: "被靜音的文字" | wordMute: "被靜音的文字" | ||||||
| regexpError: "正規表達式錯誤" | regexpError: "正規表達式錯誤" | ||||||
| regexpErrorDescription: "{tab} 靜音文字的第 {line} 行的正規表達式有錯誤:" | regexpErrorDescription: "{tab} 靜音文字的第 {line} 行的正規表達式有錯誤:" | ||||||
| instanceMute: "實例的靜音" | instanceMute: "被靜音的實例" | ||||||
| userSaysSomething: "{name}說了什麼" | userSaysSomething: "{name}說了什麼" | ||||||
| makeActive: "啟用" | makeActive: "啟用" | ||||||
| display: "檢視" | display: "檢視" | ||||||
| @@ -861,6 +865,8 @@ failedToFetchAccountInformation: "取得帳戶資訊失敗" | |||||||
| rateLimitExceeded: "已超過速率限制" | rateLimitExceeded: "已超過速率限制" | ||||||
| cropImage: "圖片裁剪" | cropImage: "圖片裁剪" | ||||||
| cropImageAsk: "要剪裁圖片嗎?" | cropImageAsk: "要剪裁圖片嗎?" | ||||||
|  | cropYes: "裁剪" | ||||||
|  | cropNo: "使用原圖" | ||||||
| file: "檔案" | file: "檔案" | ||||||
| recentNHours: "過去{n}小時" | recentNHours: "過去{n}小時" | ||||||
| recentNDays: "過去{n}天" | recentNDays: "過去{n}天" | ||||||
| @@ -939,6 +945,13 @@ cannotPerformTemporaryDescription: "由於超過操作次數限制,暫時無 | |||||||
| preset: "預設值" | preset: "預設值" | ||||||
| selectFromPresets: "從預設值中選擇" | selectFromPresets: "從預設值中選擇" | ||||||
| achievements: "成就" | achievements: "成就" | ||||||
|  | gotInvalidResponseError: "伺服器的回應無效" | ||||||
|  | gotInvalidResponseErrorDescription: "伺服器可能已關閉或者在維護中,請稍後再試。" | ||||||
|  | thisPostMayBeAnnoying: "這篇貼文可能會造成別人的困擾。" | ||||||
|  | thisPostMayBeAnnoyingHome: "發布到首頁" | ||||||
|  | thisPostMayBeAnnoyingCancel: "退出" | ||||||
|  | thisPostMayBeAnnoyingIgnore: "直接發布貼文" | ||||||
|  | collapseRenotes: "省略顯示已看過的轉發貼文" | ||||||
| _achievements: | _achievements: | ||||||
|   earnedAt: "獲得日期" |   earnedAt: "獲得日期" | ||||||
|   _types: |   _types: | ||||||
| @@ -1181,7 +1194,7 @@ _role: | |||||||
|   name: "角色名稱" |   name: "角色名稱" | ||||||
|   description: "角色描述 " |   description: "角色描述 " | ||||||
|   permission: "角色的權限" |   permission: "角色的權限" | ||||||
|   descriptionOfPermission: "<b>審核員</b>執行與審核相關的基本操作。\n<b>管理員</b>能變更實例的全部設定。" |   descriptionOfPermission: "<b>審查員</b>執行與審查相關的基本操作。\n<b>管理員</b>能變更實例的全部設定" | ||||||
|   assignTarget: "指派目標" |   assignTarget: "指派目標" | ||||||
|   descriptionOfAssignTarget: "<b>手動</b>是以手動管理這個角色包含的人員。\n<b>符合條件</b>是設定條件以自動包含符合條件的使用者。" |   descriptionOfAssignTarget: "<b>手動</b>是以手動管理這個角色包含的人員。\n<b>符合條件</b>是設定條件以自動包含符合條件的使用者。" | ||||||
|   manual: "手動" |   manual: "手動" | ||||||
| @@ -1195,8 +1208,11 @@ _role: | |||||||
|   baseRole: "基本角色" |   baseRole: "基本角色" | ||||||
|   useBaseValue: "使用基本角色的值" |   useBaseValue: "使用基本角色的值" | ||||||
|   chooseRoleToAssign: "選擇要指派的角色" |   chooseRoleToAssign: "選擇要指派的角色" | ||||||
|   canEditMembersByModerator: "允許編輯監察員的成員" |   iconUrl: "圖示的URL" | ||||||
|   descriptionOfCanEditMembersByModerator: "如果開啟,管理員與監察員都可以為使用者指派/解除指派該角色。如果關閉,則只有管理員可以執行。" |   asBadge: "顯示為徽章" | ||||||
|  |   descriptionOfAsBadge: "開啟的話,角色圖示會顯示在用戶名旁邊。" | ||||||
|  |   canEditMembersByModerator: "允許編輯審查員的成員" | ||||||
|  |   descriptionOfCanEditMembersByModerator: "如果開啟,管理員與審查員都可以為使用者指派/解除指派該角色。如果關閉,則只有管理員可以執行。" | ||||||
|   priority: "優先級" |   priority: "優先級" | ||||||
|   _priority: |   _priority: | ||||||
|     low: "低" |     low: "低" | ||||||
| @@ -1233,7 +1249,7 @@ _role: | |||||||
|     or: "~或~" |     or: "~或~" | ||||||
|     not: "~否" |     not: "~否" | ||||||
| _sensitiveMediaDetection: | _sensitiveMediaDetection: | ||||||
|   description: "您可以使用機器學習自動檢測敏感媒體並將其用於審核。 伺服器的負荷會稍微增加。" |   description: "您可以使用機器學習自動檢測敏感媒體並將其用於審查。 伺服器的負荷會稍微增加。" | ||||||
|   sensitivity: "檢測敏感度" |   sensitivity: "檢測敏感度" | ||||||
|   sensitivityDescription: "敏感度低時,誤檢測(偽陽性)會減少。敏感度高時,漏檢(偽陰性)會減少。" |   sensitivityDescription: "敏感度低時,誤檢測(偽陽性)會減少。敏感度高時,漏檢(偽陰性)會減少。" | ||||||
|   setSensitiveFlagAutomatically: "設定 NSFW 旗標" |   setSensitiveFlagAutomatically: "設定 NSFW 旗標" | ||||||
| @@ -1320,72 +1336,6 @@ _nsfw: | |||||||
|   respect: "隱藏敏感內容" |   respect: "隱藏敏感內容" | ||||||
|   ignore: "不隱藏敏感內容" |   ignore: "不隱藏敏感內容" | ||||||
|   force: "隱藏所有內容" |   force: "隱藏所有內容" | ||||||
| _mfm: |  | ||||||
|   cheatSheet: "MFM代碼小抄" |  | ||||||
|   intro: "MFM是Misskey專用的標記語言,可以在Misskey中的各個位置使用。 您可以這裏看到MFM可用語法列表。" |  | ||||||
|   dummy: "Misskey拓展了Fediverse的世界" |  | ||||||
|   mention: "提及" |  | ||||||
|   mentionDescription: "透過 @+用戶名 來標示特定使用者。" |  | ||||||
|   hashtag: "#tag" |  | ||||||
|   hashtagDescription: "可以使用\"#\"符號後加文字表示話題標籤。" |  | ||||||
|   url: "URL" |  | ||||||
|   urlDescription: "可以展示URL位址。" |  | ||||||
|   link: "鏈接" |  | ||||||
|   linkDescription: "您可以將特定範圍的文章與 URL 相關聯。 " |  | ||||||
|   bold: "粗體" |  | ||||||
|   boldDescription: "可以將文字顯示为粗體来強調。" |  | ||||||
|   small: "縮小" |  | ||||||
|   smallDescription: "可以使內容文字變小、變淡。" |  | ||||||
|   center: "置中" |  | ||||||
|   centerDescription: "可以將內容置中顯示。" |  | ||||||
|   inlineCode: "程式碼(内嵌)" |  | ||||||
|   inlineCodeDescription: "在行內用高亮度顯示,例如程式碼語法。" |  | ||||||
|   blockCode: "程式碼(區塊)" |  | ||||||
|   blockCodeDescription: "在區塊中用高亮度顯示,例如複數行的程式碼語法。" |  | ||||||
|   inlineMath: "數學公式(內嵌)" |  | ||||||
|   inlineMathDescription: "顯示內嵌的KaTex數學公式。" |  | ||||||
|   blockMath: "數學公式(方塊)" |  | ||||||
|   blockMathDescription: "以區塊顯示複數行的KaTex數學式。" |  | ||||||
|   quote: "引用" |  | ||||||
|   quoteDescription: "可以用來表示引用的内容。" |  | ||||||
|   emoji: "自訂表情符號" |  | ||||||
|   emojiDescription: "您可以通過將自定義表情符號名稱括在冒號中來顯示自定義表情符號。 " |  | ||||||
|   search: "搜尋" |  | ||||||
|   searchDescription: "您可以顯示所輸入的搜索框。" |  | ||||||
|   flip: "翻轉" |  | ||||||
|   flipDescription: "將內容上下或左右翻轉。" |  | ||||||
|   jelly: "動畫(果凍)" |  | ||||||
|   jellyDescription: "顯示果凍一樣的動畫效果。" |  | ||||||
|   tada: "動畫(鏘~)" |  | ||||||
|   tadaDescription: "顯示「鏘~!」這種感覺的動畫效果。" |  | ||||||
|   jump: "動畫(跳動)" |  | ||||||
|   jumpDescription: "顯示跳動的動畫效果。" |  | ||||||
|   bounce: "動畫(反彈)" |  | ||||||
|   bounceDescription: "顯示有彈性的動畫效果。" |  | ||||||
|   shake: "動畫(搖晃)" |  | ||||||
|   shakeDescription: "顯示顫抖的動畫效果。" |  | ||||||
|   twitch: "動畫(顫抖)" |  | ||||||
|   twitchDescription: "顯示強烈顫抖的動畫效果。" |  | ||||||
|   spin: "動畫(旋轉)" |  | ||||||
|   spinDescription: "顯示旋轉的動畫效果。" |  | ||||||
|   x2: "大" |  | ||||||
|   x2Description: "放大顯示內容。" |  | ||||||
|   x3: "較大" |  | ||||||
|   x3Description: "放大顯示內容。" |  | ||||||
|   x4: "最大" |  | ||||||
|   x4Description: "將顯示內容放至最大。" |  | ||||||
|   blur: "模糊" |  | ||||||
|   blurDescription: "產生模糊效果。将游標放在上面即可將内容顯示出來。" |  | ||||||
|   font: "字型" |  | ||||||
|   fontDescription: "您可以設定顯示內容的字型" |  | ||||||
|   rainbow: "彩虹" |  | ||||||
|   rainbowDescription: "用彩虹色來顯示內容。" |  | ||||||
|   sparkle: "閃閃發光" |  | ||||||
|   sparkleDescription: "添加閃閃發光的粒子效果。" |  | ||||||
|   rotate: "旋轉" |  | ||||||
|   rotateDescription: "以指定的角度旋轉。" |  | ||||||
|   plain: "簡潔" |  | ||||||
|   plainDescription: "停用全部的內部語法。" |  | ||||||
| _instanceTicker: | _instanceTicker: | ||||||
|   none: "隱藏" |   none: "隱藏" | ||||||
|   remote: "向遠端使用者顯示" |   remote: "向遠端使用者顯示" | ||||||
| @@ -1590,12 +1540,15 @@ _permissions: | |||||||
|   "read:gallery-likes": "讀取喜歡的圖片" |   "read:gallery-likes": "讀取喜歡的圖片" | ||||||
|   "write:gallery-likes": "操作喜歡的圖片" |   "write:gallery-likes": "操作喜歡的圖片" | ||||||
| _auth: | _auth: | ||||||
|  |   shareAccessTitle: "應用程式的存取權限" | ||||||
|   shareAccess: "要授權「“{name}”」存取您的帳戶嗎?" |   shareAccess: "要授權「“{name}”」存取您的帳戶嗎?" | ||||||
|   shareAccessAsk: "您確定要授權這個應用程式使用您的帳戶嗎?" |   shareAccessAsk: "您確定要授權這個應用程式使用您的帳戶嗎?" | ||||||
|  |   permission: "{name}要求以下的權限" | ||||||
|   permissionAsk: "此應用程式需要以下權限" |   permissionAsk: "此應用程式需要以下權限" | ||||||
|   pleaseGoBack: "請返回至應用程式" |   pleaseGoBack: "請返回至應用程式" | ||||||
|   callback: "回到應用程式" |   callback: "回到應用程式" | ||||||
|   denied: "拒絕訪問" |   denied: "拒絕訪問" | ||||||
|  |   pleaseLogin: "必須登入以提供應用程式的存取權限。" | ||||||
| _antennaSources: | _antennaSources: | ||||||
|   all: "全部貼文" |   all: "全部貼文" | ||||||
|   homeTimeline: "來自已追隨使用者的貼文" |   homeTimeline: "來自已追隨使用者的貼文" | ||||||
| @@ -1866,5 +1819,6 @@ _deck: | |||||||
|     tl: "時間軸" |     tl: "時間軸" | ||||||
|     antenna: "天線" |     antenna: "天線" | ||||||
|     list: "清單" |     list: "清單" | ||||||
|  |     channel: "頻道" | ||||||
|     mentions: "提及" |     mentions: "提及" | ||||||
|     direct: "指定使用者" |     direct: "指定使用者" | ||||||
|   | |||||||
							
								
								
									
										26
									
								
								package.json
									
									
									
									
									
								
							
							
						
						
									
										26
									
								
								package.json
									
									
									
									
									
								
							| @@ -1,12 +1,12 @@ | |||||||
| { | { | ||||||
| 	"name": "misskey", | 	"name": "misskey", | ||||||
| 	"version": "13.2.6", | 	"version": "13.6.1", | ||||||
| 	"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@7.24.3", | 	"packageManager": "pnpm@7.27.0", | ||||||
| 	"workspaces": [ | 	"workspaces": [ | ||||||
| 		"packages/frontend", | 		"packages/frontend", | ||||||
| 		"packages/backend", | 		"packages/backend", | ||||||
| @@ -19,7 +19,7 @@ | |||||||
| 		"start": "cd packages/backend && node ./built/boot/index.js", | 		"start": "cd packages/backend && node ./built/boot/index.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/index.js", | ||||||
| 		"init": "pnpm migrate", | 		"init": "pnpm migrate", | ||||||
| 		"migrate": "cd packages/backend && pnpm typeorm migration:run -d ormconfig.js", | 		"migrate": "cd packages/backend && pnpm migrate", | ||||||
| 		"migrateandstart": "pnpm migrate && pnpm start", | 		"migrateandstart": "pnpm migrate && pnpm start", | ||||||
| 		"gulp": "pnpm exec gulp build", | 		"gulp": "pnpm exec gulp build", | ||||||
| 		"watch": "pnpm dev", | 		"watch": "pnpm dev", | ||||||
| @@ -28,8 +28,8 @@ | |||||||
| 		"cy:open": "pnpm cypress open --browser --e2e --config-file=cypress.config.ts", | 		"cy:open": "pnpm cypress open --browser --e2e --config-file=cypress.config.ts", | ||||||
| 		"cy:run": "pnpm cypress run", | 		"cy:run": "pnpm cypress run", | ||||||
| 		"e2e": "pnpm start-server-and-test start:test http://localhost:61812 cy:run", | 		"e2e": "pnpm start-server-and-test start:test http://localhost:61812 cy:run", | ||||||
| 		"jest": "cd packages/backend && pnpm cross-env NODE_ENV=test node --experimental-vm-modules --experimental-import-meta-resolve node_modules/jest/bin/jest.js --forceExit --runInBand", | 		"jest": "cd packages/backend && pnpm jest", | ||||||
| 		"jest-and-coverage": "cd packages/backend && pnpm cross-env NODE_ENV=test node --experimental-vm-modules --experimental-import-meta-resolve node_modules/jest/bin/jest.js --coverage --forceExit --runInBand", | 		"jest-and-coverage": "cd packages/backend && pnpm jest-and-coverage", | ||||||
| 		"test": "pnpm jest", | 		"test": "pnpm jest", | ||||||
| 		"test-and-coverage": "pnpm jest-and-coverage", | 		"test-and-coverage": "pnpm jest-and-coverage", | ||||||
| 		"format": "pnpm exec gulp format", | 		"format": "pnpm exec gulp format", | ||||||
| @@ -38,8 +38,8 @@ | |||||||
| 		"cleanall": "pnpm clean-all" | 		"cleanall": "pnpm clean-all" | ||||||
| 	}, | 	}, | ||||||
| 	"resolutions": { | 	"resolutions": { | ||||||
| 		"chokidar": "^3.5.3", | 		"chokidar": "3.5.3", | ||||||
| 		"lodash": "^4.17.21" | 		"lodash": "4.17.21" | ||||||
| 	}, | 	}, | ||||||
| 	"dependencies": { | 	"dependencies": { | ||||||
| 		"execa": "5.1.1", | 		"execa": "5.1.1", | ||||||
| @@ -49,19 +49,19 @@ | |||||||
| 		"gulp-replace": "1.1.4", | 		"gulp-replace": "1.1.4", | ||||||
| 		"gulp-terser": "2.1.0", | 		"gulp-terser": "2.1.0", | ||||||
| 		"js-yaml": "4.1.0", | 		"js-yaml": "4.1.0", | ||||||
| 		"typescript": "4.9.4" | 		"typescript": "4.9.5" | ||||||
| 	}, | 	}, | ||||||
| 	"devDependencies": { | 	"devDependencies": { | ||||||
| 		"@types/gulp": "4.0.10", | 		"@types/gulp": "4.0.10", | ||||||
| 		"@types/gulp-rename": "2.0.1", | 		"@types/gulp-rename": "2.0.1", | ||||||
| 		"@typescript-eslint/eslint-plugin": "5.49.0", | 		"@typescript-eslint/eslint-plugin": "5.51.0", | ||||||
| 		"@typescript-eslint/parser": "5.49.0", | 		"@typescript-eslint/parser": "5.51.0", | ||||||
| 		"cross-env": "7.0.3", | 		"cross-env": "7.0.3", | ||||||
| 		"cypress": "12.4.0", | 		"cypress": "12.5.1", | ||||||
| 		"eslint": "^8.32.0", | 		"eslint": "8.33.0", | ||||||
| 		"start-server-and-test": "1.15.3" | 		"start-server-and-test": "1.15.3" | ||||||
| 	}, | 	}, | ||||||
| 	"optionalDependencies": { | 	"optionalDependencies": { | ||||||
| 		"@tensorflow/tfjs-core": "^4.2.0" | 		"@tensorflow/tfjs-core": "4.2.0" | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										29
									
								
								packages/backend/migration/1675404035646-cleanup.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										29
									
								
								packages/backend/migration/1675404035646-cleanup.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,29 @@ | |||||||
|  | export class cleanup1675404035646 { | ||||||
|  |     name = 'cleanup1675404035646' | ||||||
|  |  | ||||||
|  |     async up(queryRunner) { | ||||||
|  |         await queryRunner.query(`ALTER TABLE "meta" DROP COLUMN "enableTwitterIntegration"`); | ||||||
|  |         await queryRunner.query(`ALTER TABLE "meta" DROP COLUMN "enableGithubIntegration"`); | ||||||
|  |         await queryRunner.query(`ALTER TABLE "meta" DROP COLUMN "enableDiscordIntegration"`); | ||||||
|  |         await queryRunner.query(`ALTER TABLE "meta" DROP COLUMN "twitterConsumerKey"`); | ||||||
|  |         await queryRunner.query(`ALTER TABLE "meta" DROP COLUMN "twitterConsumerSecret"`); | ||||||
|  |         await queryRunner.query(`ALTER TABLE "meta" DROP COLUMN "githubClientId"`); | ||||||
|  |         await queryRunner.query(`ALTER TABLE "meta" DROP COLUMN "githubClientSecret"`); | ||||||
|  |         await queryRunner.query(`ALTER TABLE "meta" DROP COLUMN "discordClientId"`); | ||||||
|  |         await queryRunner.query(`ALTER TABLE "meta" DROP COLUMN "discordClientSecret"`); | ||||||
|  |         await queryRunner.query(`ALTER TABLE "user_profile" DROP COLUMN "integrations"`); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     async down(queryRunner) { | ||||||
|  |         await queryRunner.query(`ALTER TABLE "user_profile" ADD "integrations" jsonb NOT NULL DEFAULT '{}'`); | ||||||
|  |         await queryRunner.query(`ALTER TABLE "meta" ADD "discordClientSecret" character varying(128)`); | ||||||
|  |         await queryRunner.query(`ALTER TABLE "meta" ADD "discordClientId" character varying(128)`); | ||||||
|  |         await queryRunner.query(`ALTER TABLE "meta" ADD "githubClientSecret" character varying(128)`); | ||||||
|  |         await queryRunner.query(`ALTER TABLE "meta" ADD "githubClientId" character varying(128)`); | ||||||
|  |         await queryRunner.query(`ALTER TABLE "meta" ADD "twitterConsumerSecret" character varying(128)`); | ||||||
|  |         await queryRunner.query(`ALTER TABLE "meta" ADD "twitterConsumerKey" character varying(128)`); | ||||||
|  |         await queryRunner.query(`ALTER TABLE "meta" ADD "enableDiscordIntegration" boolean NOT NULL DEFAULT false`); | ||||||
|  |         await queryRunner.query(`ALTER TABLE "meta" ADD "enableGithubIntegration" boolean NOT NULL DEFAULT false`); | ||||||
|  |         await queryRunner.query(`ALTER TABLE "meta" ADD "enableTwitterIntegration" boolean NOT NULL DEFAULT false`); | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										13
									
								
								packages/backend/migration/1675557528704-role-icon-badge.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								packages/backend/migration/1675557528704-role-icon-badge.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,13 @@ | |||||||
|  | export class roleIconBadge1675557528704 { | ||||||
|  |     name = 'roleIconBadge1675557528704' | ||||||
|  |  | ||||||
|  |     async up(queryRunner) { | ||||||
|  |         await queryRunner.query(`ALTER TABLE "role" ADD "iconUrl" character varying(512)`); | ||||||
|  |         await queryRunner.query(`ALTER TABLE "role" ADD "asBadge" boolean NOT NULL DEFAULT false`); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     async down(queryRunner) { | ||||||
|  |         await queryRunner.query(`ALTER TABLE "role" DROP COLUMN "asBadge"`); | ||||||
|  |         await queryRunner.query(`ALTER TABLE "role" DROP COLUMN "iconUrl"`); | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -1,6 +1,6 @@ | |||||||
| import { DataSource } from 'typeorm'; | import { DataSource } from 'typeorm'; | ||||||
| import { loadConfig } from './built/config.js'; | import { loadConfig } from './built/config.js'; | ||||||
| import { entities } from './built/postgre.js'; | import { entities } from './built/postgres.js'; | ||||||
|  |  | ||||||
| const config = loadConfig(); | const config = loadConfig(); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -19,34 +19,34 @@ | |||||||
| 		"test-and-coverage": "pnpm jest-and-coverage" | 		"test-and-coverage": "pnpm jest-and-coverage" | ||||||
| 	}, | 	}, | ||||||
| 	"optionalDependencies": { | 	"optionalDependencies": { | ||||||
| 		"@tensorflow/tfjs": "^4.2.0", | 		"@tensorflow/tfjs": "4.2.0", | ||||||
| 		"@tensorflow/tfjs-node": "4.2.0" | 		"@tensorflow/tfjs-node": "4.2.0" | ||||||
| 	}, | 	}, | ||||||
| 	"dependencies": { | 	"dependencies": { | ||||||
| 		"@bull-board/api": "^4.11.0", | 		"@bull-board/api": "4.11.1", | ||||||
| 		"@bull-board/fastify": "^4.11.0", | 		"@bull-board/fastify": "4.11.1", | ||||||
| 		"@bull-board/ui": "^4.11.0", | 		"@bull-board/ui": "4.11.1", | ||||||
| 		"@discordapp/twemoji": "14.0.2", | 		"@discordapp/twemoji": "14.0.2", | ||||||
| 		"@fastify/accepts": "4.1.0", | 		"@fastify/accepts": "4.1.0", | ||||||
| 		"@fastify/cookie": "^8.3.0", | 		"@fastify/cookie": "8.3.0", | ||||||
| 		"@fastify/cors": "8.2.0", | 		"@fastify/cors": "8.2.0", | ||||||
| 		"@fastify/http-proxy": "^8.4.0", | 		"@fastify/http-proxy": "8.4.0", | ||||||
| 		"@fastify/multipart": "7.4.0", | 		"@fastify/multipart": "7.4.0", | ||||||
| 		"@fastify/static": "6.7.0", | 		"@fastify/static": "6.8.0", | ||||||
| 		"@fastify/view": "7.4.1", | 		"@fastify/view": "7.4.1", | ||||||
| 		"@nestjs/common": "9.2.1", | 		"@nestjs/common": "9.3.7", | ||||||
| 		"@nestjs/core": "9.2.1", | 		"@nestjs/core": "9.3.7", | ||||||
| 		"@nestjs/testing": "9.2.1", | 		"@nestjs/testing": "9.3.7", | ||||||
| 		"@peertube/http-signature": "1.7.0", | 		"@peertube/http-signature": "1.7.0", | ||||||
| 		"@sinonjs/fake-timers": "10.0.2", | 		"@sinonjs/fake-timers": "10.0.2", | ||||||
| 		"accepts": "^1.3.8", | 		"accepts": "1.3.8", | ||||||
| 		"ajv": "8.12.0", | 		"ajv": "8.12.0", | ||||||
| 		"archiver": "5.3.1", | 		"archiver": "5.3.1", | ||||||
| 		"autwh": "0.1.0", | 		"autwh": "0.1.0", | ||||||
| 		"aws-sdk": "2.1295.0", | 		"aws-sdk": "2.1295.0", | ||||||
| 		"bcryptjs": "2.4.3", | 		"bcryptjs": "2.4.3", | ||||||
| 		"blurhash": "2.0.4", | 		"blurhash": "2.0.4", | ||||||
| 		"bull": "4.10.2", | 		"bull": "4.10.3", | ||||||
| 		"cacheable-lookup": "6.1.0", | 		"cacheable-lookup": "6.1.0", | ||||||
| 		"cbor": "8.1.0", | 		"cbor": "8.1.0", | ||||||
| 		"chalk": "5.2.0", | 		"chalk": "5.2.0", | ||||||
| @@ -62,11 +62,11 @@ | |||||||
| 		"feed": "4.2.2", | 		"feed": "4.2.2", | ||||||
| 		"file-type": "18.2.0", | 		"file-type": "18.2.0", | ||||||
| 		"fluent-ffmpeg": "2.1.2", | 		"fluent-ffmpeg": "2.1.2", | ||||||
| 		"form-data": "^4.0.0", | 		"form-data": "4.0.0", | ||||||
| 		"got": "^12.5.3", | 		"got": "12.5.3", | ||||||
| 		"hpagent": "1.2.0", | 		"hpagent": "1.2.0", | ||||||
| 		"ioredis": "4.28.5", | 		"ioredis": "4.28.5", | ||||||
| 		"ip-cidr": "3.0.11", | 		"ip-cidr": "3.1.0", | ||||||
| 		"is-svg": "4.3.2", | 		"is-svg": "4.3.2", | ||||||
| 		"js-yaml": "4.1.0", | 		"js-yaml": "4.1.0", | ||||||
| 		"jsdom": "21.1.0", | 		"jsdom": "21.1.0", | ||||||
| @@ -75,21 +75,22 @@ | |||||||
| 		"jsrsasign": "10.6.1", | 		"jsrsasign": "10.6.1", | ||||||
| 		"mfm-js": "0.23.3", | 		"mfm-js": "0.23.3", | ||||||
| 		"mime-types": "2.1.35", | 		"mime-types": "2.1.35", | ||||||
| 		"misskey-js": "0.0.14", | 		"misskey-js": "0.0.15", | ||||||
| 		"ms": "3.0.0-canary.1", | 		"ms": "3.0.0-canary.1", | ||||||
| 		"nested-property": "4.0.0", | 		"nested-property": "4.0.0", | ||||||
| 		"nodemailer": "6.9.0", | 		"node-fetch": "3.3.0", | ||||||
|  | 		"nodemailer": "6.9.1", | ||||||
| 		"nsfwjs": "2.4.2", | 		"nsfwjs": "2.4.2", | ||||||
| 		"oauth": "^0.10.0", | 		"oauth": "0.10.0", | ||||||
| 		"os-utils": "0.0.14", | 		"os-utils": "0.0.14", | ||||||
| 		"parse5": "7.1.2", | 		"parse5": "7.1.2", | ||||||
| 		"pg": "8.8.0", | 		"pg": "8.9.0", | ||||||
| 		"private-ip": "3.0.0", | 		"private-ip": "3.0.0", | ||||||
| 		"probe-image-size": "7.2.3", | 		"probe-image-size": "7.2.3", | ||||||
| 		"promise-limit": "2.7.0", | 		"promise-limit": "2.7.0", | ||||||
| 		"pug": "3.0.2", | 		"pug": "3.0.2", | ||||||
| 		"punycode": "2.3.0", | 		"punycode": "2.3.0", | ||||||
| 		"pureimage": "0.3.15", | 		"pureimage": "0.3.17", | ||||||
| 		"qrcode": "1.5.1", | 		"qrcode": "1.5.1", | ||||||
| 		"random-seed": "0.3.0", | 		"random-seed": "0.3.0", | ||||||
| 		"ratelimiter": "3.4.1", | 		"ratelimiter": "3.4.1", | ||||||
| @@ -101,23 +102,22 @@ | |||||||
| 		"rss-parser": "3.12.0", | 		"rss-parser": "3.12.0", | ||||||
| 		"rxjs": "7.8.0", | 		"rxjs": "7.8.0", | ||||||
| 		"s-age": "1.1.2", | 		"s-age": "1.1.2", | ||||||
| 		"sanitize-html": "2.8.1", | 		"sanitize-html": "2.9.0", | ||||||
| 		"seedrandom": "^3.0.5", | 		"seedrandom": "3.0.5", | ||||||
| 		"semver": "7.3.8", | 		"semver": "7.3.8", | ||||||
| 		"sharp": "0.31.3", | 		"sharp": "0.31.3", | ||||||
| 		"speakeasy": "2.0.0", | 		"speakeasy": "2.0.0", | ||||||
| 		"strict-event-emitter-types": "2.0.0", | 		"strict-event-emitter-types": "2.0.0", | ||||||
| 		"stringz": "2.1.0", | 		"stringz": "2.1.0", | ||||||
| 		"summaly": "2.7.0", | 		"summaly": "2.7.0", | ||||||
| 		"syslog-pro": "git+https://github.com/misskey-dev/SyslogPro#0.2.9-misskey.2", | 		"systeminformation": "5.17.8", | ||||||
| 		"systeminformation": "5.17.4", | 		"tinycolor2": "1.6.0", | ||||||
| 		"tinycolor2": "1.5.2", |  | ||||||
| 		"tmp": "0.2.1", | 		"tmp": "0.2.1", | ||||||
| 		"tsc-alias": "1.8.2", | 		"tsc-alias": "1.8.2", | ||||||
| 		"tsconfig-paths": "4.1.2", | 		"tsconfig-paths": "4.1.2", | ||||||
| 		"twemoji-parser": "14.0.0", | 		"twemoji-parser": "14.0.0", | ||||||
| 		"typeorm": "0.3.11", | 		"typeorm": "0.3.11", | ||||||
| 		"typescript": "4.9.4", | 		"typescript": "4.9.5", | ||||||
| 		"ulid": "2.3.0", | 		"ulid": "2.3.0", | ||||||
| 		"unzipper": "0.10.11", | 		"unzipper": "0.10.11", | ||||||
| 		"uuid": "9.0.0", | 		"uuid": "9.0.0", | ||||||
| @@ -125,31 +125,31 @@ | |||||||
| 		"web-push": "3.5.0", | 		"web-push": "3.5.0", | ||||||
| 		"websocket": "1.0.34", | 		"websocket": "1.0.34", | ||||||
| 		"ws": "8.12.0", | 		"ws": "8.12.0", | ||||||
| 		"xev": "3.0.2", | 		"xev": "3.0.2" | ||||||
| 		"node-fetch": "3.3.0" |  | ||||||
| 	}, | 	}, | ||||||
| 	"devDependencies": { | 	"devDependencies": { | ||||||
| 		"@redocly/openapi-core": "1.0.0-beta.120", | 		"@jest/globals": "29.4.2", | ||||||
| 		"@swc/cli": "^0.1.59", | 		"@redocly/openapi-core": "1.0.0-beta.123", | ||||||
| 		"@swc/core": "1.3.29", | 		"@swc/cli": "0.1.61", | ||||||
|  | 		"@swc/core": "1.3.34", | ||||||
| 		"@swc/jest": "0.2.24", | 		"@swc/jest": "0.2.24", | ||||||
| 		"@types/accepts": "1.3.5", | 		"@types/accepts": "1.3.5", | ||||||
| 		"@types/archiver": "5.3.1", | 		"@types/archiver": "5.3.1", | ||||||
| 		"@types/bcryptjs": "2.4.2", | 		"@types/bcryptjs": "2.4.2", | ||||||
| 		"@types/bull": "4.10.0", | 		"@types/bull": "4.10.0", | ||||||
| 		"@types/cbor": "6.0.0", | 		"@types/cbor": "6.0.0", | ||||||
| 		"@types/color-convert": "^2.0.0", | 		"@types/color-convert": "2.0.0", | ||||||
| 		"@types/content-disposition": "^0.5.5", | 		"@types/content-disposition": "0.5.5", | ||||||
| 		"@types/escape-regexp": "0.0.1", | 		"@types/escape-regexp": "0.0.1", | ||||||
| 		"@types/fluent-ffmpeg": "2.1.20", | 		"@types/fluent-ffmpeg": "2.1.20", | ||||||
| 		"@types/ioredis": "4.28.10", | 		"@types/ioredis": "4.28.10", | ||||||
| 		"@types/jest": "29.4.0", | 		"@types/jest": "29.4.0", | ||||||
| 		"@types/js-yaml": "4.0.5", | 		"@types/js-yaml": "4.0.5", | ||||||
| 		"@types/jsdom": "20.0.1", | 		"@types/jsdom": "21.1.0", | ||||||
| 		"@types/jsonld": "1.5.8", | 		"@types/jsonld": "1.5.8", | ||||||
| 		"@types/jsrsasign": "10.5.5", | 		"@types/jsrsasign": "10.5.5", | ||||||
| 		"@types/mime-types": "2.1.1", | 		"@types/mime-types": "2.1.1", | ||||||
| 		"@types/node": "18.11.18", | 		"@types/node": "18.13.0", | ||||||
| 		"@types/node-fetch": "3.0.3", | 		"@types/node-fetch": "3.0.3", | ||||||
| 		"@types/nodemailer": "6.4.7", | 		"@types/nodemailer": "6.4.7", | ||||||
| 		"@types/oauth": "0.9.1", | 		"@types/oauth": "0.9.1", | ||||||
| @@ -166,7 +166,6 @@ | |||||||
| 		"@types/sharp": "0.31.1", | 		"@types/sharp": "0.31.1", | ||||||
| 		"@types/sinonjs__fake-timers": "8.1.2", | 		"@types/sinonjs__fake-timers": "8.1.2", | ||||||
| 		"@types/speakeasy": "2.0.7", | 		"@types/speakeasy": "2.0.7", | ||||||
| 		"@types/syslog-pro": "^1.0.0", |  | ||||||
| 		"@types/tinycolor2": "1.4.3", | 		"@types/tinycolor2": "1.4.3", | ||||||
| 		"@types/tmp": "0.2.3", | 		"@types/tmp": "0.2.3", | ||||||
| 		"@types/unzipper": "0.10.5", | 		"@types/unzipper": "0.10.5", | ||||||
| @@ -175,13 +174,13 @@ | |||||||
| 		"@types/web-push": "3.3.2", | 		"@types/web-push": "3.3.2", | ||||||
| 		"@types/websocket": "1.0.5", | 		"@types/websocket": "1.0.5", | ||||||
| 		"@types/ws": "8.5.4", | 		"@types/ws": "8.5.4", | ||||||
| 		"@typescript-eslint/eslint-plugin": "5.49.0", | 		"@typescript-eslint/eslint-plugin": "5.51.0", | ||||||
| 		"@typescript-eslint/parser": "5.49.0", | 		"@typescript-eslint/parser": "5.51.0", | ||||||
| 		"cross-env": "7.0.3", | 		"cross-env": "7.0.3", | ||||||
| 		"eslint": "8.32.0", | 		"eslint": "8.33.0", | ||||||
| 		"eslint-plugin-import": "2.27.5", | 		"eslint-plugin-import": "2.27.5", | ||||||
| 		"execa": "6.1.0", | 		"execa": "6.1.0", | ||||||
| 		"jest": "29.4.1", | 		"jest": "29.4.2", | ||||||
| 		"jest-mock": "^29.4.1" | 		"jest-mock": "29.4.2" | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|   | |||||||
| @@ -4,7 +4,7 @@ import { DataSource } from 'typeorm'; | |||||||
| import { createRedisConnection } from '@/redis.js'; | import { createRedisConnection } from '@/redis.js'; | ||||||
| import { DI } from './di-symbols.js'; | import { DI } from './di-symbols.js'; | ||||||
| import { loadConfig } from './config.js'; | import { loadConfig } from './config.js'; | ||||||
| import { createPostgreDataSource } from './postgre.js'; | import { createPostgresDataSource } from './postgres.js'; | ||||||
| import { RepositoryModule } from './models/RepositoryModule.js'; | import { RepositoryModule } from './models/RepositoryModule.js'; | ||||||
| import type { Provider, OnApplicationShutdown } from '@nestjs/common'; | import type { Provider, OnApplicationShutdown } from '@nestjs/common'; | ||||||
|  |  | ||||||
| @@ -18,7 +18,7 @@ const $config: Provider = { | |||||||
| const $db: Provider = { | const $db: Provider = { | ||||||
| 	provide: DI.db, | 	provide: DI.db, | ||||||
| 	useFactory: async (config) => { | 	useFactory: async (config) => { | ||||||
| 		const db = createPostgreDataSource(config); | 		const db = createPostgresDataSource(config); | ||||||
| 		return await db.initialize(); | 		return await db.initialize(); | ||||||
| 	}, | 	}, | ||||||
| 	inject: [DI.config], | 	inject: [DI.config], | ||||||
|   | |||||||
| @@ -65,13 +65,9 @@ export type Source = { | |||||||
| 	deliverJobMaxAttempts?: number; | 	deliverJobMaxAttempts?: number; | ||||||
| 	inboxJobMaxAttempts?: number; | 	inboxJobMaxAttempts?: number; | ||||||
|  |  | ||||||
| 	syslog: { |  | ||||||
| 		host: string; |  | ||||||
| 		port: number; |  | ||||||
| 	}; |  | ||||||
|  |  | ||||||
| 	mediaProxy?: string; | 	mediaProxy?: string; | ||||||
| 	proxyRemoteFiles?: boolean; | 	proxyRemoteFiles?: boolean; | ||||||
|  | 	videoThumbnailGenerator?: string; | ||||||
|  |  | ||||||
| 	signToActivityPubGet?: boolean; | 	signToActivityPubGet?: boolean; | ||||||
| }; | }; | ||||||
| @@ -92,6 +88,9 @@ export type Mixin = { | |||||||
| 	userAgent: string; | 	userAgent: string; | ||||||
| 	clientEntry: string; | 	clientEntry: string; | ||||||
| 	clientManifestExists: boolean; | 	clientManifestExists: boolean; | ||||||
|  | 	mediaProxy: string; | ||||||
|  | 	externalMediaProxyEnabled: boolean; | ||||||
|  | 	videoThumbnailGenerator: string | null; | ||||||
| }; | }; | ||||||
|  |  | ||||||
| export type Config = Source & Mixin; | export type Config = Source & Mixin; | ||||||
| @@ -113,7 +112,7 @@ const path = process.env.NODE_ENV === 'test' | |||||||
|  |  | ||||||
| export function loadConfig() { | export function loadConfig() { | ||||||
| 	const meta = JSON.parse(fs.readFileSync(`${_dirname}/../../../built/meta.json`, 'utf-8')); | 	const meta = JSON.parse(fs.readFileSync(`${_dirname}/../../../built/meta.json`, 'utf-8')); | ||||||
| 	const clientManifestExists = fs.existsSync(_dirname + '/../../../built/_vite_/manifest.json') | 	const clientManifestExists = fs.existsSync(_dirname + '/../../../built/_vite_/manifest.json'); | ||||||
| 	const clientManifest = clientManifestExists ? | 	const clientManifest = clientManifestExists ? | ||||||
| 		JSON.parse(fs.readFileSync(`${_dirname}/../../../built/_vite_/manifest.json`, 'utf-8')) | 		JSON.parse(fs.readFileSync(`${_dirname}/../../../built/_vite_/manifest.json`, 'utf-8')) | ||||||
| 		: { 'src/init.ts': { file: 'src/init.ts' } }; | 		: { 'src/init.ts': { file: 'src/init.ts' } }; | ||||||
| @@ -140,6 +139,17 @@ export function loadConfig() { | |||||||
| 	mixin.clientEntry = clientManifest['src/init.ts']; | 	mixin.clientEntry = clientManifest['src/init.ts']; | ||||||
| 	mixin.clientManifestExists = clientManifestExists; | 	mixin.clientManifestExists = clientManifestExists; | ||||||
|  |  | ||||||
|  | 	const externalMediaProxy = config.mediaProxy ? | ||||||
|  | 		config.mediaProxy.endsWith('/') ? config.mediaProxy.substring(0, config.mediaProxy.length - 1) : config.mediaProxy | ||||||
|  | 		: null; | ||||||
|  | 	const internalMediaProxy = `${mixin.scheme}://${mixin.host}/proxy`; | ||||||
|  | 	mixin.mediaProxy = externalMediaProxy ?? internalMediaProxy; | ||||||
|  | 	mixin.externalMediaProxyEnabled = externalMediaProxy !== null && externalMediaProxy !== internalMediaProxy; | ||||||
|  |  | ||||||
|  | 	mixin.videoThumbnailGenerator = config.videoThumbnailGenerator ? | ||||||
|  | 		config.videoThumbnailGenerator.endsWith('/') ? config.videoThumbnailGenerator.substring(0, config.videoThumbnailGenerator.length - 1) : config.videoThumbnailGenerator | ||||||
|  | 		: null; | ||||||
|  |  | ||||||
| 	if (!config.redis.prefix) config.redis.prefix = mixin.host; | 	if (!config.redis.prefix) config.redis.prefix = mixin.host; | ||||||
|  |  | ||||||
| 	return Object.assign(config, mixin); | 	return Object.assign(config, mixin); | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ import { DI } from '@/di-symbols.js'; | |||||||
| import { bindThis } from '@/decorators.js'; | import { bindThis } from '@/decorators.js'; | ||||||
| import { CreateNotificationService } from '@/core/CreateNotificationService.js'; | import { CreateNotificationService } from '@/core/CreateNotificationService.js'; | ||||||
|  |  | ||||||
| const ACHIEVEMENT_TYPES = [ | export const ACHIEVEMENT_TYPES = [ | ||||||
| 	'notes1', | 	'notes1', | ||||||
| 	'notes10', | 	'notes10', | ||||||
| 	'notes100', | 	'notes100', | ||||||
|   | |||||||
| @@ -10,10 +10,9 @@ import { isUserRelated } from '@/misc/is-user-related.js'; | |||||||
| import { GlobalEventService } from '@/core/GlobalEventService.js'; | import { GlobalEventService } from '@/core/GlobalEventService.js'; | ||||||
| import { PushNotificationService } from '@/core/PushNotificationService.js'; | import { PushNotificationService } from '@/core/PushNotificationService.js'; | ||||||
| import * as Acct from '@/misc/acct.js'; | import * as Acct from '@/misc/acct.js'; | ||||||
| import { Cache } from '@/misc/cache.js'; |  | ||||||
| import type { Packed } from '@/misc/schema.js'; | import type { Packed } from '@/misc/schema.js'; | ||||||
| import { DI } from '@/di-symbols.js'; | import { DI } from '@/di-symbols.js'; | ||||||
| import type { MutingsRepository, BlockingsRepository, NotesRepository, AntennaNotesRepository, AntennasRepository, UserGroupJoiningsRepository, UserListJoiningsRepository } from '@/models/index.js'; | import type { MutingsRepository, NotesRepository, AntennaNotesRepository, AntennasRepository, UserGroupJoiningsRepository, UserListJoiningsRepository } from '@/models/index.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 { StreamMessages } from '@/server/api/stream/types.js'; | ||||||
| @@ -23,7 +22,6 @@ import type { OnApplicationShutdown } from '@nestjs/common'; | |||||||
| export class AntennaService implements OnApplicationShutdown { | export class AntennaService implements OnApplicationShutdown { | ||||||
| 	private antennasFetched: boolean; | 	private antennasFetched: boolean; | ||||||
| 	private antennas: Antenna[]; | 	private antennas: Antenna[]; | ||||||
| 	private blockingCache: Cache<User['id'][]>; |  | ||||||
|  |  | ||||||
| 	constructor( | 	constructor( | ||||||
| 		@Inject(DI.redisSubscriber) | 		@Inject(DI.redisSubscriber) | ||||||
| @@ -32,9 +30,6 @@ export class AntennaService implements OnApplicationShutdown { | |||||||
| 		@Inject(DI.mutingsRepository) | 		@Inject(DI.mutingsRepository) | ||||||
| 		private mutingsRepository: MutingsRepository, | 		private mutingsRepository: MutingsRepository, | ||||||
|  |  | ||||||
| 		@Inject(DI.blockingsRepository) |  | ||||||
| 		private blockingsRepository: BlockingsRepository, |  | ||||||
|  |  | ||||||
| 		@Inject(DI.notesRepository) | 		@Inject(DI.notesRepository) | ||||||
| 		private notesRepository: NotesRepository, | 		private notesRepository: NotesRepository, | ||||||
|  |  | ||||||
| @@ -52,14 +47,13 @@ export class AntennaService implements OnApplicationShutdown { | |||||||
|  |  | ||||||
| 		private utilityService: UtilityService, | 		private utilityService: UtilityService, | ||||||
| 		private idService: IdService, | 		private idService: IdService, | ||||||
| 		private globalEventServie: GlobalEventService, | 		private globalEventService: GlobalEventService, | ||||||
| 		private pushNotificationService: PushNotificationService, | 		private pushNotificationService: PushNotificationService, | ||||||
| 		private noteEntityService: NoteEntityService, | 		private noteEntityService: NoteEntityService, | ||||||
| 		private antennaEntityService: AntennaEntityService, | 		private antennaEntityService: AntennaEntityService, | ||||||
| 	) { | 	) { | ||||||
| 		this.antennasFetched = false; | 		this.antennasFetched = false; | ||||||
| 		this.antennas = []; | 		this.antennas = []; | ||||||
| 		this.blockingCache = new Cache<User['id'][]>(1000 * 60 * 5); |  | ||||||
|  |  | ||||||
| 		this.redisSubscriber.on('message', this.onRedisMessage); | 		this.redisSubscriber.on('message', this.onRedisMessage); | ||||||
| 	} | 	} | ||||||
| @@ -109,7 +103,7 @@ export class AntennaService implements OnApplicationShutdown { | |||||||
| 			read: read, | 			read: read, | ||||||
| 		}); | 		}); | ||||||
| 	 | 	 | ||||||
| 		this.globalEventServie.publishAntennaStream(antenna.id, 'note', note); | 		this.globalEventService.publishAntennaStream(antenna.id, 'note', note); | ||||||
| 	 | 	 | ||||||
| 		if (!read) { | 		if (!read) { | ||||||
| 			const mutings = await this.mutingsRepository.find({ | 			const mutings = await this.mutingsRepository.find({ | ||||||
| @@ -139,7 +133,7 @@ export class AntennaService implements OnApplicationShutdown { | |||||||
| 			setTimeout(async () => { | 			setTimeout(async () => { | ||||||
| 				const unread = await this.antennaNotesRepository.findOneBy({ antennaId: antenna.id, read: false }); | 				const unread = await this.antennaNotesRepository.findOneBy({ antennaId: antenna.id, read: false }); | ||||||
| 				if (unread) { | 				if (unread) { | ||||||
| 					this.globalEventServie.publishMainStream(antenna.userId, 'unreadAntenna', antenna); | 					this.globalEventService.publishMainStream(antenna.userId, 'unreadAntenna', antenna); | ||||||
| 					this.pushNotificationService.pushNotification(antenna.userId, 'unreadAntennaNote', { | 					this.pushNotificationService.pushNotification(antenna.userId, 'unreadAntennaNote', { | ||||||
| 						antenna: { id: antenna.id, name: antenna.name }, | 						antenna: { id: antenna.id, name: antenna.name }, | ||||||
| 						note: await this.noteEntityService.pack(note), | 						note: await this.noteEntityService.pack(note), | ||||||
| @@ -156,10 +150,6 @@ export class AntennaService implements OnApplicationShutdown { | |||||||
| 		if (note.visibility === 'specified') return false; | 		if (note.visibility === 'specified') return false; | ||||||
| 		if (note.visibility === 'followers') return false; | 		if (note.visibility === 'followers') return false; | ||||||
| 	 | 	 | ||||||
| 		// アンテナ作成者がノート作成者にブロックされていたらスキップ |  | ||||||
| 		const blockings = await this.blockingCache.fetch(noteUser.id, () => this.blockingsRepository.findBy({ blockerId: noteUser.id }).then(res => res.map(x => x.blockeeId))); |  | ||||||
| 		if (blockings.some(blocking => blocking === antenna.userId)) return false; |  | ||||||
| 	 |  | ||||||
| 		if (!antenna.withReplies && note.replyId != null) return false; | 		if (!antenna.withReplies && note.replyId != null) return false; | ||||||
| 	 | 	 | ||||||
| 		if (antenna.src === 'home') { | 		if (antenna.src === 'home') { | ||||||
|   | |||||||
| @@ -62,7 +62,6 @@ import PerUserNotesChart from './chart/charts/per-user-notes.js'; | |||||||
| import PerUserPvChart from './chart/charts/per-user-pv.js'; | import PerUserPvChart from './chart/charts/per-user-pv.js'; | ||||||
| import DriveChart from './chart/charts/drive.js'; | import DriveChart from './chart/charts/drive.js'; | ||||||
| import PerUserReactionsChart from './chart/charts/per-user-reactions.js'; | import PerUserReactionsChart from './chart/charts/per-user-reactions.js'; | ||||||
| import HashtagChart from './chart/charts/hashtag.js'; |  | ||||||
| import PerUserFollowingChart from './chart/charts/per-user-following.js'; | import PerUserFollowingChart from './chart/charts/per-user-following.js'; | ||||||
| import PerUserDriveChart from './chart/charts/per-user-drive.js'; | import PerUserDriveChart from './chart/charts/per-user-drive.js'; | ||||||
| import ApRequestChart from './chart/charts/ap-request.js'; | import ApRequestChart from './chart/charts/ap-request.js'; | ||||||
| @@ -187,7 +186,6 @@ const $PerUserNotesChart: Provider = { provide: 'PerUserNotesChart', useExisting | |||||||
| const $PerUserPvChart: Provider = { provide: 'PerUserPvChart', useExisting: PerUserPvChart }; | const $PerUserPvChart: Provider = { provide: 'PerUserPvChart', useExisting: PerUserPvChart }; | ||||||
| const $DriveChart: Provider = { provide: 'DriveChart', useExisting: DriveChart }; | const $DriveChart: Provider = { provide: 'DriveChart', useExisting: DriveChart }; | ||||||
| const $PerUserReactionsChart: Provider = { provide: 'PerUserReactionsChart', useExisting: PerUserReactionsChart }; | const $PerUserReactionsChart: Provider = { provide: 'PerUserReactionsChart', useExisting: PerUserReactionsChart }; | ||||||
| const $HashtagChart: Provider = { provide: 'HashtagChart', useExisting: HashtagChart }; |  | ||||||
| const $PerUserFollowingChart: Provider = { provide: 'PerUserFollowingChart', useExisting: PerUserFollowingChart }; | const $PerUserFollowingChart: Provider = { provide: 'PerUserFollowingChart', useExisting: PerUserFollowingChart }; | ||||||
| const $PerUserDriveChart: Provider = { provide: 'PerUserDriveChart', useExisting: PerUserDriveChart }; | const $PerUserDriveChart: Provider = { provide: 'PerUserDriveChart', useExisting: PerUserDriveChart }; | ||||||
| const $ApRequestChart: Provider = { provide: 'ApRequestChart', useExisting: ApRequestChart }; | const $ApRequestChart: Provider = { provide: 'ApRequestChart', useExisting: ApRequestChart }; | ||||||
| @@ -315,7 +313,6 @@ const $ApQuestionService: Provider = { provide: 'ApQuestionService', useExisting | |||||||
| 		PerUserPvChart, | 		PerUserPvChart, | ||||||
| 		DriveChart, | 		DriveChart, | ||||||
| 		PerUserReactionsChart, | 		PerUserReactionsChart, | ||||||
| 		HashtagChart, |  | ||||||
| 		PerUserFollowingChart, | 		PerUserFollowingChart, | ||||||
| 		PerUserDriveChart, | 		PerUserDriveChart, | ||||||
| 		ApRequestChart, | 		ApRequestChart, | ||||||
| @@ -437,7 +434,6 @@ const $ApQuestionService: Provider = { provide: 'ApQuestionService', useExisting | |||||||
| 		$PerUserPvChart, | 		$PerUserPvChart, | ||||||
| 		$DriveChart, | 		$DriveChart, | ||||||
| 		$PerUserReactionsChart, | 		$PerUserReactionsChart, | ||||||
| 		$HashtagChart, |  | ||||||
| 		$PerUserFollowingChart, | 		$PerUserFollowingChart, | ||||||
| 		$PerUserDriveChart, | 		$PerUserDriveChart, | ||||||
| 		$ApRequestChart, | 		$ApRequestChart, | ||||||
| @@ -559,7 +555,6 @@ const $ApQuestionService: Provider = { provide: 'ApQuestionService', useExisting | |||||||
| 		PerUserPvChart, | 		PerUserPvChart, | ||||||
| 		DriveChart, | 		DriveChart, | ||||||
| 		PerUserReactionsChart, | 		PerUserReactionsChart, | ||||||
| 		HashtagChart, |  | ||||||
| 		PerUserFollowingChart, | 		PerUserFollowingChart, | ||||||
| 		PerUserDriveChart, | 		PerUserDriveChart, | ||||||
| 		ApRequestChart, | 		ApRequestChart, | ||||||
| @@ -680,7 +675,6 @@ const $ApQuestionService: Provider = { provide: 'ApQuestionService', useExisting | |||||||
| 		$PerUserPvChart, | 		$PerUserPvChart, | ||||||
| 		$DriveChart, | 		$DriveChart, | ||||||
| 		$PerUserReactionsChart, | 		$PerUserReactionsChart, | ||||||
| 		$HashtagChart, |  | ||||||
| 		$PerUserFollowingChart, | 		$PerUserFollowingChart, | ||||||
| 		$PerUserDriveChart, | 		$PerUserDriveChart, | ||||||
| 		$ApRequestChart, | 		$ApRequestChart, | ||||||
|   | |||||||
| @@ -26,7 +26,7 @@ export class CreateNotificationService { | |||||||
|  |  | ||||||
| 		private notificationEntityService: NotificationEntityService, | 		private notificationEntityService: NotificationEntityService, | ||||||
| 		private idService: IdService, | 		private idService: IdService, | ||||||
| 		private globalEventServie: GlobalEventService, | 		private globalEventService: GlobalEventService, | ||||||
| 		private pushNotificationService: PushNotificationService, | 		private pushNotificationService: PushNotificationService, | ||||||
| 	) { | 	) { | ||||||
| 	} | 	} | ||||||
| @@ -60,7 +60,7 @@ export class CreateNotificationService { | |||||||
| 		const packed = await this.notificationEntityService.pack(notification, {}); | 		const packed = await this.notificationEntityService.pack(notification, {}); | ||||||
| 	 | 	 | ||||||
| 		// Publish notification event | 		// Publish notification event | ||||||
| 		this.globalEventServie.publishMainStream(notifieeId, 'notification', packed); | 		this.globalEventService.publishMainStream(notifieeId, 'notification', packed); | ||||||
| 	 | 	 | ||||||
| 		// 2秒経っても(今回作成した)通知が既読にならなかったら「未読の通知がありますよ」イベントを発行する | 		// 2秒経っても(今回作成した)通知が既読にならなかったら「未読の通知がありますよ」イベントを発行する | ||||||
| 		setTimeout(async () => { | 		setTimeout(async () => { | ||||||
| @@ -77,7 +77,7 @@ export class CreateNotificationService { | |||||||
| 			} | 			} | ||||||
| 			//#endregion | 			//#endregion | ||||||
| 	 | 	 | ||||||
| 			this.globalEventServie.publishMainStream(notifieeId, 'unreadNotification', packed); | 			this.globalEventService.publishMainStream(notifieeId, 'unreadNotification', packed); | ||||||
| 			this.pushNotificationService.pushNotification(notifieeId, 'notification', packed); | 			this.pushNotificationService.pushNotification(notifieeId, 'notification', packed); | ||||||
| 	 | 	 | ||||||
| 			if (type === 'follow') this.emailNotificationFollow(notifieeId, await this.usersRepository.findOneByOrFail({ id: data.notifierId! })); | 			if (type === 'follow') this.emailNotificationFollow(notifieeId, await this.usersRepository.findOneByOrFail({ id: data.notifierId! })); | ||||||
|   | |||||||
| @@ -120,7 +120,7 @@ export class CustomEmojiService { | |||||||
| 		const url = isLocal | 		const url = isLocal | ||||||
| 			? emojiUrl | 			? emojiUrl | ||||||
| 			: this.config.proxyRemoteFiles | 			: this.config.proxyRemoteFiles | ||||||
| 				? `${this.config.url}/proxy/${encodeURIComponent((new URL(emojiUrl)).pathname)}?${query({ url: emojiUrl })}` | 				? `${this.config.mediaProxy}/emoji.webp?${query({ url: emojiUrl })}` | ||||||
| 				: emojiUrl; | 				: emojiUrl; | ||||||
|  |  | ||||||
| 		return url; | 		return url; | ||||||
|   | |||||||
| @@ -14,7 +14,7 @@ export class DeleteAccountService { | |||||||
|  |  | ||||||
| 		private userSuspendService: UserSuspendService, | 		private userSuspendService: UserSuspendService, | ||||||
| 		private queueService: QueueService, | 		private queueService: QueueService, | ||||||
| 		private globalEventServie: GlobalEventService, | 		private globalEventService: GlobalEventService, | ||||||
| 	) { | 	) { | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| @@ -38,6 +38,6 @@ export class DeleteAccountService { | |||||||
| 		}); | 		}); | ||||||
| 	 | 	 | ||||||
| 		// Terminate streaming | 		// Terminate streaming | ||||||
| 		this.globalEventServie.publishUserEvent(user.id, 'terminate', {}); | 		this.globalEventService.publishUserEvent(user.id, 'terminate', {}); | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|   | |||||||
| @@ -60,6 +60,7 @@ export class DownloadService { | |||||||
| 			retry: { | 			retry: { | ||||||
| 				limit: 0, | 				limit: 0, | ||||||
| 			}, | 			}, | ||||||
|  | 			enableUnixSockets: false, | ||||||
| 		}).on('response', (res: Got.Response) => { | 		}).on('response', (res: Got.Response) => { | ||||||
| 			if ((process.env.NODE_ENV === 'production' || process.env.NODE_ENV === 'test') && !this.config.proxy && res.ip) { | 			if ((process.env.NODE_ENV === 'production' || process.env.NODE_ENV === 'test') && !this.config.proxy && res.ip) { | ||||||
| 				if (this.isPrivateIp(res.ip)) { | 				if (this.isPrivateIp(res.ip)) { | ||||||
|   | |||||||
| @@ -250,6 +250,14 @@ export class DriveService { | |||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public async generateAlts(path: string, type: string, generateWeb: boolean) { | 	public async generateAlts(path: string, type: string, generateWeb: boolean) { | ||||||
| 		if (type.startsWith('video/')) { | 		if (type.startsWith('video/')) { | ||||||
|  | 			if (this.config.videoThumbnailGenerator != null) { | ||||||
|  | 				// videoThumbnailGeneratorが指定されていたら動画サムネイル生成はスキップ | ||||||
|  | 				return { | ||||||
|  | 					webpublic: null, | ||||||
|  | 					thumbnail: null, | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  |  | ||||||
| 			try { | 			try { | ||||||
| 				const thumbnail = await this.videoProcessingService.generateVideoThumbnail(path); | 				const thumbnail = await this.videoProcessingService.generateVideoThumbnail(path); | ||||||
| 				return { | 				return { | ||||||
|   | |||||||
| @@ -4,7 +4,6 @@ import type { User } from '@/models/entities/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 { Hashtag } from '@/models/entities/Hashtag.js'; | import type { Hashtag } from '@/models/entities/Hashtag.js'; | ||||||
| import HashtagChart from '@/core/chart/charts/hashtag.js'; |  | ||||||
| import type { HashtagsRepository, UsersRepository } from '@/models/index.js'; | import type { HashtagsRepository, UsersRepository } from '@/models/index.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'; | ||||||
| @@ -20,7 +19,6 @@ export class HashtagService { | |||||||
|  |  | ||||||
| 		private userEntityService: UserEntityService, | 		private userEntityService: UserEntityService, | ||||||
| 		private idService: IdService, | 		private idService: IdService, | ||||||
| 		private hashtagChart: HashtagChart, |  | ||||||
| 	) { | 	) { | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| @@ -143,9 +141,5 @@ export class HashtagService { | |||||||
| 				} as Hashtag); | 				} as Hashtag); | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		if (!isUserAttached) { |  | ||||||
| 			this.hashtagChart.update(tag, user); |  | ||||||
| 		} |  | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|   | |||||||
| @@ -95,7 +95,7 @@ export class HttpRequestService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public async getJson(url: string, accept = 'application/json, */*', headers?: Record<string, string>): Promise<unknown> { | 	public async getJson<T = unknown>(url: string, accept = 'application/json, */*', headers?: Record<string, string>): Promise<T> { | ||||||
| 		const res = await this.send(url, { | 		const res = await this.send(url, { | ||||||
| 			method: 'GET', | 			method: 'GET', | ||||||
| 			headers: Object.assign({ | 			headers: Object.assign({ | ||||||
| @@ -106,7 +106,7 @@ export class HttpRequestService { | |||||||
| 			size: 1024 * 256, | 			size: 1024 * 256, | ||||||
| 		}); | 		}); | ||||||
|  |  | ||||||
| 		return await res.json(); | 		return await res.json() as T; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
|   | |||||||
| @@ -1,5 +1,4 @@ | |||||||
| import { Inject, Injectable } from '@nestjs/common'; | import { Inject, Injectable } from '@nestjs/common'; | ||||||
| import * as SyslogPro from 'syslog-pro'; |  | ||||||
| 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 Logger from '@/logger.js'; | import Logger from '@/logger.js'; | ||||||
| @@ -8,29 +7,14 @@ import type { KEYWORD } from 'color-convert/conversions'; | |||||||
|  |  | ||||||
| @Injectable() | @Injectable() | ||||||
| export class LoggerService { | export class LoggerService { | ||||||
| 	private syslogClient; |  | ||||||
|  |  | ||||||
| 	constructor( | 	constructor( | ||||||
| 		@Inject(DI.config) | 		@Inject(DI.config) | ||||||
| 		private config: Config, | 		private config: Config, | ||||||
| 	) { | 	) { | ||||||
| 		if (this.config.syslog) { |  | ||||||
| 			this.syslogClient = new SyslogPro.RFC5424({ |  | ||||||
| 				applicationName: 'Misskey', |  | ||||||
| 				timestamp: true, |  | ||||||
| 				includeStructuredData: true, |  | ||||||
| 				color: true, |  | ||||||
| 				extendedColor: true, |  | ||||||
| 				server: { |  | ||||||
| 					target: config.syslog.host, |  | ||||||
| 					port: config.syslog.port, |  | ||||||
| 				}, |  | ||||||
| 			}); |  | ||||||
| 		} |  | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public getLogger(domain: string, color?: KEYWORD | undefined, store?: boolean) { | 	public getLogger(domain: string, color?: KEYWORD | undefined, store?: boolean) { | ||||||
| 		return new Logger(domain, color, store, this.syslogClient); | 		return new Logger(domain, color, store); | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|   | |||||||
| @@ -175,7 +175,7 @@ export class NoteCreateService { | |||||||
| 		private userEntityService: UserEntityService, | 		private userEntityService: UserEntityService, | ||||||
| 		private noteEntityService: NoteEntityService, | 		private noteEntityService: NoteEntityService, | ||||||
| 		private idService: IdService, | 		private idService: IdService, | ||||||
| 		private globalEventServie: GlobalEventService, | 		private globalEventService: GlobalEventService, | ||||||
| 		private queueService: QueueService, | 		private queueService: QueueService, | ||||||
| 		private noteReadService: NoteReadService, | 		private noteReadService: NoteReadService, | ||||||
| 		private createNotificationService: CreateNotificationService, | 		private createNotificationService: CreateNotificationService, | ||||||
| @@ -535,7 +535,7 @@ export class NoteCreateService { | |||||||
| 			// Pack the note | 			// Pack the note | ||||||
| 			const noteObj = await this.noteEntityService.pack(note); | 			const noteObj = await this.noteEntityService.pack(note); | ||||||
|  |  | ||||||
| 			this.globalEventServie.publishNotesStream(noteObj); | 			this.globalEventService.publishNotesStream(noteObj); | ||||||
|  |  | ||||||
| 			this.webhookService.getActiveWebhooks().then(webhooks => { | 			this.webhookService.getActiveWebhooks().then(webhooks => { | ||||||
| 				webhooks = webhooks.filter(x => x.userId === user.id && x.on.includes('note')); | 				webhooks = webhooks.filter(x => x.userId === user.id && x.on.includes('note')); | ||||||
| @@ -561,7 +561,7 @@ export class NoteCreateService { | |||||||
|  |  | ||||||
| 					if (!threadMuted) { | 					if (!threadMuted) { | ||||||
| 						nm.push(data.reply.userId, 'reply'); | 						nm.push(data.reply.userId, 'reply'); | ||||||
| 						this.globalEventServie.publishMainStream(data.reply.userId, 'reply', noteObj); | 						this.globalEventService.publishMainStream(data.reply.userId, 'reply', noteObj); | ||||||
|  |  | ||||||
| 						const webhooks = (await this.webhookService.getActiveWebhooks()).filter(x => x.userId === data.reply!.userId && x.on.includes('reply')); | 						const webhooks = (await this.webhookService.getActiveWebhooks()).filter(x => x.userId === data.reply!.userId && x.on.includes('reply')); | ||||||
| 						for (const webhook of webhooks) { | 						for (const webhook of webhooks) { | ||||||
| @@ -584,7 +584,7 @@ export class NoteCreateService { | |||||||
|  |  | ||||||
| 				// Publish event | 				// Publish event | ||||||
| 				if ((user.id !== data.renote.userId) && data.renote.userHost === null) { | 				if ((user.id !== data.renote.userId) && data.renote.userHost === null) { | ||||||
| 					this.globalEventServie.publishMainStream(data.renote.userId, 'renote', noteObj); | 					this.globalEventService.publishMainStream(data.renote.userId, 'renote', noteObj); | ||||||
|  |  | ||||||
| 					const webhooks = (await this.webhookService.getActiveWebhooks()).filter(x => x.userId === data.renote!.userId && x.on.includes('renote')); | 					const webhooks = (await this.webhookService.getActiveWebhooks()).filter(x => x.userId === data.renote!.userId && x.on.includes('renote')); | ||||||
| 					for (const webhook of webhooks) { | 					for (const webhook of webhooks) { | ||||||
| @@ -684,7 +684,7 @@ export class NoteCreateService { | |||||||
| 				detail: true, | 				detail: true, | ||||||
| 			}); | 			}); | ||||||
|  |  | ||||||
| 			this.globalEventServie.publishMainStream(u.id, 'mention', detailPackedNote); | 			this.globalEventService.publishMainStream(u.id, 'mention', detailPackedNote); | ||||||
|  |  | ||||||
| 			const webhooks = (await this.webhookService.getActiveWebhooks()).filter(x => x.userId === u.id && x.on.includes('mention')); | 			const webhooks = (await this.webhookService.getActiveWebhooks()).filter(x => x.userId === u.id && x.on.includes('mention')); | ||||||
| 			for (const webhook of webhooks) { | 			for (const webhook of webhooks) { | ||||||
|   | |||||||
| @@ -34,7 +34,7 @@ export class NoteDeleteService { | |||||||
|  |  | ||||||
| 		private userEntityService: UserEntityService, | 		private userEntityService: UserEntityService, | ||||||
| 		private noteEntityService: NoteEntityService, | 		private noteEntityService: NoteEntityService, | ||||||
| 		private globalEventServie: GlobalEventService, | 		private globalEventService: GlobalEventService, | ||||||
| 		private relayService: RelayService, | 		private relayService: RelayService, | ||||||
| 		private federatedInstanceService: FederatedInstanceService, | 		private federatedInstanceService: FederatedInstanceService, | ||||||
| 		private apRendererService: ApRendererService, | 		private apRendererService: ApRendererService, | ||||||
| @@ -63,7 +63,7 @@ export class NoteDeleteService { | |||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		if (!quiet) { | 		if (!quiet) { | ||||||
| 			this.globalEventServie.publishNoteStream(note.id, 'deleted', { | 			this.globalEventService.publishNoteStream(note.id, 'deleted', { | ||||||
| 				deletedAt: deletedAt, | 				deletedAt: deletedAt, | ||||||
| 			}); | 			}); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -9,9 +9,9 @@ import { IdService } from '@/core/IdService.js'; | |||||||
| import { GlobalEventService } from '@/core/GlobalEventService.js'; | import { GlobalEventService } from '@/core/GlobalEventService.js'; | ||||||
| import type { UsersRepository, NoteUnreadsRepository, MutingsRepository, NoteThreadMutingsRepository, FollowingsRepository, ChannelFollowingsRepository, AntennaNotesRepository } from '@/models/index.js'; | import type { UsersRepository, NoteUnreadsRepository, MutingsRepository, NoteThreadMutingsRepository, FollowingsRepository, ChannelFollowingsRepository, AntennaNotesRepository } from '@/models/index.js'; | ||||||
| import { UserEntityService } from '@/core/entities/UserEntityService.js'; | import { UserEntityService } from '@/core/entities/UserEntityService.js'; | ||||||
|  | import { bindThis } from '@/decorators.js'; | ||||||
| import { NotificationService } from './NotificationService.js'; | import { NotificationService } from './NotificationService.js'; | ||||||
| import { AntennaService } from './AntennaService.js'; | import { AntennaService } from './AntennaService.js'; | ||||||
| import { bindThis } from '@/decorators.js'; |  | ||||||
| import { PushNotificationService } from './PushNotificationService.js'; | import { PushNotificationService } from './PushNotificationService.js'; | ||||||
|  |  | ||||||
| @Injectable() | @Injectable() | ||||||
| @@ -40,7 +40,7 @@ export class NoteReadService { | |||||||
|  |  | ||||||
| 		private userEntityService: UserEntityService, | 		private userEntityService: UserEntityService, | ||||||
| 		private idService: IdService, | 		private idService: IdService, | ||||||
| 		private globalEventServie: GlobalEventService, | 		private globalEventService: GlobalEventService, | ||||||
| 		private notificationService: NotificationService, | 		private notificationService: NotificationService, | ||||||
| 		private antennaService: AntennaService, | 		private antennaService: AntennaService, | ||||||
| 		private pushNotificationService: PushNotificationService, | 		private pushNotificationService: PushNotificationService, | ||||||
| @@ -87,13 +87,13 @@ export class NoteReadService { | |||||||
| 			if (exist == null) return; | 			if (exist == null) return; | ||||||
| 	 | 	 | ||||||
| 			if (params.isMentioned) { | 			if (params.isMentioned) { | ||||||
| 				this.globalEventServie.publishMainStream(userId, 'unreadMention', note.id); | 				this.globalEventService.publishMainStream(userId, 'unreadMention', note.id); | ||||||
| 			} | 			} | ||||||
| 			if (params.isSpecified) { | 			if (params.isSpecified) { | ||||||
| 				this.globalEventServie.publishMainStream(userId, 'unreadSpecifiedNote', note.id); | 				this.globalEventService.publishMainStream(userId, 'unreadSpecifiedNote', note.id); | ||||||
| 			} | 			} | ||||||
| 			if (note.channelId) { | 			if (note.channelId) { | ||||||
| 				this.globalEventServie.publishMainStream(userId, 'unreadChannel', note.id); | 				this.globalEventService.publishMainStream(userId, 'unreadChannel', note.id); | ||||||
| 			} | 			} | ||||||
| 		}, 2000); | 		}, 2000); | ||||||
| 	}	 | 	}	 | ||||||
| @@ -107,12 +107,6 @@ export class NoteReadService { | |||||||
| 			followingChannels: Set<Channel['id']>; | 			followingChannels: Set<Channel['id']>; | ||||||
| 		}, | 		}, | ||||||
| 	): Promise<void> { | 	): Promise<void> { | ||||||
| 		const following = info?.following ? info.following : new Set<string>((await this.followingsRepository.find({ |  | ||||||
| 			where: { |  | ||||||
| 				followerId: userId, |  | ||||||
| 			}, |  | ||||||
| 			select: ['followeeId'], |  | ||||||
| 		})).map(x => x.followeeId)); |  | ||||||
| 		const followingChannels = info?.followingChannels ? info.followingChannels : new Set<string>((await this.channelFollowingsRepository.find({ | 		const followingChannels = info?.followingChannels ? info.followingChannels : new Set<string>((await this.channelFollowingsRepository.find({ | ||||||
| 			where: { | 			where: { | ||||||
| 				followerId: userId, | 				followerId: userId, | ||||||
| @@ -139,7 +133,7 @@ export class NoteReadService { | |||||||
| 	 | 	 | ||||||
| 			if (note.user != null) { // たぶんnullになることは無いはずだけど一応 | 			if (note.user != null) { // たぶんnullになることは無いはずだけど一応 | ||||||
| 				for (const antenna of myAntennas) { | 				for (const antenna of myAntennas) { | ||||||
| 					if (await this.antennaService.checkHitAntenna(antenna, note, note.user, undefined, Array.from(following))) { | 					if (await this.antennaService.checkHitAntenna(antenna, note, note.user)) { | ||||||
| 						readAntennaNotes.push(note); | 						readAntennaNotes.push(note); | ||||||
| 					} | 					} | ||||||
| 				} | 				} | ||||||
| @@ -161,7 +155,7 @@ export class NoteReadService { | |||||||
| 			}).then(mentionsCount => { | 			}).then(mentionsCount => { | ||||||
| 				if (mentionsCount === 0) { | 				if (mentionsCount === 0) { | ||||||
| 					// 全て既読になったイベントを発行 | 					// 全て既読になったイベントを発行 | ||||||
| 					this.globalEventServie.publishMainStream(userId, 'readAllUnreadMentions'); | 					this.globalEventService.publishMainStream(userId, 'readAllUnreadMentions'); | ||||||
| 				} | 				} | ||||||
| 			}); | 			}); | ||||||
| 	 | 	 | ||||||
| @@ -171,7 +165,7 @@ export class NoteReadService { | |||||||
| 			}).then(specifiedCount => { | 			}).then(specifiedCount => { | ||||||
| 				if (specifiedCount === 0) { | 				if (specifiedCount === 0) { | ||||||
| 					// 全て既読になったイベントを発行 | 					// 全て既読になったイベントを発行 | ||||||
| 					this.globalEventServie.publishMainStream(userId, 'readAllUnreadSpecifiedNotes'); | 					this.globalEventService.publishMainStream(userId, 'readAllUnreadSpecifiedNotes'); | ||||||
| 				} | 				} | ||||||
| 			}); | 			}); | ||||||
| 	 | 	 | ||||||
| @@ -181,7 +175,7 @@ export class NoteReadService { | |||||||
| 			}).then(channelNoteCount => { | 			}).then(channelNoteCount => { | ||||||
| 				if (channelNoteCount === 0) { | 				if (channelNoteCount === 0) { | ||||||
| 					// 全て既読になったイベントを発行 | 					// 全て既読になったイベントを発行 | ||||||
| 					this.globalEventServie.publishMainStream(userId, 'readAllChannels'); | 					this.globalEventService.publishMainStream(userId, 'readAllChannels'); | ||||||
| 				} | 				} | ||||||
| 			}); | 			}); | ||||||
| 	 | 	 | ||||||
| @@ -206,14 +200,14 @@ export class NoteReadService { | |||||||
| 				}); | 				}); | ||||||
| 	 | 	 | ||||||
| 				if (count === 0) { | 				if (count === 0) { | ||||||
| 					this.globalEventServie.publishMainStream(userId, 'readAntenna', antenna); | 					this.globalEventService.publishMainStream(userId, 'readAntenna', antenna); | ||||||
| 					this.pushNotificationService.pushNotification(userId, 'readAntenna', { antennaId: antenna.id }); | 					this.pushNotificationService.pushNotification(userId, 'readAntenna', { antennaId: antenna.id }); | ||||||
| 				} | 				} | ||||||
| 			} | 			} | ||||||
| 	 | 	 | ||||||
| 			this.userEntityService.getHasUnreadAntenna(userId).then(unread => { | 			this.userEntityService.getHasUnreadAntenna(userId).then(unread => { | ||||||
| 				if (!unread) { | 				if (!unread) { | ||||||
| 					this.globalEventServie.publishMainStream(userId, 'readAllAntennas'); | 					this.globalEventService.publishMainStream(userId, 'readAllAntennas'); | ||||||
| 					this.pushNotificationService.pushNotification(userId, 'readAllAntennas', undefined); | 					this.pushNotificationService.pushNotification(userId, 'readAllAntennas', undefined); | ||||||
| 				} | 				} | ||||||
| 			}); | 			}); | ||||||
|   | |||||||
| @@ -1,17 +1,17 @@ | |||||||
| import { Inject, Injectable } from '@nestjs/common'; | import { Inject, Injectable } from '@nestjs/common'; | ||||||
| import { Not } from 'typeorm'; | import { Not } from 'typeorm'; | ||||||
| import { DI } from '@/di-symbols.js'; | import { DI } from '@/di-symbols.js'; | ||||||
| import type { NotesRepository, UsersRepository, BlockingsRepository, PollsRepository, PollVotesRepository } from '@/models/index.js'; | import type { NotesRepository, UsersRepository, PollsRepository, PollVotesRepository } from '@/models/index.js'; | ||||||
| import type { Note } from '@/models/entities/Note.js'; | import type { Note } from '@/models/entities/Note.js'; | ||||||
| import { RelayService } from '@/core/RelayService.js'; | import { RelayService } from '@/core/RelayService.js'; | ||||||
| import type { CacheableUser } from '@/models/entities/User.js'; | import type { CacheableUser } from '@/models/entities/User.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 { CreateNotificationService } from '@/core/CreateNotificationService.js'; |  | ||||||
| import { ApRendererService } from '@/core/activitypub/ApRendererService.js'; | import { ApRendererService } from '@/core/activitypub/ApRendererService.js'; | ||||||
| import { UserEntityService } from '@/core/entities/UserEntityService.js'; | import { UserEntityService } from '@/core/entities/UserEntityService.js'; | ||||||
| import { ApDeliverManagerService } from '@/core/activitypub/ApDeliverManagerService.js'; | import { ApDeliverManagerService } from '@/core/activitypub/ApDeliverManagerService.js'; | ||||||
| import { bindThis } from '@/decorators.js'; | import { bindThis } from '@/decorators.js'; | ||||||
|  | import { UserBlockingService } from '@/core/UserBlockingService.js'; | ||||||
|  |  | ||||||
| @Injectable() | @Injectable() | ||||||
| export class PollService { | export class PollService { | ||||||
| @@ -28,14 +28,11 @@ export class PollService { | |||||||
| 		@Inject(DI.pollVotesRepository) | 		@Inject(DI.pollVotesRepository) | ||||||
| 		private pollVotesRepository: PollVotesRepository, | 		private pollVotesRepository: PollVotesRepository, | ||||||
|  |  | ||||||
| 		@Inject(DI.blockingsRepository) |  | ||||||
| 		private blockingsRepository: BlockingsRepository, |  | ||||||
|  |  | ||||||
| 		private userEntityService: UserEntityService, | 		private userEntityService: UserEntityService, | ||||||
| 		private idService: IdService, | 		private idService: IdService, | ||||||
| 		private relayService: RelayService, | 		private relayService: RelayService, | ||||||
| 		private globalEventServie: GlobalEventService, | 		private globalEventService: GlobalEventService, | ||||||
| 		private createNotificationService: CreateNotificationService, | 		private userBlockingService: UserBlockingService, | ||||||
| 		private apRendererService: ApRendererService, | 		private apRendererService: ApRendererService, | ||||||
| 		private apDeliverManagerService: ApDeliverManagerService, | 		private apDeliverManagerService: ApDeliverManagerService, | ||||||
| 	) { | 	) { | ||||||
| @@ -52,11 +49,8 @@ export class PollService { | |||||||
| 	 | 	 | ||||||
| 		// Check blocking | 		// Check blocking | ||||||
| 		if (note.userId !== user.id) { | 		if (note.userId !== user.id) { | ||||||
| 			const block = await this.blockingsRepository.findOneBy({ | 			const blocked = await this.userBlockingService.checkBlocked(note.userId, user.id); | ||||||
| 				blockerId: note.userId, | 			if (blocked) { | ||||||
| 				blockeeId: user.id, |  | ||||||
| 			}); |  | ||||||
| 			if (block) { |  | ||||||
| 				throw new Error('blocked'); | 				throw new Error('blocked'); | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| @@ -88,7 +82,7 @@ export class PollService { | |||||||
| 		const index = choice + 1; // In SQL, array index is 1 based | 		const index = choice + 1; // In SQL, array index is 1 based | ||||||
| 		await this.pollsRepository.query(`UPDATE poll SET votes[${index}] = votes[${index}] + 1 WHERE "noteId" = '${poll.noteId}'`); | 		await this.pollsRepository.query(`UPDATE poll SET votes[${index}] = votes[${index}] + 1 WHERE "noteId" = '${poll.noteId}'`); | ||||||
| 	 | 	 | ||||||
| 		this.globalEventServie.publishNoteStream(note.id, 'pollVoted', { | 		this.globalEventService.publishNoteStream(note.id, 'pollVoted', { | ||||||
| 			choice: choice, | 			choice: choice, | ||||||
| 			userId: user.id, | 			userId: user.id, | ||||||
| 		}); | 		}); | ||||||
|   | |||||||
| @@ -1,10 +1,10 @@ | |||||||
| import { Inject, Injectable } from '@nestjs/common'; | import { Inject, Injectable } from '@nestjs/common'; | ||||||
| import { Brackets } from 'typeorm'; | import { Brackets, ObjectLiteral } from 'typeorm'; | ||||||
| import { DI } from '@/di-symbols.js'; | import { DI } from '@/di-symbols.js'; | ||||||
| import type { User } from '@/models/entities/User.js'; | import type { User } from '@/models/entities/User.js'; | ||||||
| import type { UserProfilesRepository, FollowingsRepository, ChannelFollowingsRepository, MutedNotesRepository, BlockingsRepository, NoteThreadMutingsRepository, MutingsRepository } from '@/models/index.js'; | import type { UserProfilesRepository, FollowingsRepository, ChannelFollowingsRepository, MutedNotesRepository, BlockingsRepository, NoteThreadMutingsRepository, MutingsRepository } from '@/models/index.js'; | ||||||
| import type { SelectQueryBuilder } from 'typeorm'; |  | ||||||
| import { bindThis } from '@/decorators.js'; | import { bindThis } from '@/decorators.js'; | ||||||
|  | import type { SelectQueryBuilder } from 'typeorm'; | ||||||
|  |  | ||||||
| @Injectable() | @Injectable() | ||||||
| export class QueryService { | export class QueryService { | ||||||
| @@ -32,7 +32,7 @@ export class QueryService { | |||||||
| 	) { | 	) { | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	public makePaginationQuery<T>(q: SelectQueryBuilder<T>, sinceId?: string, untilId?: string, sinceDate?: number, untilDate?: number): SelectQueryBuilder<T> { | 	public makePaginationQuery<T extends ObjectLiteral>(q: SelectQueryBuilder<T>, sinceId?: string, untilId?: string, sinceDate?: number, untilDate?: number): SelectQueryBuilder<T> { | ||||||
| 		if (sinceId && untilId) { | 		if (sinceId && untilId) { | ||||||
| 			q.andWhere(`${q.alias}.id > :sinceId`, { sinceId: sinceId }); | 			q.andWhere(`${q.alias}.id > :sinceId`, { sinceId: sinceId }); | ||||||
| 			q.andWhere(`${q.alias}.id < :untilId`, { untilId: untilId }); | 			q.andWhere(`${q.alias}.id < :untilId`, { untilId: untilId }); | ||||||
|   | |||||||
| @@ -18,7 +18,8 @@ import { UserEntityService } from '@/core/entities/UserEntityService.js'; | |||||||
| import { ApRendererService } from '@/core/activitypub/ApRendererService.js'; | import { ApRendererService } from '@/core/activitypub/ApRendererService.js'; | ||||||
| import { MetaService } from '@/core/MetaService.js'; | import { MetaService } from '@/core/MetaService.js'; | ||||||
| import { bindThis } from '@/decorators.js'; | import { bindThis } from '@/decorators.js'; | ||||||
| import { UtilityService } from './UtilityService.js'; | import { UtilityService } from '@/core/UtilityService.js'; | ||||||
|  | import { UserBlockingService } from '@/core/UserBlockingService.js'; | ||||||
|  |  | ||||||
| const legacies: Record<string, string> = { | const legacies: Record<string, string> = { | ||||||
| 	'like': '👍', | 	'like': '👍', | ||||||
| @@ -73,8 +74,9 @@ export class ReactionService { | |||||||
| 		private metaService: MetaService, | 		private metaService: MetaService, | ||||||
| 		private userEntityService: UserEntityService, | 		private userEntityService: UserEntityService, | ||||||
| 		private noteEntityService: NoteEntityService, | 		private noteEntityService: NoteEntityService, | ||||||
|  | 		private userBlockingService: UserBlockingService, | ||||||
| 		private idService: IdService, | 		private idService: IdService, | ||||||
| 		private globalEventServie: GlobalEventService, | 		private globalEventService: GlobalEventService, | ||||||
| 		private apRendererService: ApRendererService, | 		private apRendererService: ApRendererService, | ||||||
| 		private apDeliverManagerService: ApDeliverManagerService, | 		private apDeliverManagerService: ApDeliverManagerService, | ||||||
| 		private createNotificationService: CreateNotificationService, | 		private createNotificationService: CreateNotificationService, | ||||||
| @@ -86,11 +88,8 @@ export class ReactionService { | |||||||
| 	public async create(user: { id: User['id']; host: User['host']; isBot: User['isBot'] }, note: Note, reaction?: string) { | 	public async create(user: { id: User['id']; host: User['host']; isBot: User['isBot'] }, note: Note, reaction?: string) { | ||||||
| 		// Check blocking | 		// Check blocking | ||||||
| 		if (note.userId !== user.id) { | 		if (note.userId !== user.id) { | ||||||
| 			const block = await this.blockingsRepository.findOneBy({ | 			const blocked = await this.userBlockingService.checkBlocked(note.userId, user.id); | ||||||
| 				blockerId: note.userId, | 			if (blocked) { | ||||||
| 				blockeeId: user.id, |  | ||||||
| 			}); |  | ||||||
| 			if (block) { |  | ||||||
| 				throw new IdentifiableError('e70412a4-7197-4726-8e74-f3e0deb92aa7'); | 				throw new IdentifiableError('e70412a4-7197-4726-8e74-f3e0deb92aa7'); | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| @@ -157,7 +156,7 @@ export class ReactionService { | |||||||
| 			select: ['name', 'host', 'originalUrl', 'publicUrl'], | 			select: ['name', 'host', 'originalUrl', 'publicUrl'], | ||||||
| 		}); | 		}); | ||||||
| 	 | 	 | ||||||
| 		this.globalEventServie.publishNoteStream(note.id, 'reacted', { | 		this.globalEventService.publishNoteStream(note.id, 'reacted', { | ||||||
| 			reaction: decodedReaction.reaction, | 			reaction: decodedReaction.reaction, | ||||||
| 			emoji: emoji != null ? { | 			emoji: emoji != null ? { | ||||||
| 				name: emoji.host ? `${emoji.name}@${emoji.host}` : `${emoji.name}@.`, | 				name: emoji.host ? `${emoji.name}@${emoji.host}` : `${emoji.name}@.`, | ||||||
| @@ -229,7 +228,7 @@ export class ReactionService { | |||||||
| 	 | 	 | ||||||
| 		if (!user.isBot) this.notesRepository.decrement({ id: note.id }, 'score', 1); | 		if (!user.isBot) this.notesRepository.decrement({ id: note.id }, 'score', 1); | ||||||
| 	 | 	 | ||||||
| 		this.globalEventServie.publishNoteStream(note.id, 'unreacted', { | 		this.globalEventService.publishNoteStream(note.id, 'unreacted', { | ||||||
| 			reaction: this.decodeReaction(exist.reaction).reaction, | 			reaction: this.decodeReaction(exist.reaction).reaction, | ||||||
| 			userId: user.id, | 			userId: user.id, | ||||||
| 		}); | 		}); | ||||||
|   | |||||||
| @@ -202,6 +202,25 @@ export class RoleService implements OnApplicationShutdown { | |||||||
| 		return [...assignedRoles, ...matchedCondRoles]; | 		return [...assignedRoles, ...matchedCondRoles]; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	/** | ||||||
|  | 	 * 指定ユーザーのバッジロール一覧取得 | ||||||
|  | 	 */ | ||||||
|  | 	@bindThis | ||||||
|  | 	public async getUserBadgeRoles(userId: User['id']) { | ||||||
|  | 		const assigns = await this.roleAssignmentByUserIdCache.fetch(userId, () => this.roleAssignmentsRepository.findBy({ userId })); | ||||||
|  | 		const assignedRoleIds = assigns.map(x => x.roleId); | ||||||
|  | 		const roles = await this.rolesCache.fetch(null, () => this.rolesRepository.findBy({})); | ||||||
|  | 		const assignedBadgeRoles = roles.filter(r => r.asBadge && assignedRoleIds.includes(r.id)); | ||||||
|  | 		const badgeCondRoles = roles.filter(r => r.asBadge && (r.target === 'conditional')); | ||||||
|  | 		if (badgeCondRoles.length > 0) { | ||||||
|  | 			const user = roles.some(r => r.target === 'conditional') ? await this.userCacheService.findById(userId) : null; | ||||||
|  | 			const matchedBadgeCondRoles = badgeCondRoles.filter(r => this.evalCond(user!, r.condFormula)); | ||||||
|  | 			return [...assignedBadgeRoles, ...matchedBadgeCondRoles]; | ||||||
|  | 		} else { | ||||||
|  | 			return assignedBadgeRoles; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public async getUserPolicies(userId: User['id'] | null): Promise<RolePolicies> { | 	public async getUserPolicies(userId: User['id'] | null): Promise<RolePolicies> { | ||||||
| 		const meta = await this.metaService.fetch(); | 		const meta = await this.metaService.fetch(); | ||||||
|   | |||||||
| @@ -1,5 +1,6 @@ | |||||||
|  |  | ||||||
| import { Inject, Injectable } from '@nestjs/common'; | import { Inject, Injectable, OnApplicationShutdown } from '@nestjs/common'; | ||||||
|  | import Redis from 'ioredis'; | ||||||
| import { IdService } from '@/core/IdService.js'; | import { IdService } from '@/core/IdService.js'; | ||||||
| import type { CacheableUser, User } from '@/models/entities/User.js'; | import type { CacheableUser, User } from '@/models/entities/User.js'; | ||||||
| import type { Blocking } from '@/models/entities/Blocking.js'; | import type { Blocking } from '@/models/entities/Blocking.js'; | ||||||
| @@ -7,7 +8,6 @@ import { QueueService } from '@/core/QueueService.js'; | |||||||
| import { GlobalEventService } from '@/core/GlobalEventService.js'; | import { GlobalEventService } from '@/core/GlobalEventService.js'; | ||||||
| import PerUserFollowingChart from '@/core/chart/charts/per-user-following.js'; | import PerUserFollowingChart from '@/core/chart/charts/per-user-following.js'; | ||||||
| import { DI } from '@/di-symbols.js'; | import { DI } from '@/di-symbols.js'; | ||||||
| import logger from '@/logger.js'; |  | ||||||
| import type { UsersRepository, FollowingsRepository, FollowRequestsRepository, BlockingsRepository, UserListsRepository, UserListJoiningsRepository } from '@/models/index.js'; | import type { UsersRepository, FollowingsRepository, FollowRequestsRepository, BlockingsRepository, UserListsRepository, UserListJoiningsRepository } from '@/models/index.js'; | ||||||
| import Logger from '@/logger.js'; | import Logger from '@/logger.js'; | ||||||
| import { UserEntityService } from '@/core/entities/UserEntityService.js'; | import { UserEntityService } from '@/core/entities/UserEntityService.js'; | ||||||
| @@ -15,12 +15,20 @@ import { ApRendererService } from '@/core/activitypub/ApRendererService.js'; | |||||||
| import { LoggerService } from '@/core/LoggerService.js'; | import { LoggerService } from '@/core/LoggerService.js'; | ||||||
| import { WebhookService } from '@/core/WebhookService.js'; | import { WebhookService } from '@/core/WebhookService.js'; | ||||||
| import { bindThis } from '@/decorators.js'; | import { bindThis } from '@/decorators.js'; | ||||||
|  | import { Cache } from '@/misc/cache.js'; | ||||||
|  | import { StreamMessages } from '@/server/api/stream/types.js'; | ||||||
|  |  | ||||||
| @Injectable() | @Injectable() | ||||||
| export class UserBlockingService { | export class UserBlockingService implements OnApplicationShutdown { | ||||||
| 	private logger: Logger; | 	private logger: Logger; | ||||||
|  |  | ||||||
|  | 	// キーがユーザーIDで、値がそのユーザーがブロックしているユーザーのIDのリストなキャッシュ | ||||||
|  | 	private blockingsByUserIdCache: Cache<User['id'][]>; | ||||||
|  |  | ||||||
| 	constructor( | 	constructor( | ||||||
|  | 		@Inject(DI.redisSubscriber) | ||||||
|  | 		private redisSubscriber: Redis.Redis, | ||||||
|  |  | ||||||
| 		@Inject(DI.usersRepository) | 		@Inject(DI.usersRepository) | ||||||
| 		private usersRepository: UsersRepository, | 		private usersRepository: UsersRepository, | ||||||
|  |  | ||||||
| @@ -42,13 +50,44 @@ export class UserBlockingService { | |||||||
| 		private userEntityService: UserEntityService, | 		private userEntityService: UserEntityService, | ||||||
| 		private idService: IdService, | 		private idService: IdService, | ||||||
| 		private queueService: QueueService, | 		private queueService: QueueService, | ||||||
| 		private globalEventServie: GlobalEventService, | 		private globalEventService: GlobalEventService, | ||||||
| 		private webhookService: WebhookService, | 		private webhookService: WebhookService, | ||||||
| 		private apRendererService: ApRendererService, | 		private apRendererService: ApRendererService, | ||||||
| 		private perUserFollowingChart: PerUserFollowingChart, | 		private perUserFollowingChart: PerUserFollowingChart, | ||||||
| 		private loggerService: LoggerService, | 		private loggerService: LoggerService, | ||||||
| 	) { | 	) { | ||||||
| 		this.logger = this.loggerService.getLogger('user-block'); | 		this.logger = this.loggerService.getLogger('user-block'); | ||||||
|  |  | ||||||
|  | 		this.blockingsByUserIdCache = new Cache<User['id'][]>(Infinity); | ||||||
|  |  | ||||||
|  | 		this.redisSubscriber.on('message', this.onMessage); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	@bindThis | ||||||
|  | 	private async onMessage(_: string, data: string): Promise<void> { | ||||||
|  | 		const obj = JSON.parse(data); | ||||||
|  |  | ||||||
|  | 		if (obj.channel === 'internal') { | ||||||
|  | 			const { type, body } = obj.message as StreamMessages['internal']['payload']; | ||||||
|  | 			switch (type) { | ||||||
|  | 				case 'blockingCreated': { | ||||||
|  | 					const cached = this.blockingsByUserIdCache.get(body.blockerId); | ||||||
|  | 					if (cached) { | ||||||
|  | 						this.blockingsByUserIdCache.set(body.blockerId, [...cached, ...[body.blockeeId]]); | ||||||
|  | 					} | ||||||
|  | 					break; | ||||||
|  | 				} | ||||||
|  | 				case 'blockingDeleted': { | ||||||
|  | 					const cached = this.blockingsByUserIdCache.get(body.blockerId); | ||||||
|  | 					if (cached) { | ||||||
|  | 						this.blockingsByUserIdCache.set(body.blockerId, cached.filter(x => x !== body.blockeeId)); | ||||||
|  | 					} | ||||||
|  | 					break; | ||||||
|  | 				} | ||||||
|  | 				default: | ||||||
|  | 					break; | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| @@ -72,6 +111,11 @@ export class UserBlockingService { | |||||||
|  |  | ||||||
| 		await this.blockingsRepository.insert(blocking); | 		await this.blockingsRepository.insert(blocking); | ||||||
|  |  | ||||||
|  | 		this.globalEventService.publishInternalEvent('blockingCreated', { | ||||||
|  | 			blockerId: blocker.id, | ||||||
|  | 			blockeeId: blockee.id, | ||||||
|  | 		}); | ||||||
|  |  | ||||||
| 		if (this.userEntityService.isLocalUser(blocker) && this.userEntityService.isRemoteUser(blockee)) { | 		if (this.userEntityService.isLocalUser(blocker) && this.userEntityService.isRemoteUser(blockee)) { | ||||||
| 			const content = this.apRendererService.renderActivity(this.apRendererService.renderBlock(blocking)); | 			const content = this.apRendererService.renderActivity(this.apRendererService.renderBlock(blocking)); | ||||||
| 			this.queueService.deliver(blocker, content, blockee.inbox); | 			this.queueService.deliver(blocker, content, blockee.inbox); | ||||||
| @@ -97,15 +141,15 @@ export class UserBlockingService { | |||||||
| 		if (this.userEntityService.isLocalUser(followee)) { | 		if (this.userEntityService.isLocalUser(followee)) { | ||||||
| 			this.userEntityService.pack(followee, followee, { | 			this.userEntityService.pack(followee, followee, { | ||||||
| 				detail: true, | 				detail: true, | ||||||
| 			}).then(packed => this.globalEventServie.publishMainStream(followee.id, 'meUpdated', packed)); | 			}).then(packed => this.globalEventService.publishMainStream(followee.id, 'meUpdated', packed)); | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		if (this.userEntityService.isLocalUser(follower)) { | 		if (this.userEntityService.isLocalUser(follower)) { | ||||||
| 			this.userEntityService.pack(followee, follower, { | 			this.userEntityService.pack(followee, follower, { | ||||||
| 				detail: true, | 				detail: true, | ||||||
| 			}).then(async packed => { | 			}).then(async packed => { | ||||||
| 				this.globalEventServie.publishUserEvent(follower.id, 'unfollow', packed); | 				this.globalEventService.publishUserEvent(follower.id, 'unfollow', packed); | ||||||
| 				this.globalEventServie.publishMainStream(follower.id, 'unfollow', packed); | 				this.globalEventService.publishMainStream(follower.id, 'unfollow', packed); | ||||||
|  |  | ||||||
| 				const webhooks = (await this.webhookService.getActiveWebhooks()).filter(x => x.userId === follower.id && x.on.includes('unfollow')); | 				const webhooks = (await this.webhookService.getActiveWebhooks()).filter(x => x.userId === follower.id && x.on.includes('unfollow')); | ||||||
| 				for (const webhook of webhooks) { | 				for (const webhook of webhooks) { | ||||||
| @@ -152,8 +196,8 @@ export class UserBlockingService { | |||||||
| 			this.userEntityService.pack(followee, follower, { | 			this.userEntityService.pack(followee, follower, { | ||||||
| 				detail: true, | 				detail: true, | ||||||
| 			}).then(async packed => { | 			}).then(async packed => { | ||||||
| 				this.globalEventServie.publishUserEvent(follower.id, 'unfollow', packed); | 				this.globalEventService.publishUserEvent(follower.id, 'unfollow', packed); | ||||||
| 				this.globalEventServie.publishMainStream(follower.id, 'unfollow', packed); | 				this.globalEventService.publishMainStream(follower.id, 'unfollow', packed); | ||||||
|  |  | ||||||
| 				const webhooks = (await this.webhookService.getActiveWebhooks()).filter(x => x.userId === follower.id && x.on.includes('unfollow')); | 				const webhooks = (await this.webhookService.getActiveWebhooks()).filter(x => x.userId === follower.id && x.on.includes('unfollow')); | ||||||
| 				for (const webhook of webhooks) { | 				for (const webhook of webhooks) { | ||||||
| @@ -210,10 +254,31 @@ export class UserBlockingService { | |||||||
|  |  | ||||||
| 		await this.blockingsRepository.delete(blocking.id); | 		await this.blockingsRepository.delete(blocking.id); | ||||||
|  |  | ||||||
|  | 		this.globalEventService.publishInternalEvent('blockingDeleted', { | ||||||
|  | 			blockerId: blocker.id, | ||||||
|  | 			blockeeId: blockee.id, | ||||||
|  | 		}); | ||||||
|  |  | ||||||
| 		// deliver if remote bloking | 		// deliver if remote bloking | ||||||
| 		if (this.userEntityService.isLocalUser(blocker) && this.userEntityService.isRemoteUser(blockee)) { | 		if (this.userEntityService.isLocalUser(blocker) && this.userEntityService.isRemoteUser(blockee)) { | ||||||
| 			const content = this.apRendererService.renderActivity(this.apRendererService.renderUndo(this.apRendererService.renderBlock(blocking), blocker)); | 			const content = this.apRendererService.renderActivity(this.apRendererService.renderUndo(this.apRendererService.renderBlock(blocking), blocker)); | ||||||
| 			this.queueService.deliver(blocker, content, blockee.inbox); | 			this.queueService.deliver(blocker, content, blockee.inbox); | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	@bindThis | ||||||
|  | 	public async checkBlocked(blockerId: User['id'], blockeeId: User['id']): Promise<boolean> { | ||||||
|  | 		const blockedUserIds = await this.blockingsByUserIdCache.fetch(blockerId, () => this.blockingsRepository.find({ | ||||||
|  | 			where: { | ||||||
|  | 				blockerId, | ||||||
|  | 			}, | ||||||
|  | 			select: ['blockeeId'], | ||||||
|  | 		}).then(records => records.map(record => record.blockeeId))); | ||||||
|  | 		return blockedUserIds.includes(blockeeId); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	@bindThis | ||||||
|  | 	public onApplicationShutdown(signal?: string | undefined) { | ||||||
|  | 		this.redisSubscriber.off('message', this.onMessage); | ||||||
|  | 	} | ||||||
| } | } | ||||||
|   | |||||||
| @@ -12,10 +12,11 @@ import { FederatedInstanceService } from '@/core/FederatedInstanceService.js'; | |||||||
| import { WebhookService } from '@/core/WebhookService.js'; | import { WebhookService } from '@/core/WebhookService.js'; | ||||||
| import { CreateNotificationService } from '@/core/CreateNotificationService.js'; | import { CreateNotificationService } from '@/core/CreateNotificationService.js'; | ||||||
| import { DI } from '@/di-symbols.js'; | import { DI } from '@/di-symbols.js'; | ||||||
| import type { BlockingsRepository, FollowingsRepository, FollowRequestsRepository, InstancesRepository, UserProfilesRepository, UsersRepository } from '@/models/index.js'; | import type { FollowingsRepository, FollowRequestsRepository, InstancesRepository, UserProfilesRepository, UsersRepository } from '@/models/index.js'; | ||||||
| import { UserEntityService } from '@/core/entities/UserEntityService.js'; | import { UserEntityService } from '@/core/entities/UserEntityService.js'; | ||||||
| import { ApRendererService } from '@/core/activitypub/ApRendererService.js'; | import { ApRendererService } from '@/core/activitypub/ApRendererService.js'; | ||||||
| import { bindThis } from '@/decorators.js'; | import { bindThis } from '@/decorators.js'; | ||||||
|  | import { UserBlockingService } from '@/core/UserBlockingService.js'; | ||||||
| import Logger from '../logger.js'; | import Logger from '../logger.js'; | ||||||
|  |  | ||||||
| const logger = new Logger('following/create'); | const logger = new Logger('following/create'); | ||||||
| @@ -48,21 +49,18 @@ export class UserFollowingService { | |||||||
| 		@Inject(DI.followRequestsRepository) | 		@Inject(DI.followRequestsRepository) | ||||||
| 		private followRequestsRepository: FollowRequestsRepository, | 		private followRequestsRepository: FollowRequestsRepository, | ||||||
|  |  | ||||||
| 		@Inject(DI.blockingsRepository) |  | ||||||
| 		private blockingsRepository: BlockingsRepository, |  | ||||||
|  |  | ||||||
| 		@Inject(DI.instancesRepository) | 		@Inject(DI.instancesRepository) | ||||||
| 		private instancesRepository: InstancesRepository, | 		private instancesRepository: InstancesRepository, | ||||||
|  |  | ||||||
| 		private userEntityService: UserEntityService, | 		private userEntityService: UserEntityService, | ||||||
|  | 		private userBlockingService: UserBlockingService, | ||||||
| 		private idService: IdService, | 		private idService: IdService, | ||||||
| 		private queueService: QueueService, | 		private queueService: QueueService, | ||||||
| 		private globalEventServie: GlobalEventService, | 		private globalEventService: GlobalEventService, | ||||||
| 		private createNotificationService: CreateNotificationService, | 		private createNotificationService: CreateNotificationService, | ||||||
| 		private federatedInstanceService: FederatedInstanceService, | 		private federatedInstanceService: FederatedInstanceService, | ||||||
| 		private webhookService: WebhookService, | 		private webhookService: WebhookService, | ||||||
| 		private apRendererService: ApRendererService, | 		private apRendererService: ApRendererService, | ||||||
| 		private globalEventService: GlobalEventService, |  | ||||||
| 		private perUserFollowingChart: PerUserFollowingChart, | 		private perUserFollowingChart: PerUserFollowingChart, | ||||||
| 		private instanceChart: InstanceChart, | 		private instanceChart: InstanceChart, | ||||||
| 	) { | 	) { | ||||||
| @@ -77,28 +75,22 @@ export class UserFollowingService { | |||||||
|  |  | ||||||
| 		// check blocking | 		// check blocking | ||||||
| 		const [blocking, blocked] = await Promise.all([ | 		const [blocking, blocked] = await Promise.all([ | ||||||
| 			this.blockingsRepository.findOneBy({ | 			this.userBlockingService.checkBlocked(follower.id, followee.id), | ||||||
| 				blockerId: follower.id, | 			this.userBlockingService.checkBlocked(followee.id, follower.id), | ||||||
| 				blockeeId: followee.id, |  | ||||||
| 			}), |  | ||||||
| 			this.blockingsRepository.findOneBy({ |  | ||||||
| 				blockerId: followee.id, |  | ||||||
| 				blockeeId: follower.id, |  | ||||||
| 			}), |  | ||||||
| 		]); | 		]); | ||||||
|  |  | ||||||
| 		if (this.userEntityService.isRemoteUser(follower) && this.userEntityService.isLocalUser(followee) && blocked) { | 		if (this.userEntityService.isRemoteUser(follower) && this.userEntityService.isLocalUser(followee) && blocked) { | ||||||
| 		// リモートフォローを受けてブロックしていた場合は、エラーにするのではなくRejectを送り返しておしまい。 | 			// リモートフォローを受けてブロックしていた場合は、エラーにするのではなくRejectを送り返しておしまい。 | ||||||
| 			const content = this.apRendererService.renderActivity(this.apRendererService.renderReject(this.apRendererService.renderFollow(follower, followee, requestId), followee)); | 			const content = this.apRendererService.renderActivity(this.apRendererService.renderReject(this.apRendererService.renderFollow(follower, followee, requestId), followee)); | ||||||
| 			this.queueService.deliver(followee, content, follower.inbox); | 			this.queueService.deliver(followee, content, follower.inbox); | ||||||
| 			return; | 			return; | ||||||
| 		} else if (this.userEntityService.isRemoteUser(follower) && this.userEntityService.isLocalUser(followee) && blocking) { | 		} else if (this.userEntityService.isRemoteUser(follower) && this.userEntityService.isLocalUser(followee) && blocking) { | ||||||
| 		// リモートフォローを受けてブロックされているはずの場合だったら、ブロック解除しておく。 | 			// リモートフォローを受けてブロックされているはずの場合だったら、ブロック解除しておく。 | ||||||
| 			await this.blockingsRepository.delete(blocking.id); | 			await this.userBlockingService.unblock(follower, followee); | ||||||
| 		} else { | 		} else { | ||||||
| 		// それ以外は単純に例外 | 			// それ以外は単純に例外 | ||||||
| 			if (blocking != null) throw new IdentifiableError('710e8fb0-b8c3-4922-be49-d5d93d8e6a6e', 'blocking'); | 			if (blocking) throw new IdentifiableError('710e8fb0-b8c3-4922-be49-d5d93d8e6a6e', 'blocking'); | ||||||
| 			if (blocked != null) throw new IdentifiableError('3338392a-f764-498d-8855-db939dcf8c48', 'blocked'); | 			if (blocked) throw new IdentifiableError('3338392a-f764-498d-8855-db939dcf8c48', 'blocked'); | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		const followeeProfile = await this.userProfilesRepository.findOneByOrFail({ userId: followee.id }); | 		const followeeProfile = await this.userProfilesRepository.findOneByOrFail({ userId: followee.id }); | ||||||
| @@ -227,8 +219,8 @@ export class UserFollowingService { | |||||||
| 			this.userEntityService.pack(followee.id, follower, { | 			this.userEntityService.pack(followee.id, follower, { | ||||||
| 				detail: true, | 				detail: true, | ||||||
| 			}).then(async packed => { | 			}).then(async packed => { | ||||||
| 				this.globalEventServie.publishUserEvent(follower.id, 'follow', packed as Packed<'UserDetailedNotMe'>); | 				this.globalEventService.publishUserEvent(follower.id, 'follow', packed as Packed<'UserDetailedNotMe'>); | ||||||
| 				this.globalEventServie.publishMainStream(follower.id, 'follow', packed as Packed<'UserDetailedNotMe'>); | 				this.globalEventService.publishMainStream(follower.id, 'follow', packed as Packed<'UserDetailedNotMe'>); | ||||||
| 	 | 	 | ||||||
| 				const webhooks = (await this.webhookService.getActiveWebhooks()).filter(x => x.userId === follower.id && x.on.includes('follow')); | 				const webhooks = (await this.webhookService.getActiveWebhooks()).filter(x => x.userId === follower.id && x.on.includes('follow')); | ||||||
| 				for (const webhook of webhooks) { | 				for (const webhook of webhooks) { | ||||||
| @@ -242,7 +234,7 @@ export class UserFollowingService { | |||||||
| 		// Publish followed event | 		// Publish followed event | ||||||
| 		if (this.userEntityService.isLocalUser(followee)) { | 		if (this.userEntityService.isLocalUser(followee)) { | ||||||
| 			this.userEntityService.pack(follower.id, followee).then(async packed => { | 			this.userEntityService.pack(follower.id, followee).then(async packed => { | ||||||
| 				this.globalEventServie.publishMainStream(followee.id, 'followed', packed); | 				this.globalEventService.publishMainStream(followee.id, 'followed', packed); | ||||||
| 	 | 	 | ||||||
| 				const webhooks = (await this.webhookService.getActiveWebhooks()).filter(x => x.userId === followee.id && x.on.includes('followed')); | 				const webhooks = (await this.webhookService.getActiveWebhooks()).filter(x => x.userId === followee.id && x.on.includes('followed')); | ||||||
| 				for (const webhook of webhooks) { | 				for (const webhook of webhooks) { | ||||||
| @@ -288,8 +280,8 @@ export class UserFollowingService { | |||||||
| 			this.userEntityService.pack(followee.id, follower, { | 			this.userEntityService.pack(followee.id, follower, { | ||||||
| 				detail: true, | 				detail: true, | ||||||
| 			}).then(async packed => { | 			}).then(async packed => { | ||||||
| 				this.globalEventServie.publishUserEvent(follower.id, 'unfollow', packed); | 				this.globalEventService.publishUserEvent(follower.id, 'unfollow', packed); | ||||||
| 				this.globalEventServie.publishMainStream(follower.id, 'unfollow', packed); | 				this.globalEventService.publishMainStream(follower.id, 'unfollow', packed); | ||||||
| 	 | 	 | ||||||
| 				const webhooks = (await this.webhookService.getActiveWebhooks()).filter(x => x.userId === follower.id && x.on.includes('unfollow')); | 				const webhooks = (await this.webhookService.getActiveWebhooks()).filter(x => x.userId === follower.id && x.on.includes('unfollow')); | ||||||
| 				for (const webhook of webhooks) { | 				for (const webhook of webhooks) { | ||||||
| @@ -357,18 +349,12 @@ export class UserFollowingService { | |||||||
| 	 | 	 | ||||||
| 		// check blocking | 		// check blocking | ||||||
| 		const [blocking, blocked] = await Promise.all([ | 		const [blocking, blocked] = await Promise.all([ | ||||||
| 			this.blockingsRepository.findOneBy({ | 			this.userBlockingService.checkBlocked(follower.id, followee.id), | ||||||
| 				blockerId: follower.id, | 			this.userBlockingService.checkBlocked(followee.id, follower.id), | ||||||
| 				blockeeId: followee.id, |  | ||||||
| 			}), |  | ||||||
| 			this.blockingsRepository.findOneBy({ |  | ||||||
| 				blockerId: followee.id, |  | ||||||
| 				blockeeId: follower.id, |  | ||||||
| 			}), |  | ||||||
| 		]); | 		]); | ||||||
| 	 | 	 | ||||||
| 		if (blocking != null) throw new Error('blocking'); | 		if (blocking) throw new Error('blocking'); | ||||||
| 		if (blocked != null) throw new Error('blocked'); | 		if (blocked) throw new Error('blocked'); | ||||||
| 	 | 	 | ||||||
| 		const followRequest = await this.followRequestsRepository.insert({ | 		const followRequest = await this.followRequestsRepository.insert({ | ||||||
| 			id: this.idService.genId(), | 			id: this.idService.genId(), | ||||||
| @@ -388,11 +374,11 @@ export class UserFollowingService { | |||||||
| 	 | 	 | ||||||
| 		// Publish receiveRequest event | 		// Publish receiveRequest event | ||||||
| 		if (this.userEntityService.isLocalUser(followee)) { | 		if (this.userEntityService.isLocalUser(followee)) { | ||||||
| 			this.userEntityService.pack(follower.id, followee).then(packed => this.globalEventServie.publishMainStream(followee.id, 'receiveFollowRequest', packed)); | 			this.userEntityService.pack(follower.id, followee).then(packed => this.globalEventService.publishMainStream(followee.id, 'receiveFollowRequest', packed)); | ||||||
| 	 | 	 | ||||||
| 			this.userEntityService.pack(followee.id, followee, { | 			this.userEntityService.pack(followee.id, followee, { | ||||||
| 				detail: true, | 				detail: true, | ||||||
| 			}).then(packed => this.globalEventServie.publishMainStream(followee.id, 'meUpdated', packed)); | 			}).then(packed => this.globalEventService.publishMainStream(followee.id, 'meUpdated', packed)); | ||||||
| 	 | 	 | ||||||
| 			// 通知を作成 | 			// 通知を作成 | ||||||
| 			this.createNotificationService.createNotification(followee.id, 'receiveFollowRequest', { | 			this.createNotificationService.createNotification(followee.id, 'receiveFollowRequest', { | ||||||
| @@ -440,7 +426,7 @@ export class UserFollowingService { | |||||||
| 	 | 	 | ||||||
| 		this.userEntityService.pack(followee.id, followee, { | 		this.userEntityService.pack(followee.id, followee, { | ||||||
| 			detail: true, | 			detail: true, | ||||||
| 		}).then(packed => this.globalEventServie.publishMainStream(followee.id, 'meUpdated', packed)); | 		}).then(packed => this.globalEventService.publishMainStream(followee.id, 'meUpdated', packed)); | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| @@ -468,7 +454,7 @@ export class UserFollowingService { | |||||||
| 	 | 	 | ||||||
| 		this.userEntityService.pack(followee.id, followee, { | 		this.userEntityService.pack(followee.id, followee, { | ||||||
| 			detail: true, | 			detail: true, | ||||||
| 		}).then(packed => this.globalEventServie.publishMainStream(followee.id, 'meUpdated', packed)); | 		}).then(packed => this.globalEventService.publishMainStream(followee.id, 'meUpdated', packed)); | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| @@ -583,8 +569,8 @@ export class UserFollowingService { | |||||||
| 			detail: true, | 			detail: true, | ||||||
| 		}); | 		}); | ||||||
|  |  | ||||||
| 		this.globalEventServie.publishUserEvent(follower.id, 'unfollow', packedFollowee); | 		this.globalEventService.publishUserEvent(follower.id, 'unfollow', packedFollowee); | ||||||
| 		this.globalEventServie.publishMainStream(follower.id, 'unfollow', packedFollowee); | 		this.globalEventService.publishMainStream(follower.id, 'unfollow', packedFollowee); | ||||||
|  |  | ||||||
| 		const webhooks = (await this.webhookService.getActiveWebhooks()).filter(x => x.userId === follower.id && x.on.includes('unfollow')); | 		const webhooks = (await this.webhookService.getActiveWebhooks()).filter(x => x.userId === follower.id && x.on.includes('unfollow')); | ||||||
| 		for (const webhook of webhooks) { | 		for (const webhook of webhooks) { | ||||||
|   | |||||||
| @@ -14,6 +14,8 @@ import { RoleService } from '@/core/RoleService.js'; | |||||||
|  |  | ||||||
| @Injectable() | @Injectable() | ||||||
| export class UserListService { | export class UserListService { | ||||||
|  | 	public static TooManyUsersError = class extends Error {}; | ||||||
|  |  | ||||||
| 	constructor( | 	constructor( | ||||||
| 		@Inject(DI.usersRepository) | 		@Inject(DI.usersRepository) | ||||||
| 		private usersRepository: UsersRepository, | 		private usersRepository: UsersRepository, | ||||||
| @@ -25,7 +27,7 @@ export class UserListService { | |||||||
| 		private idService: IdService, | 		private idService: IdService, | ||||||
| 		private userFollowingService: UserFollowingService, | 		private userFollowingService: UserFollowingService, | ||||||
| 		private roleService: RoleService, | 		private roleService: RoleService, | ||||||
| 		private globalEventServie: GlobalEventService, | 		private globalEventService: GlobalEventService, | ||||||
| 		private proxyAccountService: ProxyAccountService, | 		private proxyAccountService: ProxyAccountService, | ||||||
| 	) { | 	) { | ||||||
| 	} | 	} | ||||||
| @@ -36,7 +38,7 @@ export class UserListService { | |||||||
| 			userListId: list.id, | 			userListId: list.id, | ||||||
| 		}); | 		}); | ||||||
| 		if (currentCount > (await this.roleService.getUserPolicies(me.id)).userEachUserListsLimit) { | 		if (currentCount > (await this.roleService.getUserPolicies(me.id)).userEachUserListsLimit) { | ||||||
| 			throw new Error('Too many users'); | 			throw new UserListService.TooManyUsersError(); | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		await this.userListJoiningsRepository.insert({ | 		await this.userListJoiningsRepository.insert({ | ||||||
| @@ -46,7 +48,7 @@ export class UserListService { | |||||||
| 			userListId: list.id, | 			userListId: list.id, | ||||||
| 		} as UserListJoining); | 		} as UserListJoining); | ||||||
| 	 | 	 | ||||||
| 		this.globalEventServie.publishUserListStream(list.id, 'userAdded', await this.userEntityService.pack(target)); | 		this.globalEventService.publishUserListStream(list.id, 'userAdded', await this.userEntityService.pack(target)); | ||||||
| 	 | 	 | ||||||
| 		// このインスタンス内にこのリモートユーザーをフォローしているユーザーがいなくても投稿を受け取るためにダミーのユーザーがフォローしたということにする | 		// このインスタンス内にこのリモートユーザーをフォローしているユーザーがいなくても投稿を受け取るためにダミーのユーザーがフォローしたということにする | ||||||
| 		if (this.userEntityService.isRemoteUser(target)) { | 		if (this.userEntityService.isRemoteUser(target)) { | ||||||
|   | |||||||
| @@ -18,7 +18,7 @@ export class UserMutingService { | |||||||
|  |  | ||||||
| 		private idService: IdService, | 		private idService: IdService, | ||||||
| 		private queueService: QueueService, | 		private queueService: QueueService, | ||||||
| 		private globalEventServie: GlobalEventService, | 		private globalEventService: GlobalEventService, | ||||||
| 	) { | 	) { | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|   | |||||||
| @@ -6,6 +6,7 @@ import { ImageProcessingService } from '@/core/ImageProcessingService.js'; | |||||||
| import type { IImage } from '@/core/ImageProcessingService.js'; | import type { IImage } from '@/core/ImageProcessingService.js'; | ||||||
| import { createTempDir } from '@/misc/create-temp.js'; | import { createTempDir } from '@/misc/create-temp.js'; | ||||||
| import { bindThis } from '@/decorators.js'; | import { bindThis } from '@/decorators.js'; | ||||||
|  | import { appendQuery, query } from '@/misc/prelude/url.js'; | ||||||
|  |  | ||||||
| @Injectable() | @Injectable() | ||||||
| export class VideoProcessingService { | export class VideoProcessingService { | ||||||
| @@ -41,5 +42,18 @@ export class VideoProcessingService { | |||||||
| 			cleanup(); | 			cleanup(); | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	@bindThis | ||||||
|  | 	public getExternalVideoThumbnailUrl(url: string): string | null { | ||||||
|  | 		if (this.config.videoThumbnailGenerator == null) return null; | ||||||
|  |  | ||||||
|  | 		return appendQuery( | ||||||
|  | 			`${this.config.videoThumbnailGenerator}/thumbnail.webp`, | ||||||
|  | 			query({ | ||||||
|  | 				thumbnail: '1', | ||||||
|  | 				url, | ||||||
|  | 			}) | ||||||
|  | 		) | ||||||
|  | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -44,16 +44,25 @@ export class WebhookService implements OnApplicationShutdown { | |||||||
| 			switch (type) { | 			switch (type) { | ||||||
| 				case 'webhookCreated': | 				case 'webhookCreated': | ||||||
| 					if (body.active) { | 					if (body.active) { | ||||||
| 						this.webhooks.push(body); | 						this.webhooks.push({ | ||||||
|  | 							...body, | ||||||
|  | 							createdAt: new Date(body.createdAt), | ||||||
|  | 						}); | ||||||
| 					} | 					} | ||||||
| 					break; | 					break; | ||||||
| 				case 'webhookUpdated': | 				case 'webhookUpdated': | ||||||
| 					if (body.active) { | 					if (body.active) { | ||||||
| 						const i = this.webhooks.findIndex(a => a.id === body.id); | 						const i = this.webhooks.findIndex(a => a.id === body.id); | ||||||
| 						if (i > -1) { | 						if (i > -1) { | ||||||
| 							this.webhooks[i] = body; | 							this.webhooks[i] = { | ||||||
|  | 								...body, | ||||||
|  | 								createdAt: new Date(body.createdAt), | ||||||
|  | 							}; | ||||||
| 						} else { | 						} else { | ||||||
| 							this.webhooks.push(body); | 							this.webhooks.push({ | ||||||
|  | 								...body, | ||||||
|  | 								createdAt: new Date(body.createdAt), | ||||||
|  | 							}); | ||||||
| 						} | 						} | ||||||
| 					} else { | 					} else { | ||||||
| 						this.webhooks = this.webhooks.filter(a => a.id !== body.id); | 						this.webhooks = this.webhooks.filter(a => a.id !== body.id); | ||||||
|   | |||||||
| @@ -274,7 +274,7 @@ export class ApRendererService { | |||||||
| 		} as any; | 		} as any; | ||||||
|  |  | ||||||
| 		if (reaction.startsWith(':')) { | 		if (reaction.startsWith(':')) { | ||||||
| 			const name = reaction.replace(/:/g, ''); | 			const name = reaction.replaceAll(':', ''); | ||||||
| 			const emoji = await this.emojisRepository.findOneBy({ | 			const emoji = await this.emojisRepository.findOneBy({ | ||||||
| 				name, | 				name, | ||||||
| 				host: IsNull(), | 				host: IsNull(), | ||||||
|   | |||||||
| @@ -48,6 +48,10 @@ export class ApImageService { | |||||||
| 			throw new Error('invalid image: url not privided'); | 			throw new Error('invalid image: url not privided'); | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
|  | 		if (!image.url.startsWith('https://')) { | ||||||
|  | 			throw new Error('invalid image: unexpected shcema of url: ' + image.url); | ||||||
|  | 		} | ||||||
|  |  | ||||||
| 		this.logger.info(`Creating the Image: ${image.url}`); | 		this.logger.info(`Creating the Image: ${image.url}`); | ||||||
|  |  | ||||||
| 		const instance = await this.metaService.fetch(); | 		const instance = await this.metaService.fetch(); | ||||||
|   | |||||||
| @@ -1,8 +1,7 @@ | |||||||
| import { forwardRef, Inject, Injectable } from '@nestjs/common'; | import { forwardRef, Inject, Injectable } from '@nestjs/common'; | ||||||
| import promiseLimit from 'promise-limit'; | import promiseLimit from 'promise-limit'; | ||||||
| import { DI } from '@/di-symbols.js'; | import { DI } from '@/di-symbols.js'; | ||||||
| import type { MessagingMessagesRepository, PollsRepository, EmojisRepository } from '@/models/index.js'; | import type { MessagingMessagesRepository, PollsRepository, EmojisRepository, UsersRepository } from '@/models/index.js'; | ||||||
| import type { UsersRepository } from '@/models/index.js'; |  | ||||||
| import type { Config } from '@/config.js'; | import type { Config } from '@/config.js'; | ||||||
| import type { CacheableRemoteUser } from '@/models/entities/User.js'; | import type { CacheableRemoteUser } from '@/models/entities/User.js'; | ||||||
| import type { Note } from '@/models/entities/Note.js'; | import type { Note } from '@/models/entities/Note.js'; | ||||||
| @@ -18,6 +17,7 @@ import { PollService } from '@/core/PollService.js'; | |||||||
| import { StatusError } from '@/misc/status-error.js'; | import { StatusError } from '@/misc/status-error.js'; | ||||||
| import { UtilityService } from '@/core/UtilityService.js'; | import { UtilityService } from '@/core/UtilityService.js'; | ||||||
| import { MessagingService } from '@/core/MessagingService.js'; | import { MessagingService } from '@/core/MessagingService.js'; | ||||||
|  | import { bindThis } from '@/decorators.js'; | ||||||
| import { getOneApId, getApId, getOneApHrefNullable, validPost, isEmoji, getApType } from '../type.js'; | import { getOneApId, getApId, getOneApHrefNullable, validPost, isEmoji, getApType } from '../type.js'; | ||||||
| // eslint-disable-next-line @typescript-eslint/consistent-type-imports | // eslint-disable-next-line @typescript-eslint/consistent-type-imports | ||||||
| import { ApLoggerService } from '../ApLoggerService.js'; | import { ApLoggerService } from '../ApLoggerService.js'; | ||||||
| @@ -32,7 +32,6 @@ import { ApQuestionService } from './ApQuestionService.js'; | |||||||
| import { ApImageService } from './ApImageService.js'; | import { ApImageService } from './ApImageService.js'; | ||||||
| import type { Resolver } from '../ApResolverService.js'; | import type { Resolver } from '../ApResolverService.js'; | ||||||
| import type { IObject, IPost } from '../type.js'; | import type { IObject, IPost } from '../type.js'; | ||||||
| import { bindThis } from '@/decorators.js'; |  | ||||||
|  |  | ||||||
| @Injectable() | @Injectable() | ||||||
| export class ApNoteService { | export class ApNoteService { | ||||||
| @@ -134,6 +133,16 @@ export class ApNoteService { | |||||||
| 	 | 	 | ||||||
| 		this.logger.debug(`Note fetched: ${JSON.stringify(note, null, 2)}`); | 		this.logger.debug(`Note fetched: ${JSON.stringify(note, null, 2)}`); | ||||||
|  |  | ||||||
|  | 		if (note.id && !note.id.startsWith('https://')) { | ||||||
|  | 			throw new Error('unexpected shcema of note.id: ' + note.id); | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		const url = getOneApHrefNullable(note.url); | ||||||
|  |  | ||||||
|  | 		if (url && !url.startsWith('https://')) { | ||||||
|  | 			throw new Error('unexpected shcema of note url: ' + url); | ||||||
|  | 		} | ||||||
|  | 	 | ||||||
| 		this.logger.info(`Creating the Note: ${note.id}`); | 		this.logger.info(`Creating the Note: ${note.id}`); | ||||||
| 	 | 	 | ||||||
| 		// 投稿者をフェッチ | 		// 投稿者をフェッチ | ||||||
| @@ -307,7 +316,7 @@ export class ApNoteService { | |||||||
| 			apEmojis, | 			apEmojis, | ||||||
| 			poll, | 			poll, | ||||||
| 			uri: note.id, | 			uri: note.id, | ||||||
| 			url: getOneApHrefNullable(note.url), | 			url: url, | ||||||
| 		}, silent); | 		}, silent); | ||||||
| 	} | 	} | ||||||
| 	 | 	 | ||||||
|   | |||||||
| @@ -29,6 +29,7 @@ import { UserNotePining } from '@/models/entities/UserNotePining.js'; | |||||||
| import { StatusError } from '@/misc/status-error.js'; | import { StatusError } from '@/misc/status-error.js'; | ||||||
| import type { UtilityService } from '@/core/UtilityService.js'; | import type { UtilityService } from '@/core/UtilityService.js'; | ||||||
| import type { UserEntityService } from '@/core/entities/UserEntityService.js'; | import type { UserEntityService } from '@/core/entities/UserEntityService.js'; | ||||||
|  | import { bindThis } from '@/decorators.js'; | ||||||
| import { getApId, getApType, getOneApHrefNullable, isActor, isCollection, isCollectionOrOrderedCollection, isPropertyValue } from '../type.js'; | import { getApId, getApType, getOneApHrefNullable, isActor, isCollection, isCollectionOrOrderedCollection, isPropertyValue } from '../type.js'; | ||||||
| import { extractApHashtags } from './tag.js'; | import { extractApHashtags } from './tag.js'; | ||||||
| import type { OnModuleInit } from '@nestjs/common'; | import type { OnModuleInit } from '@nestjs/common'; | ||||||
| @@ -43,37 +44,6 @@ import type { IActor, IObject, IApPropertyValue } from '../type.js'; | |||||||
| const nameLength = 128; | const nameLength = 128; | ||||||
| const summaryLength = 2048; | const summaryLength = 2048; | ||||||
|  |  | ||||||
| const services: { |  | ||||||
| 	[x: string]: (id: string, username: string) => any |  | ||||||
| } = { |  | ||||||
| 	'misskey:authentication:twitter': (userId, screenName) => ({ userId, screenName }), |  | ||||||
| 	'misskey:authentication:github': (id, login) => ({ id, login }), |  | ||||||
| 	'misskey:authentication:discord': (id, name) => $discord(id, name), |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| const $discord = (id: string, name: string) => { |  | ||||||
| 	if (typeof name !== 'string') { |  | ||||||
| 		name = 'unknown#0000'; |  | ||||||
| 	} |  | ||||||
| 	const [username, discriminator] = name.split('#'); |  | ||||||
| 	return { id, username, discriminator }; |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| function addService(target: { [x: string]: any }, source: IApPropertyValue) { |  | ||||||
| 	const service = services[source.name]; |  | ||||||
|  |  | ||||||
| 	if (typeof source.value !== 'string') { |  | ||||||
| 		source.value = 'unknown'; |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	const [id, username] = source.value.split('@'); |  | ||||||
|  |  | ||||||
| 	if (service) { |  | ||||||
| 		target[source.name.split(':')[2]] = service(id, username); |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
| import { bindThis } from '@/decorators.js'; |  | ||||||
|  |  | ||||||
| @Injectable() | @Injectable() | ||||||
| export class ApPersonService implements OnModuleInit { | export class ApPersonService implements OnModuleInit { | ||||||
| 	private utilityService: UtilityService; | 	private utilityService: UtilityService; | ||||||
| @@ -282,6 +252,12 @@ export class ApPersonService implements OnModuleInit { | |||||||
|  |  | ||||||
| 		const bday = person['vcard:bday']?.match(/^\d{4}-\d{2}-\d{2}/); | 		const bday = person['vcard:bday']?.match(/^\d{4}-\d{2}-\d{2}/); | ||||||
|  |  | ||||||
|  | 		const url = getOneApHrefNullable(person.url); | ||||||
|  |  | ||||||
|  | 		if (url && !url.startsWith('https://')) { | ||||||
|  | 			throw new Error('unexpected shcema of person url: ' + url); | ||||||
|  | 		} | ||||||
|  |  | ||||||
| 		// Create user | 		// Create user | ||||||
| 		let user: IRemoteUser; | 		let user: IRemoteUser; | ||||||
| 		try { | 		try { | ||||||
| @@ -313,7 +289,7 @@ export class ApPersonService implements OnModuleInit { | |||||||
| 				await transactionalEntityManager.save(new UserProfile({ | 				await transactionalEntityManager.save(new UserProfile({ | ||||||
| 					userId: user.id, | 					userId: user.id, | ||||||
| 					description: person.summary ? this.apMfmService.htmlToMfm(truncate(person.summary, summaryLength), person.tag) : null, | 					description: person.summary ? this.apMfmService.htmlToMfm(truncate(person.summary, summaryLength), person.tag) : null, | ||||||
| 					url: getOneApHrefNullable(person.url), | 					url: url, | ||||||
| 					fields, | 					fields, | ||||||
| 					birthday: bday ? bday[0] : null, | 					birthday: bday ? bday[0] : null, | ||||||
| 					location: person['vcard:Address'] ?? null, | 					location: person['vcard:Address'] ?? null, | ||||||
| @@ -455,6 +431,12 @@ export class ApPersonService implements OnModuleInit { | |||||||
|  |  | ||||||
| 		const bday = person['vcard:bday']?.match(/^\d{4}-\d{2}-\d{2}/); | 		const bday = person['vcard:bday']?.match(/^\d{4}-\d{2}-\d{2}/); | ||||||
|  |  | ||||||
|  | 		const url = getOneApHrefNullable(person.url); | ||||||
|  |  | ||||||
|  | 		if (url && !url.startsWith('https://')) { | ||||||
|  | 			throw new Error('unexpected shcema of person url: ' + url); | ||||||
|  | 		} | ||||||
|  |  | ||||||
| 		const updates = { | 		const updates = { | ||||||
| 			lastFetchedAt: new Date(), | 			lastFetchedAt: new Date(), | ||||||
| 			inbox: person.inbox, | 			inbox: person.inbox, | ||||||
| @@ -489,7 +471,7 @@ export class ApPersonService implements OnModuleInit { | |||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		await this.userProfilesRepository.update({ userId: exist.id }, { | 		await this.userProfilesRepository.update({ userId: exist.id }, { | ||||||
| 			url: getOneApHrefNullable(person.url), | 			url: url, | ||||||
| 			fields, | 			fields, | ||||||
| 			description: person.summary ? this.apMfmService.htmlToMfm(truncate(person.summary, summaryLength), person.tag) : null, | 			description: person.summary ? this.apMfmService.htmlToMfm(truncate(person.summary, summaryLength), person.tag) : null, | ||||||
| 			birthday: bday ? bday[0] : null, | 			birthday: bday ? bday[0] : null, | ||||||
| @@ -540,22 +522,16 @@ export class ApPersonService implements OnModuleInit { | |||||||
| 		name: string, | 		name: string, | ||||||
| 		value: string | 		value: string | ||||||
| 	}[] = []; | 	}[] = []; | ||||||
| 		const services: { [x: string]: any } = {}; |  | ||||||
|  |  | ||||||
| 		if (Array.isArray(attachments)) { | 		if (Array.isArray(attachments)) { | ||||||
| 			for (const attachment of attachments.filter(isPropertyValue)) { | 			for (const attachment of attachments.filter(isPropertyValue)) { | ||||||
| 				if (isPropertyValue(attachment.identifier)) { | 				fields.push({ | ||||||
| 					addService(services, attachment.identifier); | 					name: attachment.name, | ||||||
| 				} else { | 					value: this.mfmService.fromHtml(attachment.value), | ||||||
| 					fields.push({ | 				}); | ||||||
| 						name: attachment.name, |  | ||||||
| 						value: this.mfmService.fromHtml(attachment.value), |  | ||||||
| 					}); |  | ||||||
| 				} |  | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		return { fields, services }; | 		return { fields }; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
|   | |||||||
| @@ -10,7 +10,6 @@ import PerUserNotesChart from './charts/per-user-notes.js'; | |||||||
| import PerUserPvChart from './charts/per-user-pv.js'; | import PerUserPvChart from './charts/per-user-pv.js'; | ||||||
| import DriveChart from './charts/drive.js'; | import DriveChart from './charts/drive.js'; | ||||||
| import PerUserReactionsChart from './charts/per-user-reactions.js'; | import PerUserReactionsChart from './charts/per-user-reactions.js'; | ||||||
| import HashtagChart from './charts/hashtag.js'; |  | ||||||
| import PerUserFollowingChart from './charts/per-user-following.js'; | import PerUserFollowingChart from './charts/per-user-following.js'; | ||||||
| import PerUserDriveChart from './charts/per-user-drive.js'; | import PerUserDriveChart from './charts/per-user-drive.js'; | ||||||
| import ApRequestChart from './charts/ap-request.js'; | import ApRequestChart from './charts/ap-request.js'; | ||||||
| @@ -31,7 +30,6 @@ export class ChartManagementService implements OnApplicationShutdown { | |||||||
| 		private perUserPvChart: PerUserPvChart, | 		private perUserPvChart: PerUserPvChart, | ||||||
| 		private driveChart: DriveChart, | 		private driveChart: DriveChart, | ||||||
| 		private perUserReactionsChart: PerUserReactionsChart, | 		private perUserReactionsChart: PerUserReactionsChart, | ||||||
| 		private hashtagChart: HashtagChart, |  | ||||||
| 		private perUserFollowingChart: PerUserFollowingChart, | 		private perUserFollowingChart: PerUserFollowingChart, | ||||||
| 		private perUserDriveChart: PerUserDriveChart, | 		private perUserDriveChart: PerUserDriveChart, | ||||||
| 		private apRequestChart: ApRequestChart, | 		private apRequestChart: ApRequestChart, | ||||||
| @@ -46,7 +44,6 @@ export class ChartManagementService implements OnApplicationShutdown { | |||||||
| 			this.perUserPvChart, | 			this.perUserPvChart, | ||||||
| 			this.driveChart, | 			this.driveChart, | ||||||
| 			this.perUserReactionsChart, | 			this.perUserReactionsChart, | ||||||
| 			this.hashtagChart, |  | ||||||
| 			this.perUserFollowingChart, | 			this.perUserFollowingChart, | ||||||
| 			this.perUserDriveChart, | 			this.perUserDriveChart, | ||||||
| 			this.apRequestChart, | 			this.apRequestChart, | ||||||
|   | |||||||
| @@ -1,10 +0,0 @@ | |||||||
| import Chart from '../../core.js'; |  | ||||||
|  |  | ||||||
| export const name = 'hashtag'; |  | ||||||
|  |  | ||||||
| export const schema = { |  | ||||||
| 	'local.users': { uniqueIncrement: true }, |  | ||||||
| 	'remote.users': { uniqueIncrement: true }, |  | ||||||
| } as const; |  | ||||||
|  |  | ||||||
| export const entity = Chart.schemaToEntity(name, schema, true); |  | ||||||
| @@ -1,45 +0,0 @@ | |||||||
| import { Injectable, Inject } from '@nestjs/common'; |  | ||||||
| import { DataSource } from 'typeorm'; |  | ||||||
| import type { User } from '@/models/entities/User.js'; |  | ||||||
| import { AppLockService } from '@/core/AppLockService.js'; |  | ||||||
| import { DI } from '@/di-symbols.js'; |  | ||||||
| import { UserEntityService } from '@/core/entities/UserEntityService.js'; |  | ||||||
| import { bindThis } from '@/decorators.js'; |  | ||||||
| import Chart from '../core.js'; |  | ||||||
| import { ChartLoggerService } from '../ChartLoggerService.js'; |  | ||||||
| import { name, schema } from './entities/hashtag.js'; |  | ||||||
| import type { KVs } from '../core.js'; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * ハッシュタグに関するチャート |  | ||||||
|  */ |  | ||||||
| // eslint-disable-next-line import/no-default-export |  | ||||||
| @Injectable() |  | ||||||
| export default class HashtagChart extends Chart<typeof schema> { |  | ||||||
| 	constructor( |  | ||||||
| 		@Inject(DI.db) |  | ||||||
| 		private db: DataSource, |  | ||||||
|  |  | ||||||
| 		private appLockService: AppLockService, |  | ||||||
| 		private userEntityService: UserEntityService, |  | ||||||
| 		private chartLoggerService: ChartLoggerService, |  | ||||||
| 	) { |  | ||||||
| 		super(db, (k) => appLockService.getChartInsertLock(k), chartLoggerService.logger, name, schema, true); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	protected async tickMajor(): Promise<Partial<KVs<typeof schema>>> { |  | ||||||
| 		return {}; |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	protected async tickMinor(): Promise<Partial<KVs<typeof schema>>> { |  | ||||||
| 		return {}; |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	@bindThis |  | ||||||
| 	public async update(hashtag: string, user: { id: User['id'], host: User['host'] }): Promise<void> { |  | ||||||
| 		await this.commit({ |  | ||||||
| 			'local.users': this.userEntityService.isLocalUser(user) ? [user.id] : [], |  | ||||||
| 			'remote.users': this.userEntityService.isLocalUser(user) ? [] : [user.id], |  | ||||||
| 		}, hashtag); |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
| @@ -11,9 +11,9 @@ import type Logger from '@/logger.js'; | |||||||
| import { bindThis } from '@/decorators.js'; | import { bindThis } from '@/decorators.js'; | ||||||
| import type { Repository, DataSource } from 'typeorm'; | import type { Repository, DataSource } from 'typeorm'; | ||||||
|  |  | ||||||
| const columnPrefix = '___' as const; | const COLUMN_PREFIX = '___' as const; | ||||||
| const uniqueTempColumnPrefix = 'unique_temp___' as const; | const UNIQUE_TEMP_COLUMN_PREFIX = 'unique_temp___' as const; | ||||||
| const columnDot = '_' as const; | const COLUMN_DELIMITER = '_' as const; | ||||||
|  |  | ||||||
| type Schema = Record<string, { | type Schema = Record<string, { | ||||||
| 	uniqueIncrement?: boolean; | 	uniqueIncrement?: boolean; | ||||||
| @@ -26,14 +26,14 @@ type Schema = Record<string, { | |||||||
| 	accumulate?: boolean; | 	accumulate?: boolean; | ||||||
| }>; | }>; | ||||||
|  |  | ||||||
| type KeyToColumnName<T extends string> = T extends `${infer R1}.${infer R2}` ? `${R1}${typeof columnDot}${KeyToColumnName<R2>}` : T; | type KeyToColumnName<T extends string> = T extends `${infer R1}.${infer R2}` ? `${R1}${typeof COLUMN_DELIMITER}${KeyToColumnName<R2>}` : T; | ||||||
|  |  | ||||||
| type Columns<S extends Schema> = { | type Columns<S extends Schema> = { | ||||||
| 	[K in keyof S as `${typeof columnPrefix}${KeyToColumnName<string & K>}`]: number; | 	[K in keyof S as `${typeof COLUMN_PREFIX}${KeyToColumnName<string & K>}`]: number; | ||||||
| }; | }; | ||||||
|  |  | ||||||
| type TempColumnsForUnique<S extends Schema> = { | type TempColumnsForUnique<S extends Schema> = { | ||||||
| 	[K in keyof S as `${typeof uniqueTempColumnPrefix}${KeyToColumnName<string & K>}`]: S[K]['uniqueIncrement'] extends true ? string[] : never; | 	[K in keyof S as `${typeof UNIQUE_TEMP_COLUMN_PREFIX}${KeyToColumnName<string & K>}`]: S[K]['uniqueIncrement'] extends true ? string[] : never; | ||||||
| }; | }; | ||||||
|  |  | ||||||
| type RawRecord<S extends Schema> = { | type RawRecord<S extends Schema> = { | ||||||
| @@ -138,20 +138,20 @@ export default abstract class Chart<T extends Schema> { | |||||||
| 	private static convertSchemaToColumnDefinitions(schema: Schema): Record<string, { type: string; array?: boolean; default?: any; }> { | 	private static convertSchemaToColumnDefinitions(schema: Schema): Record<string, { type: string; array?: boolean; default?: any; }> { | ||||||
| 		const columns = {} as Record<string, { type: string; array?: boolean; default?: any; }>; | 		const columns = {} as Record<string, { type: string; array?: boolean; default?: any; }>; | ||||||
| 		for (const [k, v] of Object.entries(schema)) { | 		for (const [k, v] of Object.entries(schema)) { | ||||||
| 			const name = k.replaceAll('.', columnDot); | 			const name = k.replaceAll('.', COLUMN_DELIMITER); | ||||||
| 			const type = v.range === 'big' ? 'bigint' : v.range === 'small' ? 'smallint' : 'integer'; | 			const type = v.range === 'big' ? 'bigint' : v.range === 'small' ? 'smallint' : 'integer'; | ||||||
| 			if (v.uniqueIncrement) { | 			if (v.uniqueIncrement) { | ||||||
| 				columns[uniqueTempColumnPrefix + name] = { | 				columns[UNIQUE_TEMP_COLUMN_PREFIX + name] = { | ||||||
| 					type: 'varchar', | 					type: 'varchar', | ||||||
| 					array: true, | 					array: true, | ||||||
| 					default: '{}', | 					default: '{}', | ||||||
| 				}; | 				}; | ||||||
| 				columns[columnPrefix + name] = { | 				columns[COLUMN_PREFIX + name] = { | ||||||
| 					type, | 					type, | ||||||
| 					default: 0, | 					default: 0, | ||||||
| 				}; | 				}; | ||||||
| 			} else { | 			} else { | ||||||
| 				columns[columnPrefix + name] = { | 				columns[COLUMN_PREFIX + name] = { | ||||||
| 					type, | 					type, | ||||||
| 					default: 0, | 					default: 0, | ||||||
| 				}; | 				}; | ||||||
| @@ -253,8 +253,8 @@ export default abstract class Chart<T extends Schema> { | |||||||
| 	@bindThis | 	@bindThis | ||||||
| 	private convertRawRecord(x: RawRecord<T>): KVs<T> { | 	private convertRawRecord(x: RawRecord<T>): KVs<T> { | ||||||
| 		const kvs = {} as Record<string, number>; | 		const kvs = {} as Record<string, number>; | ||||||
| 		for (const k of Object.keys(x).filter((k) => k.startsWith(columnPrefix)) as (keyof Columns<T>)[]) { | 		for (const k of Object.keys(x).filter((k) => k.startsWith(COLUMN_PREFIX)) as (keyof Columns<T>)[]) { | ||||||
| 			kvs[(k as string).substr(columnPrefix.length).split(columnDot).join('.')] = x[k] as unknown as number; | 			kvs[(k as string).substr(COLUMN_PREFIX.length).split(COLUMN_DELIMITER).join('.')] = x[k] as unknown as number; | ||||||
| 		} | 		} | ||||||
| 		return kvs as KVs<T>; | 		return kvs as KVs<T>; | ||||||
| 	} | 	} | ||||||
| @@ -357,8 +357,8 @@ export default abstract class Chart<T extends Schema> { | |||||||
|  |  | ||||||
| 			const columns = {} as Record<string, number | unknown[]>; | 			const columns = {} as Record<string, number | unknown[]>; | ||||||
| 			for (const [k, v] of Object.entries(data)) { | 			for (const [k, v] of Object.entries(data)) { | ||||||
| 				const name = k.replaceAll('.', columnDot); | 				const name = k.replaceAll('.', COLUMN_DELIMITER); | ||||||
| 				columns[columnPrefix + name] = v; | 				columns[COLUMN_PREFIX + name] = v; | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
| 			// 新規ログ挿入 | 			// 新規ログ挿入 | ||||||
| @@ -419,13 +419,13 @@ export default abstract class Chart<T extends Schema> { | |||||||
| 			const queryForDay: Record<keyof RawRecord<T>, number | (() => string)> = {} as any; | 			const queryForDay: Record<keyof RawRecord<T>, number | (() => string)> = {} as any; | ||||||
| 			for (const [k, v] of Object.entries(finalDiffs)) { | 			for (const [k, v] of Object.entries(finalDiffs)) { | ||||||
| 				if (typeof v === 'number') { | 				if (typeof v === 'number') { | ||||||
| 					const name = columnPrefix + k.replaceAll('.', columnDot) as string & keyof Columns<T>; | 					const name = COLUMN_PREFIX + k.replaceAll('.', COLUMN_DELIMITER) as string & keyof Columns<T>; | ||||||
| 					if (v > 0) queryForHour[name] = () => `"${name}" + ${v}`; | 					if (v > 0) queryForHour[name] = () => `"${name}" + ${v}`; | ||||||
| 					if (v < 0) queryForHour[name] = () => `"${name}" - ${Math.abs(v)}`; | 					if (v < 0) queryForHour[name] = () => `"${name}" - ${Math.abs(v)}`; | ||||||
| 					if (v > 0) queryForDay[name] = () => `"${name}" + ${v}`; | 					if (v > 0) queryForDay[name] = () => `"${name}" + ${v}`; | ||||||
| 					if (v < 0) queryForDay[name] = () => `"${name}" - ${Math.abs(v)}`; | 					if (v < 0) queryForDay[name] = () => `"${name}" - ${Math.abs(v)}`; | ||||||
| 				} else if (Array.isArray(v) && v.length > 0) { // ユニークインクリメント | 				} else if (Array.isArray(v) && v.length > 0) { // ユニークインクリメント | ||||||
| 					const tempColumnName = uniqueTempColumnPrefix + k.replaceAll('.', columnDot) as string & keyof TempColumnsForUnique<T>; | 					const tempColumnName = UNIQUE_TEMP_COLUMN_PREFIX + k.replaceAll('.', COLUMN_DELIMITER) as string & keyof TempColumnsForUnique<T>; | ||||||
| 					// TODO: item をSQLエスケープ | 					// TODO: item をSQLエスケープ | ||||||
| 					const itemsForHour = v.filter(item => !(logHour[tempColumnName] as unknown as string[]).includes(item)).map(item => `"${item}"`); | 					const itemsForHour = v.filter(item => !(logHour[tempColumnName] as unknown as string[]).includes(item)).map(item => `"${item}"`); | ||||||
| 					const itemsForDay = v.filter(item => !(logDay[tempColumnName] as unknown as string[]).includes(item)).map(item => `"${item}"`); | 					const itemsForDay = v.filter(item => !(logDay[tempColumnName] as unknown as string[]).includes(item)).map(item => `"${item}"`); | ||||||
| @@ -437,8 +437,8 @@ export default abstract class Chart<T extends Schema> { | |||||||
| 			// bake unique count | 			// bake unique count | ||||||
| 			for (const [k, v] of Object.entries(finalDiffs)) { | 			for (const [k, v] of Object.entries(finalDiffs)) { | ||||||
| 				if (this.schema[k].uniqueIncrement) { | 				if (this.schema[k].uniqueIncrement) { | ||||||
| 					const name = columnPrefix + k.replaceAll('.', columnDot) as keyof Columns<T>; | 					const name = COLUMN_PREFIX + k.replaceAll('.', COLUMN_DELIMITER) as keyof Columns<T>; | ||||||
| 					const tempColumnName = uniqueTempColumnPrefix + k.replaceAll('.', columnDot) as keyof TempColumnsForUnique<T>; | 					const tempColumnName = UNIQUE_TEMP_COLUMN_PREFIX + k.replaceAll('.', COLUMN_DELIMITER) as keyof TempColumnsForUnique<T>; | ||||||
| 					queryForHour[name] = new Set([...(v as string[]), ...(logHour[tempColumnName] as unknown as string[])]).size; | 					queryForHour[name] = new Set([...(v as string[]), ...(logHour[tempColumnName] as unknown as string[])]).size; | ||||||
| 					queryForDay[name] = new Set([...(v as string[]), ...(logDay[tempColumnName] as unknown as string[])]).size; | 					queryForDay[name] = new Set([...(v as string[]), ...(logDay[tempColumnName] as unknown as string[])]).size; | ||||||
| 				} | 				} | ||||||
| @@ -449,15 +449,15 @@ export default abstract class Chart<T extends Schema> { | |||||||
| 			for (const [k, v] of Object.entries(this.schema)) { | 			for (const [k, v] of Object.entries(this.schema)) { | ||||||
| 				const intersection = v.intersection; | 				const intersection = v.intersection; | ||||||
| 				if (intersection) { | 				if (intersection) { | ||||||
| 					const name = columnPrefix + k.replaceAll('.', columnDot) as keyof Columns<T>; | 					const name = COLUMN_PREFIX + k.replaceAll('.', COLUMN_DELIMITER) as keyof Columns<T>; | ||||||
| 					const firstKey = intersection[0]; | 					const firstKey = intersection[0]; | ||||||
| 					const firstTempColumnName = uniqueTempColumnPrefix + firstKey.replaceAll('.', columnDot) as keyof TempColumnsForUnique<T>; | 					const firstTempColumnName = UNIQUE_TEMP_COLUMN_PREFIX + firstKey.replaceAll('.', COLUMN_DELIMITER) as keyof TempColumnsForUnique<T>; | ||||||
| 					const firstValues = finalDiffs[firstKey] as string[] | undefined; | 					const firstValues = finalDiffs[firstKey] as string[] | undefined; | ||||||
| 					const currentValuesForHour = new Set([...(firstValues ?? []), ...(logHour[firstTempColumnName] as unknown as string[])]); | 					const currentValuesForHour = new Set([...(firstValues ?? []), ...(logHour[firstTempColumnName] as unknown as string[])]); | ||||||
| 					const currentValuesForDay = new Set([...(firstValues ?? []), ...(logDay[firstTempColumnName] as unknown as string[])]); | 					const currentValuesForDay = new Set([...(firstValues ?? []), ...(logDay[firstTempColumnName] as unknown as string[])]); | ||||||
| 					for (let i = 1; i < intersection.length; i++) { | 					for (let i = 1; i < intersection.length; i++) { | ||||||
| 						const targetKey = intersection[i]; | 						const targetKey = intersection[i]; | ||||||
| 						const targetTempColumnName = uniqueTempColumnPrefix + targetKey.replaceAll('.', columnDot) as keyof TempColumnsForUnique<T>; | 						const targetTempColumnName = UNIQUE_TEMP_COLUMN_PREFIX + targetKey.replaceAll('.', COLUMN_DELIMITER) as keyof TempColumnsForUnique<T>; | ||||||
| 						const targetValues = finalDiffs[targetKey] as string[] | undefined; | 						const targetValues = finalDiffs[targetKey] as string[] | undefined; | ||||||
| 						const targetValuesForHour = new Set([...(targetValues ?? []), ...(logHour[targetTempColumnName] as unknown as string[])]); | 						const targetValuesForHour = new Set([...(targetValues ?? []), ...(logHour[targetTempColumnName] as unknown as string[])]); | ||||||
| 						const targetValuesForDay = new Set([...(targetValues ?? []), ...(logDay[targetTempColumnName] as unknown as string[])]); | 						const targetValuesForDay = new Set([...(targetValues ?? []), ...(logDay[targetTempColumnName] as unknown as string[])]); | ||||||
| @@ -510,7 +510,7 @@ export default abstract class Chart<T extends Schema> { | |||||||
|  |  | ||||||
| 		const columns = {} as Record<keyof Columns<T>, number>; | 		const columns = {} as Record<keyof Columns<T>, number>; | ||||||
| 		for (const [k, v] of Object.entries(data) as ([keyof typeof data, number])[]) { | 		for (const [k, v] of Object.entries(data) as ([keyof typeof data, number])[]) { | ||||||
| 			const name = columnPrefix + (k as string).replaceAll('.', columnDot) as keyof Columns<T>; | 			const name = COLUMN_PREFIX + (k as string).replaceAll('.', COLUMN_DELIMITER) as keyof Columns<T>; | ||||||
| 			columns[name] = v; | 			columns[name] = v; | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| @@ -556,7 +556,7 @@ export default abstract class Chart<T extends Schema> { | |||||||
| 		const columns = {} as Record<keyof TempColumnsForUnique<T>, []>; | 		const columns = {} as Record<keyof TempColumnsForUnique<T>, []>; | ||||||
| 		for (const [k, v] of Object.entries(this.schema)) { | 		for (const [k, v] of Object.entries(this.schema)) { | ||||||
| 			if (v.uniqueIncrement) { | 			if (v.uniqueIncrement) { | ||||||
| 				const name = uniqueTempColumnPrefix + k.replaceAll('.', columnDot) as keyof TempColumnsForUnique<T>; | 				const name = UNIQUE_TEMP_COLUMN_PREFIX + k.replaceAll('.', COLUMN_DELIMITER) as keyof TempColumnsForUnique<T>; | ||||||
| 				columns[name] = []; | 				columns[name] = []; | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
|   | |||||||
| @@ -7,7 +7,6 @@ import { entity as PerUserNotesChart } from './charts/entities/per-user-notes.js | |||||||
| import { entity as PerUserPvChart } from './charts/entities/per-user-pv.js'; | import { entity as PerUserPvChart } from './charts/entities/per-user-pv.js'; | ||||||
| import { entity as DriveChart } from './charts/entities/drive.js'; | import { entity as DriveChart } from './charts/entities/drive.js'; | ||||||
| import { entity as PerUserReactionsChart } from './charts/entities/per-user-reactions.js'; | import { entity as PerUserReactionsChart } from './charts/entities/per-user-reactions.js'; | ||||||
| import { entity as HashtagChart } from './charts/entities/hashtag.js'; |  | ||||||
| import { entity as PerUserFollowingChart } from './charts/entities/per-user-following.js'; | import { entity as PerUserFollowingChart } from './charts/entities/per-user-following.js'; | ||||||
| import { entity as PerUserDriveChart } from './charts/entities/per-user-drive.js'; | import { entity as PerUserDriveChart } from './charts/entities/per-user-drive.js'; | ||||||
| import { entity as ApRequestChart } from './charts/entities/ap-request.js'; | import { entity as ApRequestChart } from './charts/entities/ap-request.js'; | ||||||
| @@ -27,7 +26,6 @@ export const entities = [ | |||||||
| 	PerUserPvChart.hour, PerUserPvChart.day, | 	PerUserPvChart.hour, PerUserPvChart.day, | ||||||
| 	DriveChart.hour, DriveChart.day, | 	DriveChart.hour, DriveChart.day, | ||||||
| 	PerUserReactionsChart.hour, PerUserReactionsChart.day, | 	PerUserReactionsChart.hour, PerUserReactionsChart.day, | ||||||
| 	HashtagChart.hour, HashtagChart.day, |  | ||||||
| 	PerUserFollowingChart.hour, PerUserFollowingChart.day, | 	PerUserFollowingChart.hour, PerUserFollowingChart.day, | ||||||
| 	PerUserDriveChart.hour, PerUserDriveChart.day, | 	PerUserDriveChart.hour, PerUserDriveChart.day, | ||||||
| 	ApRequestChart.hour, ApRequestChart.day, | 	ApRequestChart.hour, ApRequestChart.day, | ||||||
|   | |||||||
| @@ -54,7 +54,7 @@ export class ChannelEntityService { | |||||||
| 			name: channel.name, | 			name: channel.name, | ||||||
| 			description: channel.description, | 			description: channel.description, | ||||||
| 			userId: channel.userId, | 			userId: channel.userId, | ||||||
| 			bannerUrl: banner ? this.driveFileEntityService.getPublicUrl(banner, false) : null, | 			bannerUrl: banner ? this.driveFileEntityService.getPublicUrl(banner) : null, | ||||||
| 			usersCount: channel.usersCount, | 			usersCount: channel.usersCount, | ||||||
| 			notesCount: channel.notesCount, | 			notesCount: channel.notesCount, | ||||||
|  |  | ||||||
|   | |||||||
| @@ -13,6 +13,7 @@ import { deepClone } from '@/misc/clone.js'; | |||||||
| import { UtilityService } from '../UtilityService.js'; | import { UtilityService } from '../UtilityService.js'; | ||||||
| import { UserEntityService } from './UserEntityService.js'; | import { UserEntityService } from './UserEntityService.js'; | ||||||
| import { DriveFolderEntityService } from './DriveFolderEntityService.js'; | import { DriveFolderEntityService } from './DriveFolderEntityService.js'; | ||||||
|  | import { VideoProcessingService } from '../VideoProcessingService.js'; | ||||||
|  |  | ||||||
| type PackOptions = { | type PackOptions = { | ||||||
| 	detail?: boolean, | 	detail?: boolean, | ||||||
| @@ -20,6 +21,7 @@ type PackOptions = { | |||||||
| 	withUser?: boolean, | 	withUser?: boolean, | ||||||
| }; | }; | ||||||
| import { bindThis } from '@/decorators.js'; | import { bindThis } from '@/decorators.js'; | ||||||
|  | import { isMimeImage } from '@/misc/is-mime-image.js'; | ||||||
|  |  | ||||||
| @Injectable() | @Injectable() | ||||||
| export class DriveFileEntityService { | export class DriveFileEntityService { | ||||||
| @@ -42,6 +44,7 @@ export class DriveFileEntityService { | |||||||
|  |  | ||||||
| 		private utilityService: UtilityService, | 		private utilityService: UtilityService, | ||||||
| 		private driveFolderEntityService: DriveFolderEntityService, | 		private driveFolderEntityService: DriveFolderEntityService, | ||||||
|  | 		private videoProcessingService: VideoProcessingService, | ||||||
| 	) { | 	) { | ||||||
| 	} | 	} | ||||||
| 	 | 	 | ||||||
| @@ -71,27 +74,65 @@ export class DriveFileEntityService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public getPublicUrl(file: DriveFile, thumbnail = false): string | null { | 	private getProxiedUrl(url: string, mode?: 'static' | 'avatar'): string | null { | ||||||
|  | 		return appendQuery( | ||||||
|  | 			`${this.config.mediaProxy}/${mode ?? 'image'}.webp`, | ||||||
|  | 			query({ | ||||||
|  | 				url, | ||||||
|  | 				...(mode ? { [mode]: '1' } : {}), | ||||||
|  | 			}) | ||||||
|  | 		) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	@bindThis | ||||||
|  | 	public getThumbnailUrl(file: DriveFile): string | null { | ||||||
|  | 		if (file.type.startsWith('video')) { | ||||||
|  | 			if (file.thumbnailUrl) return file.thumbnailUrl; | ||||||
|  |  | ||||||
|  | 			if (this.config.videoThumbnailGenerator == null) { | ||||||
|  | 				return this.videoProcessingService.getExternalVideoThumbnailUrl(file.webpublicUrl ?? file.url ?? file.uri); | ||||||
|  | 			} | ||||||
|  | 		} else if (file.uri != null && file.userHost != null && this.config.externalMediaProxyEnabled) { | ||||||
|  | 			// 動画ではなくリモートかつメディアプロキシ | ||||||
|  | 			return this.getProxiedUrl(file.uri, 'static'); | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if (file.uri != null && file.isLink && this.config.proxyRemoteFiles) { | ||||||
|  | 			// リモートかつ期限切れはローカルプロキシを試みる | ||||||
|  | 			// 従来は/files/${thumbnailAccessKey}にアクセスしていたが、 | ||||||
|  | 			// /filesはメディアプロキシにリダイレクトするようにしたため直接メディアプロキシを指定する | ||||||
|  | 			return this.getProxiedUrl(file.uri, 'static'); | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		const url = file.webpublicUrl ?? file.url; | ||||||
|  |  | ||||||
|  | 		return file.thumbnailUrl ?? (isMimeImage(file.type, 'sharp-convertible-image') ? this.getProxiedUrl(url, 'static') : null); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	@bindThis | ||||||
|  | 	public getPublicUrl(file: DriveFile, mode?: 'avatar'): string | null { // static = thumbnail | ||||||
| 		// リモートかつメディアプロキシ | 		// リモートかつメディアプロキシ | ||||||
| 		if (file.uri != null && file.userHost != null && this.config.mediaProxy != null) { | 		if (file.uri != null && file.userHost != null && this.config.externalMediaProxyEnabled) { | ||||||
| 			return appendQuery(this.config.mediaProxy, query({ | 			return this.getProxiedUrl(file.uri, mode); | ||||||
| 				url: file.uri, |  | ||||||
| 				thumbnail: thumbnail ? '1' : undefined, |  | ||||||
| 			})); |  | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		// リモートかつ期限切れはローカルプロキシを試みる | 		// リモートかつ期限切れはローカルプロキシを試みる | ||||||
| 		if (file.uri != null && file.isLink && this.config.proxyRemoteFiles) { | 		if (file.uri != null && file.isLink && this.config.proxyRemoteFiles) { | ||||||
| 			const key = thumbnail ? file.thumbnailAccessKey : file.webpublicAccessKey; | 			const key = file.webpublicAccessKey; | ||||||
|  |  | ||||||
| 			if (key && !key.match('/')) {	// 古いものはここにオブジェクトストレージキーが入ってるので除外 | 			if (key && !key.match('/')) {	// 古いものはここにオブジェクトストレージキーが入ってるので除外 | ||||||
| 				return `${this.config.url}/files/${key}`; | 				const url = `${this.config.url}/files/${key}`; | ||||||
|  | 				if (mode === 'avatar') return this.getProxiedUrl(file.uri, 'avatar'); | ||||||
|  | 				return url; | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		const isImage = file.type && ['image/png', 'image/apng', 'image/gif', 'image/jpeg', 'image/webp', 'image/avif', 'image/svg+xml'].includes(file.type); | 		const url = file.webpublicUrl ?? file.url; | ||||||
|  |  | ||||||
| 		return thumbnail ? (file.thumbnailUrl ?? (isImage ? (file.webpublicUrl ?? file.url) : null)) : (file.webpublicUrl ?? file.url); | 		if (mode === 'avatar') { | ||||||
|  | 			return this.getProxiedUrl(url, 'avatar'); | ||||||
|  | 		} | ||||||
|  | 		return url; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| @@ -166,8 +207,8 @@ export class DriveFileEntityService { | |||||||
| 			isSensitive: file.isSensitive, | 			isSensitive: file.isSensitive, | ||||||
| 			blurhash: file.blurhash, | 			blurhash: file.blurhash, | ||||||
| 			properties: opts.self ? file.properties : this.getPublicProperties(file), | 			properties: opts.self ? file.properties : this.getPublicProperties(file), | ||||||
| 			url: opts.self ? file.url : this.getPublicUrl(file, false), | 			url: opts.self ? file.url : this.getPublicUrl(file), | ||||||
| 			thumbnailUrl: this.getPublicUrl(file, true), | 			thumbnailUrl: this.getThumbnailUrl(file), | ||||||
| 			comment: file.comment, | 			comment: file.comment, | ||||||
| 			folderId: file.folderId, | 			folderId: file.folderId, | ||||||
| 			folder: opts.detail && file.folderId ? this.driveFolderEntityService.pack(file.folderId, { | 			folder: opts.detail && file.folderId ? this.driveFolderEntityService.pack(file.folderId, { | ||||||
| @@ -201,8 +242,8 @@ export class DriveFileEntityService { | |||||||
| 			isSensitive: file.isSensitive, | 			isSensitive: file.isSensitive, | ||||||
| 			blurhash: file.blurhash, | 			blurhash: file.blurhash, | ||||||
| 			properties: opts.self ? file.properties : this.getPublicProperties(file), | 			properties: opts.self ? file.properties : this.getPublicProperties(file), | ||||||
| 			url: opts.self ? file.url : this.getPublicUrl(file, false), | 			url: opts.self ? file.url : this.getPublicUrl(file), | ||||||
| 			thumbnailUrl: this.getPublicUrl(file, true), | 			thumbnailUrl: this.getThumbnailUrl(file), | ||||||
| 			comment: file.comment, | 			comment: file.comment, | ||||||
| 			folderId: file.folderId, | 			folderId: file.folderId, | ||||||
| 			folder: opts.detail && file.folderId ? this.driveFolderEntityService.pack(file.folderId, { | 			folder: opts.detail && file.folderId ? this.driveFolderEntityService.pack(file.folderId, { | ||||||
|   | |||||||
| @@ -2,7 +2,7 @@ import { Inject, Injectable } from '@nestjs/common'; | |||||||
| import { In } from 'typeorm'; | import { In } from 'typeorm'; | ||||||
| import { ModuleRef } from '@nestjs/core'; | import { ModuleRef } from '@nestjs/core'; | ||||||
| import { DI } from '@/di-symbols.js'; | import { DI } from '@/di-symbols.js'; | ||||||
| import type { AccessTokensRepository, NoteReactionsRepository, NotificationsRepository } from '@/models/index.js'; | import type { AccessTokensRepository, NoteReactionsRepository, NotificationsRepository, User } from '@/models/index.js'; | ||||||
| import { awaitAll } from '@/misc/prelude/await-all.js'; | import { awaitAll } from '@/misc/prelude/await-all.js'; | ||||||
| import type { Notification } from '@/models/entities/Notification.js'; | import type { Notification } from '@/models/entities/Notification.js'; | ||||||
| import type { NoteReaction } from '@/models/entities/NoteReaction.js'; | import type { NoteReaction } from '@/models/entities/NoteReaction.js'; | ||||||
|   | |||||||
| @@ -56,11 +56,13 @@ export class RoleEntityService { | |||||||
| 			name: role.name, | 			name: role.name, | ||||||
| 			description: role.description, | 			description: role.description, | ||||||
| 			color: role.color, | 			color: role.color, | ||||||
|  | 			iconUrl: role.iconUrl, | ||||||
| 			target: role.target, | 			target: role.target, | ||||||
| 			condFormula: role.condFormula, | 			condFormula: role.condFormula, | ||||||
| 			isPublic: role.isPublic, | 			isPublic: role.isPublic, | ||||||
| 			isAdministrator: role.isAdministrator, | 			isAdministrator: role.isAdministrator, | ||||||
| 			isModerator: role.isModerator, | 			isModerator: role.isModerator, | ||||||
|  | 			asBadge: role.asBadge, | ||||||
| 			canEditMembersByModerator: role.canEditMembersByModerator, | 			canEditMembersByModerator: role.canEditMembersByModerator, | ||||||
| 			policies: policies, | 			policies: policies, | ||||||
| 			usersCount: assigns.length, | 			usersCount: assigns.length, | ||||||
|   | |||||||
| @@ -314,10 +314,10 @@ export class UserEntityService implements OnModuleInit { | |||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public async getAvatarUrl(user: User): Promise<string> { | 	public async getAvatarUrl(user: User): Promise<string> { | ||||||
| 		if (user.avatar) { | 		if (user.avatar) { | ||||||
| 			return this.driveFileEntityService.getPublicUrl(user.avatar, true) ?? this.getIdenticonUrl(user.id); | 			return this.driveFileEntityService.getPublicUrl(user.avatar, 'avatar') ?? this.getIdenticonUrl(user.id); | ||||||
| 		} else if (user.avatarId) { | 		} else if (user.avatarId) { | ||||||
| 			const avatar = await this.driveFilesRepository.findOneByOrFail({ id: user.avatarId }); | 			const avatar = await this.driveFilesRepository.findOneByOrFail({ id: user.avatarId }); | ||||||
| 			return this.driveFileEntityService.getPublicUrl(avatar, true) ?? this.getIdenticonUrl(user.id); | 			return this.driveFileEntityService.getPublicUrl(avatar, 'avatar') ?? this.getIdenticonUrl(user.id); | ||||||
| 		} else { | 		} else { | ||||||
| 			return this.getIdenticonUrl(user.id); | 			return this.getIdenticonUrl(user.id); | ||||||
| 		} | 		} | ||||||
| @@ -326,7 +326,7 @@ export class UserEntityService implements OnModuleInit { | |||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public getAvatarUrlSync(user: User): string { | 	public getAvatarUrlSync(user: User): string { | ||||||
| 		if (user.avatar) { | 		if (user.avatar) { | ||||||
| 			return this.driveFileEntityService.getPublicUrl(user.avatar, true) ?? this.getIdenticonUrl(user.id); | 			return this.driveFileEntityService.getPublicUrl(user.avatar, 'avatar') ?? this.getIdenticonUrl(user.id); | ||||||
| 		} else { | 		} else { | ||||||
| 			return this.getIdenticonUrl(user.id); | 			return this.getIdenticonUrl(user.id); | ||||||
| 		} | 		} | ||||||
| @@ -415,6 +415,11 @@ export class UserEntityService implements OnModuleInit { | |||||||
| 			} : undefined) : undefined, | 			} : undefined) : undefined, | ||||||
| 			emojis: this.customEmojiService.populateEmojis(user.emojis, user.host), | 			emojis: this.customEmojiService.populateEmojis(user.emojis, user.host), | ||||||
| 			onlineStatus: this.getOnlineStatus(user), | 			onlineStatus: this.getOnlineStatus(user), | ||||||
|  | 			// パフォーマンス上の理由でローカルユーザーのみ | ||||||
|  | 			badgeRoles: user.host == null ? this.roleService.getUserBadgeRoles(user.id).then(rs => rs.map(r => ({ | ||||||
|  | 				name: r.name, | ||||||
|  | 				iconUrl: r.iconUrl, | ||||||
|  | 			}))) : undefined, | ||||||
|  |  | ||||||
| 			...(opts.detail ? { | 			...(opts.detail ? { | ||||||
| 				url: profile!.url, | 				url: profile!.url, | ||||||
| @@ -422,7 +427,7 @@ export class UserEntityService implements OnModuleInit { | |||||||
| 				createdAt: user.createdAt.toISOString(), | 				createdAt: user.createdAt.toISOString(), | ||||||
| 				updatedAt: user.updatedAt ? user.updatedAt.toISOString() : null, | 				updatedAt: user.updatedAt ? user.updatedAt.toISOString() : null, | ||||||
| 				lastFetchedAt: user.lastFetchedAt ? user.lastFetchedAt.toISOString() : null, | 				lastFetchedAt: user.lastFetchedAt ? user.lastFetchedAt.toISOString() : null, | ||||||
| 				bannerUrl: user.banner ? this.driveFileEntityService.getPublicUrl(user.banner, false) : null, | 				bannerUrl: user.banner ? this.driveFileEntityService.getPublicUrl(user.banner) : null, | ||||||
| 				bannerBlurhash: user.banner?.blurhash ?? null, | 				bannerBlurhash: user.banner?.blurhash ?? null, | ||||||
| 				isLocked: user.isLocked, | 				isLocked: user.isLocked, | ||||||
| 				isSilenced: this.roleService.getUserPolicies(user.id).then(r => !r.canPublicNote), | 				isSilenced: this.roleService.getUserPolicies(user.id).then(r => !r.canPublicNote), | ||||||
| @@ -454,6 +459,7 @@ export class UserEntityService implements OnModuleInit { | |||||||
| 					id: role.id, | 					id: role.id, | ||||||
| 					name: role.name, | 					name: role.name, | ||||||
| 					color: role.color, | 					color: role.color, | ||||||
|  | 					iconUrl: role.iconUrl, | ||||||
| 					description: role.description, | 					description: role.description, | ||||||
| 					isModerator: role.isModerator, | 					isModerator: role.isModerator, | ||||||
| 					isAdministrator: role.isAdministrator, | 					isAdministrator: role.isAdministrator, | ||||||
| @@ -489,7 +495,6 @@ export class UserEntityService implements OnModuleInit { | |||||||
| 				hasUnreadMessagingMessage: this.getHasUnreadMessagingMessage(user.id), | 				hasUnreadMessagingMessage: this.getHasUnreadMessagingMessage(user.id), | ||||||
| 				hasUnreadNotification: this.getHasUnreadNotification(user.id), | 				hasUnreadNotification: this.getHasUnreadNotification(user.id), | ||||||
| 				hasPendingReceivedFollowRequest: this.getHasPendingReceivedFollowRequest(user.id), | 				hasPendingReceivedFollowRequest: this.getHasPendingReceivedFollowRequest(user.id), | ||||||
| 				integrations: profile!.integrations, |  | ||||||
| 				mutedWords: profile!.mutedWords, | 				mutedWords: profile!.mutedWords, | ||||||
| 				mutedInstances: profile!.mutedInstances, | 				mutedInstances: profile!.mutedInstances, | ||||||
| 				mutingNotificationTypes: profile!.mutingNotificationTypes, | 				mutingNotificationTypes: profile!.mutingNotificationTypes, | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ | |||||||
|  * The getter will return a .bind version of the function |  * The getter will return a .bind version of the function | ||||||
|  * and memoize the result against a symbol on the instance |  * and memoize the result against a symbol on the instance | ||||||
|  */ |  */ | ||||||
| export function bindThis(target, key, descriptor) { | export function bindThis(target: any, key: string, descriptor: any) { | ||||||
| 	let fn = descriptor.value; | 	let fn = descriptor.value; | ||||||
|  |  | ||||||
| 	if (typeof fn !== 'function') { | 	if (typeof fn !== 'function') { | ||||||
| @@ -34,7 +34,7 @@ export function bindThis(target, key, descriptor) { | |||||||
| 			}); | 			}); | ||||||
| 			return boundFn; | 			return boundFn; | ||||||
| 		}, | 		}, | ||||||
| 		set(value) { | 		set(value: any) { | ||||||
| 			fn = value; | 			fn = value; | ||||||
| 		}, | 		}, | ||||||
| 	}; | 	}; | ||||||
|   | |||||||
| @@ -17,15 +17,13 @@ export default class Logger { | |||||||
| 	private context: Context; | 	private context: Context; | ||||||
| 	private parentLogger: Logger | null = null; | 	private parentLogger: Logger | null = null; | ||||||
| 	private store: boolean; | 	private store: boolean; | ||||||
| 	private syslogClient: any | null = null; |  | ||||||
|  |  | ||||||
| 	constructor(context: string, color?: KEYWORD, store = true, syslogClient = null) { | 	constructor(context: string, color?: KEYWORD, store = true) { | ||||||
| 		this.context = { | 		this.context = { | ||||||
| 			name: context, | 			name: context, | ||||||
| 			color: color, | 			color: color, | ||||||
| 		}; | 		}; | ||||||
| 		this.store = store; | 		this.store = store; | ||||||
| 		this.syslogClient = syslogClient; |  | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| @@ -47,7 +45,7 @@ export default class Logger { | |||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		const time = dateFormat(new Date(), 'HH:mm:ss'); | 		const time = dateFormat(new Date(), 'HH:mm:ss'); | ||||||
| 		const worker = cluster.isPrimary ? '*' : cluster.worker.id; | 		const worker = cluster.isPrimary ? '*' : cluster.worker!.id; | ||||||
| 		const l = | 		const l = | ||||||
| 			level === 'error' ? important ? chalk.bgRed.white('ERR ') : chalk.red('ERR ') : | 			level === 'error' ? important ? chalk.bgRed.white('ERR ') : chalk.red('ERR ') : | ||||||
| 			level === 'warning' ? chalk.yellow('WARN') : | 			level === 'warning' ? chalk.yellow('WARN') : | ||||||
| @@ -69,20 +67,6 @@ export default class Logger { | |||||||
|  |  | ||||||
| 		console.log(important ? chalk.bold(log) : log); | 		console.log(important ? chalk.bold(log) : log); | ||||||
| 		if (level === 'error' && data) console.log(data); | 		if (level === 'error' && data) console.log(data); | ||||||
|  |  | ||||||
| 		if (store) { |  | ||||||
| 			if (this.syslogClient) { |  | ||||||
| 				const send = |  | ||||||
| 					level === 'error' ? this.syslogClient.error : |  | ||||||
| 					level === 'warning' ? this.syslogClient.warning : |  | ||||||
| 					level === 'success' ? this.syslogClient.info : |  | ||||||
| 					level === 'debug' ? this.syslogClient.info : |  | ||||||
| 					level === 'info' ? this.syslogClient.info : |  | ||||||
| 					null as never; |  | ||||||
|  |  | ||||||
| 				send.bind(this.syslogClient)(message).catch(() => {}); |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
|   | |||||||
| @@ -1,5 +1,7 @@ | |||||||
| import { bindThis } from '@/decorators.js'; | import { bindThis } from '@/decorators.js'; | ||||||
|  |  | ||||||
|  | // TODO: メモリ節約のためあまり参照されないキーを定期的に削除できるようにする? | ||||||
|  |  | ||||||
| export class Cache<T> { | export class Cache<T> { | ||||||
| 	public cache: Map<string | null, { date: number; value: T; }>; | 	public cache: Map<string | null, { date: number; value: T; }>; | ||||||
| 	private lifetime: number; | 	private lifetime: number; | ||||||
|   | |||||||
| @@ -4,7 +4,7 @@ import { unique } from '@/misc/prelude/array.js'; | |||||||
| export function extractCustomEmojisFromMfm(nodes: mfm.MfmNode[]): string[] { | export function extractCustomEmojisFromMfm(nodes: mfm.MfmNode[]): string[] { | ||||||
| 	const emojiNodes = mfm.extract(nodes, (node) => { | 	const emojiNodes = mfm.extract(nodes, (node) => { | ||||||
| 		return (node.type === 'emojiCode' && node.props.name.length <= 100); | 		return (node.type === 'emojiCode' && node.props.name.length <= 100); | ||||||
| 	}); | 	}) as mfm.MfmEmojiCode[]; | ||||||
|  |  | ||||||
| 	return unique(emojiNodes.map(x => x.props.name)); | 	return unique(emojiNodes.map(x => x.props.name)); | ||||||
| } | } | ||||||
|   | |||||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user