Compare commits
	
		
			205 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | 40de14415c | ||
|   | a1f3bd6865 | ||
|   | c925e3d281 | ||
|   | 7c9330a02f | ||
|   | ca99468970 | ||
|   | 8a5c6240b4 | ||
|   | fca820c90c | ||
|   | becc4d2e54 | ||
|   | c645f9f99f | ||
|   | a5341cbd7d | ||
|   | 3dd363a6c5 | ||
|   | e630803922 | ||
|   | cb3b167d61 | ||
|   | 774b7fed1f | ||
|   | cbf526b043 | ||
|   | 48331bc851 | ||
|   | bda3d69539 | ||
|   | 71900e0231 | ||
|   | 694f08c79b | ||
|   | c328584bb6 | ||
|   | 4c01198811 | ||
|   | 7781497b42 | ||
|   | cde0eb621d | ||
|   | 474b8789a7 | ||
|   | 2189acdde1 | ||
|   | c174f23389 | ||
|   | 4e23500732 | ||
|   | b965f5e4a9 | ||
|   | d1a69abf81 | ||
|   | d2ef0efbff | ||
|   | ce5c78d0d2 | ||
|   | 28bea88da0 | ||
|   | a2e6f459e7 | ||
|   | 0026c45fe0 | ||
|   | 424292f335 | ||
|   | 8ca2f24df6 | ||
|   | fb7e5a3fac | ||
|   | 5dd24e44d1 | ||
|   | f7c6ea93d7 | ||
|   | 7658351041 | ||
|   | 833e2869e7 | ||
|   | 5b3a07ee9e | ||
|   | 3dfe3aa9a4 | ||
|   | f68e13d905 | ||
|   | 3157d81e95 | ||
|   | bd13ea3d2c | ||
|   | f0cb587c89 | ||
|   | 84e2ee220b | ||
|   | a7977c6642 | ||
|   | 1dfcd45704 | ||
|   | 8013cd2e79 | ||
|   | 9c69501404 | ||
|   | 0c8d874e3a | ||
|   | 0fb9c372dd | ||
|   | 3bb7afe544 | ||
|   | 29399e1ddc | ||
|   | dfc1410bb0 | ||
|   | 9b72e02da3 | ||
|   | 5ec07ede7c | ||
|   | 72d4ad4c45 | ||
|   | a6fb6150a3 | ||
|   | b9ee14fe5b | ||
|   | 6e68a78d6a | ||
|   | 870f7608be | ||
|   | 69869307bf | ||
|   | b161f38710 | ||
|   | a7f464147d | ||
|   | 8eb87c8e4d | ||
|   | 7925b130e8 | ||
|   | 16e3cb01ca | ||
|   | 543ba2b3b7 | ||
|   | beb9cd5710 | ||
|   | b5fa8767da | ||
|   | f846b207b6 | ||
|   | c6b07acdcc | ||
|   | b055f516c0 | ||
|   | 716ffcace6 | ||
|   | 980bf1306e | ||
|   | ea92254b73 | ||
|   | 16ba1b3708 | ||
|   | 47b6f466ec | ||
|   | 2e76fcdf6f | ||
|   | 7ce0f79f7f | ||
|   | 0c59dd3da7 | ||
|   | 2aa73fdf6c | ||
|   | cd5615d354 | ||
|   | 8c883653c9 | ||
|   | 36170a11f5 | ||
|   | 0f546b47d1 | ||
|   | 60df819c60 | ||
|   | 0e1b5d6f14 | ||
|   | bde22208fe | ||
|   | d4eb1def61 | ||
|   | 14cff15c89 | ||
|   | e8c5307f66 | ||
|   | dd52be3a01 | ||
|   | 8f9ce23e52 | ||
|   | 7c5fc2c423 | ||
|   | 63df2c851e | ||
|   | 4db787c4ee | ||
|   | 839a626716 | ||
|   | ef7ad05c0b | ||
|   | e24b0ceb80 | ||
|   | 71c42bef9b | ||
|   | 8caf288ac1 | ||
|   | 8f2049bcd2 | ||
|   | d0aba46ee3 | ||
|   | 57c94a5cf0 | ||
|   | c1f1e0ee7c | ||
|   | 192ea9738d | ||
|   | 37b849ad1f | ||
|   | 4e68126c06 | ||
|   | c7fbf5637f | ||
|   | 5cf5b66696 | ||
|   | 7436a58ea1 | ||
|   | b7b8fd4b59 | ||
|   | 55d4d3418e | ||
|   | 1c8419cea0 | ||
|   | e8d4f3eac3 | ||
|   | 1b21bad202 | ||
|   | 30f600e03e | ||
|   | f34f9f6ea5 | ||
|   | 8d4c5deb8d | ||
|   | 2f41f12aea | ||
|   | 9f0e0dc8ce | ||
|   | a71682f6f0 | ||
|   | 5d3d5cd59c | ||
|   | 451bc0b444 | ||
|   | d7a2d59f41 | ||
|   | 7deb4691fb | ||
|   | 9965bc8f94 | ||
|   | 317770fb23 | ||
|   | 19c204ea03 | ||
|   | b9feacab85 | ||
|   | 56c7359a0c | ||
|   | a7c1afffc6 | ||
|   | fdb745b4a8 | ||
|   | b3d8134c7a | ||
|   | 0879ab50b8 | ||
|   | 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 | 
| @@ -131,11 +131,20 @@ proxyBypassHosts: | |||||||
|  |  | ||||||
| # Media Proxy | # Media Proxy | ||||||
| # Reference Implementation: https://github.com/misskey-dev/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": "/workspace", | ||||||
|  | 	"features": { | ||||||
|  | 		"ghcr.io/devcontainers-contrib/features/pnpm:2": {} | ||||||
|  | 	}, | ||||||
|  | 	"forwardPorts": [3000], | ||||||
|  | 	"postCreateCommand": "sudo chmod 755 .devcontainer/init.sh && .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 | ||||||
							
								
								
									
										53
									
								
								.devcontainer/docker-compose.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										53
									
								
								.devcontainer/docker-compose.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,53 @@ | |||||||
|  | version: '3.8' | ||||||
|  |  | ||||||
|  | services: | ||||||
|  |   app: | ||||||
|  |     build:  | ||||||
|  |       context: . | ||||||
|  |       dockerfile: Dockerfile | ||||||
|  |  | ||||||
|  |     volumes: | ||||||
|  |       - ../:/workspace:cached | ||||||
|  |  | ||||||
|  |     command: sleep infinity | ||||||
|  |  | ||||||
|  |     networks: | ||||||
|  |       - internal_network | ||||||
|  |       - external_network | ||||||
|  |  | ||||||
|  |   redis: | ||||||
|  |     restart: always | ||||||
|  |     image: redis:7-alpine | ||||||
|  |     networks: | ||||||
|  |       - internal_network | ||||||
|  |     volumes: | ||||||
|  |       - redis-data:/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: | ||||||
|  |       - postgres-data:/var/lib/postgresql/data | ||||||
|  |     healthcheck: | ||||||
|  |       test: "pg_isready -U $$POSTGRES_USER -d $$POSTGRES_DB" | ||||||
|  |       interval: 5s | ||||||
|  |       retries: 20 | ||||||
|  |  | ||||||
|  | volumes: | ||||||
|  |   postgres-data: | ||||||
|  |   redis-data: | ||||||
|  |  | ||||||
|  | networks: | ||||||
|  |   internal_network: | ||||||
|  |     internal: true | ||||||
|  |   external_network: | ||||||
							
								
								
									
										10
									
								
								.devcontainer/init.sh
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										10
									
								
								.devcontainer/init.sh
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,10 @@ | |||||||
|  | #!/bin/bash | ||||||
|  |  | ||||||
|  | set -xe | ||||||
|  |  | ||||||
|  | sudo chown -R node /workspace | ||||||
|  | git submodule update --init | ||||||
|  | pnpm install --frozen-lockfile | ||||||
|  | cp .devcontainer/devcontainer.yml .config/default.yml | ||||||
|  | pnpm build | ||||||
|  | pnpm migrate | ||||||
| @@ -5,6 +5,7 @@ indent_style = tab | |||||||
| indent_size = 2 | indent_size = 2 | ||||||
| charset = utf-8 | charset = utf-8 | ||||||
| insert_final_newline = true | insert_final_newline = true | ||||||
|  | end_of_line = lf | ||||||
|  |  | ||||||
| [*.yml] | [*.yml] | ||||||
| indent_style = space | indent_style = space | ||||||
|   | |||||||
							
								
								
									
										1
									
								
								.gitattributes
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.gitattributes
									
									
									
									
										vendored
									
									
								
							| @@ -5,3 +5,4 @@ | |||||||
| *.glb -diff -text | *.glb -diff -text | ||||||
| *.blend -diff -text | *.blend -diff -text | ||||||
| *.afdesign -diff -text | *.afdesign -diff -text | ||||||
|  | * text=auto eol=lf | ||||||
|   | |||||||
							
								
								
									
										36
									
								
								.github/workflows/check_copyright_year.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										36
									
								
								.github/workflows/check_copyright_year.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,18 +1,18 @@ | |||||||
| name: Check copyright year | name: Check copyright year | ||||||
|  |  | ||||||
| on: | on: | ||||||
|   push: |   push: | ||||||
|     branches: |     branches: | ||||||
|       - master |       - master | ||||||
|       - develop |       - develop | ||||||
|  |  | ||||||
| jobs: | jobs: | ||||||
|   check_copyright_year: |   check_copyright_year: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|     - uses: actions/checkout@v3.2.0 |     - uses: actions/checkout@v3.2.0 | ||||||
|     - run: | |     - run: | | ||||||
|         if [ "$(grep Copyright COPYING | sed -e 's/.*2014-\([0-9]*\) .*/\1/g')" -ne "$(date +%Y)" ]; then |         if [ "$(grep Copyright COPYING | sed -e 's/.*2014-\([0-9]*\) .*/\1/g')" -ne "$(date +%Y)" ]; then | ||||||
|           echo "Please change copyright year!" |           echo "Please change copyright year!" | ||||||
|           exit 1 |           exit 1 | ||||||
|         fi |         fi | ||||||
|   | |||||||
							
								
								
									
										8
									
								
								.github/workflows/docker-develop.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										8
									
								
								.github/workflows/docker-develop.yml
									
									
									
									
										vendored
									
									
								
							| @@ -15,7 +15,10 @@ jobs: | |||||||
|       - 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 |       - name: Set up Docker Buildx | ||||||
|  |         id: buildx | ||||||
|         uses: docker/setup-buildx-action@v2.3.0 |         uses: docker/setup-buildx-action@v2.3.0 | ||||||
|  |         with: | ||||||
|  |           platforms: linux/amd64,linux/arm64 | ||||||
|       - name: Docker meta |       - name: Docker meta | ||||||
|         id: meta |         id: meta | ||||||
|         uses: docker/metadata-action@v4 |         uses: docker/metadata-action@v4 | ||||||
| @@ -27,10 +30,13 @@ jobs: | |||||||
|           username: ${{ secrets.DOCKER_USERNAME }} |           username: ${{ secrets.DOCKER_USERNAME }} | ||||||
|           password: ${{ secrets.DOCKER_PASSWORD }} |           password: ${{ secrets.DOCKER_PASSWORD }} | ||||||
|       - name: Build and Push to Docker Hub |       - name: Build and Push to Docker Hub | ||||||
|         uses: docker/build-push-action@v3 |         uses: docker/build-push-action@v4 | ||||||
|         with: |         with: | ||||||
|  |           builder: ${{ steps.buildx.outputs.name }} | ||||||
|           context: . |           context: . | ||||||
|           push: true |           push: true | ||||||
|  |           platforms: ${{ steps.buildx.outputs.platforms }} | ||||||
|  |           provenance: false | ||||||
|           tags: misskey/misskey:develop |           tags: misskey/misskey:develop | ||||||
|           labels: develop |           labels: develop | ||||||
|           cache-from: type=gha |           cache-from: type=gha | ||||||
|   | |||||||
							
								
								
									
										12
									
								
								.github/workflows/docker.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										12
									
								
								.github/workflows/docker.yml
									
									
									
									
										vendored
									
									
								
							| @@ -13,6 +13,11 @@ 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 | ||||||
|  |         id: buildx | ||||||
|  |         uses: docker/setup-buildx-action@v2.3.0 | ||||||
|  |         with: | ||||||
|  |           platforms: linux/amd64,linux/arm64 | ||||||
|       - name: Docker meta |       - name: Docker meta | ||||||
|         id: meta |         id: meta | ||||||
|         uses: docker/metadata-action@v4 |         uses: docker/metadata-action@v4 | ||||||
| @@ -31,9 +36,14 @@ jobs: | |||||||
|           username: ${{ secrets.DOCKER_USERNAME }} |           username: ${{ secrets.DOCKER_USERNAME }} | ||||||
|           password: ${{ secrets.DOCKER_PASSWORD }} |           password: ${{ secrets.DOCKER_PASSWORD }} | ||||||
|       - name: Build and Push to Docker Hub |       - name: Build and Push to Docker Hub | ||||||
|         uses: docker/build-push-action@v3 |         uses: docker/build-push-action@v4 | ||||||
|         with: |         with: | ||||||
|  |           builder: ${{ steps.buildx.outputs.name }} | ||||||
|           context: . |           context: . | ||||||
|           push: true |           push: true | ||||||
|  |           platforms: ${{ steps.buildx.outputs.platforms }} | ||||||
|  |           provenance: false | ||||||
|           tags: ${{ steps.meta.outputs.tags }} |           tags: ${{ steps.meta.outputs.tags }} | ||||||
|           labels: ${{ steps.meta.outputs.labels }} |           labels: ${{ steps.meta.outputs.labels }} | ||||||
|  |           cache-from: type=gha | ||||||
|  |           cache-to: type=gha,mode=max | ||||||
|   | |||||||
							
								
								
									
										133
									
								
								.github/workflows/lint.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										133
									
								
								.github/workflows/lint.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,54 +1,79 @@ | |||||||
| name: Lint | name: Lint | ||||||
|  |  | ||||||
| on: | on: | ||||||
|   push: |   push: | ||||||
|     branches: |     branches: | ||||||
|       - master |       - master | ||||||
|       - develop |       - develop | ||||||
|   pull_request: |   pull_request: | ||||||
|  |  | ||||||
| jobs: | jobs: | ||||||
|   pnpm_install: |   pnpm_install: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|     - uses: actions/checkout@v3.3.0 |     - uses: actions/checkout@v3.3.0 | ||||||
|       with: |       with: | ||||||
|         fetch-depth: 0 |         fetch-depth: 0 | ||||||
|         submodules: true |         submodules: true | ||||||
|     - uses: pnpm/action-setup@v2 |     - uses: pnpm/action-setup@v2 | ||||||
|       with: |       with: | ||||||
|         version: 7 |         version: 7 | ||||||
|         run_install: false |         run_install: false | ||||||
|     - uses: actions/setup-node@v3.6.0 |     - uses: actions/setup-node@v3.6.0 | ||||||
|       with: |       with: | ||||||
|         node-version: 18.x |         node-version: 18.x | ||||||
|         cache: 'pnpm' |         cache: 'pnpm' | ||||||
|     - run: corepack enable |     - run: corepack enable | ||||||
|     - run: pnpm i --frozen-lockfile |     - run: pnpm i --frozen-lockfile | ||||||
|  |  | ||||||
|   lint: |   lint: | ||||||
|     needs: [pnpm_install] |     needs: [pnpm_install] | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     continue-on-error: true |     continue-on-error: true | ||||||
|     strategy: |     strategy: | ||||||
|       matrix: |       matrix: | ||||||
|         workspace: |         workspace: | ||||||
|         - backend |         - backend | ||||||
|         - frontend |         - frontend | ||||||
|         - sw |         - sw | ||||||
|     steps: |     steps: | ||||||
|     - uses: actions/checkout@v3.3.0 |     - uses: actions/checkout@v3.3.0 | ||||||
|       with: |       with: | ||||||
|         fetch-depth: 0 |         fetch-depth: 0 | ||||||
|         submodules: true |         submodules: true | ||||||
|     - uses: pnpm/action-setup@v2 |     - uses: pnpm/action-setup@v2 | ||||||
|       with: |       with: | ||||||
|         version: 7 |         version: 7 | ||||||
|         run_install: false |         run_install: false | ||||||
|     - uses: actions/setup-node@v3.6.0 |     - uses: actions/setup-node@v3.6.0 | ||||||
|       with: |       with: | ||||||
|         node-version: 18.x |         node-version: 18.x | ||||||
|         cache: 'pnpm' |         cache: 'pnpm' | ||||||
|     - run: corepack enable |     - run: corepack enable | ||||||
|     - run: pnpm i --frozen-lockfile |     - run: pnpm i --frozen-lockfile | ||||||
|     - run: pnpm --filter ${{ matrix.workspace }} run lint |     - run: pnpm --filter ${{ matrix.workspace }} run eslint | ||||||
|  |  | ||||||
|  |   typecheck: | ||||||
|  |     needs: [pnpm_install] | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     continue-on-error: true | ||||||
|  |     strategy: | ||||||
|  |       matrix: | ||||||
|  |         workspace: | ||||||
|  |         - backend | ||||||
|  |     steps: | ||||||
|  |     - uses: actions/checkout@v3.3.0 | ||||||
|  |       with: | ||||||
|  |         fetch-depth: 0 | ||||||
|  |         submodules: true | ||||||
|  |     - uses: pnpm/action-setup@v2 | ||||||
|  |       with: | ||||||
|  |         version: 7 | ||||||
|  |         run_install: false | ||||||
|  |     - uses: actions/setup-node@v3.6.0 | ||||||
|  |       with: | ||||||
|  |         node-version: 18.x | ||||||
|  |         cache: 'pnpm' | ||||||
|  |     - run: corepack enable | ||||||
|  |     - run: pnpm i --frozen-lockfile | ||||||
|  |     - run: pnpm --filter ${{ matrix.workspace }} run typecheck | ||||||
|   | |||||||
							
								
								
									
										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 | ||||||
|   | |||||||
							
								
								
									
										3
									
								
								.vscode/settings.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								.vscode/settings.json
									
									
									
									
										vendored
									
									
								
							| @@ -1,5 +1,6 @@ | |||||||
| { | { | ||||||
|     "search.exclude": { |     "search.exclude": { | ||||||
|         "**/node_modules": true |         "**/node_modules": true | ||||||
|     } |     }, | ||||||
|  |     "typescript.tsdk": "node_modules/typescript/lib" | ||||||
| } | } | ||||||
							
								
								
									
										103
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						
									
										103
									
								
								CHANGELOG.md
									
									
									
									
									
								
							| @@ -8,6 +8,109 @@ | |||||||
|  |  | ||||||
| You should also include the user name that made the change. | You should also include the user name that made the change. | ||||||
| --> | --> | ||||||
|  | ## 13.7.4 (unreleased) | ||||||
|  |  | ||||||
|  | ### Note | ||||||
|  | 13.7.0以前から直接このバージョンにアップデートする場合は全ての通知が削除**されません。** | ||||||
|  |  | ||||||
|  | ### Bugfixes | ||||||
|  | - 全ての通知が削除されてしまうのを修正 | ||||||
|  |  | ||||||
|  | ## 13.7.3 (2023/02/23) | ||||||
|  |  | ||||||
|  | ### Note | ||||||
|  | ~~13.7.0以前から直接このバージョンにアップデートする場合は全ての通知が削除**されません。**~~ | ||||||
|  |  | ||||||
|  | ### Improvements | ||||||
|  |  | ||||||
|  | ### Bugfixes | ||||||
|  | - Client: 「キャッシュを削除」した後、ローカルのカスタム絵文字が表示されなくなるされなくなる問題を修正 | ||||||
|  | - Client: 通知設定画面で以前からグループの招待を有効化していた場合、通知の表示に失敗する問題の修正 | ||||||
|  | - Client: 通知設定画面に古いトグルが残っていた問題を修正 | ||||||
|  |  | ||||||
|  | ## 13.7.2 (2023/02/23) | ||||||
|  |  | ||||||
|  | ### Note | ||||||
|  | 13.7.0以前からアップデートする場合は全ての通知が削除されます。 | ||||||
|  |  | ||||||
|  | ### Improvements | ||||||
|  | - enhance: make pwa icon maskable | ||||||
|  | - chore(client): tweak custom emoji size | ||||||
|  |  | ||||||
|  | ### Bugfixes | ||||||
|  | - マイグレーションが失敗することがあるのを修正 | ||||||
|  |  | ||||||
|  | ## 13.7.1 (2023/02/23) | ||||||
|  |  | ||||||
|  | ### Improvements | ||||||
|  | - pnpm buildではswcを使うように | ||||||
|  |  | ||||||
|  | ### Bugfixes | ||||||
|  | - NODE_ENV=productionでビルドできないのを修正 | ||||||
|  |  | ||||||
|  | ## 13.7.0 (2023/02/22) | ||||||
|  |  | ||||||
|  | ### Changes | ||||||
|  | - チャット機能が削除されました | ||||||
|  |  | ||||||
|  | ### Improvements | ||||||
|  | - Server: URLプレビュー(summaly)はプロキシを通すように | ||||||
|  | - Client: 2FA設定のUIをまともにした | ||||||
|  | - セキュリティキーの名前を変更できるように | ||||||
|  | - enhance(client): add quiz preset for play | ||||||
|  | - 広告開始時期を設定できるように | ||||||
|  | - みつけるで公開ロール一覧とそのメンバーを閲覧できるように | ||||||
|  | - enhance(client): MFMのx3, x4が含まれていたらノートをたたむように | ||||||
|  | - enhance(client): make possible to reload page of window | ||||||
|  |  | ||||||
|  | ### Bugfixes | ||||||
|  | - ユーザー検索ダイアログでローカルユーザーを絞って検索できない問題を修正 | ||||||
|  | - fix(client): MkHeader及びデッキのカラムでチャンネル一覧を選択したとき、最大5個までしか表示されない | ||||||
|  | - 管理画面の広告を10個以上見えるように | ||||||
|  | - Moderation note が保存できない | ||||||
|  | - ユーザーのハッシュタグ検索が機能していないのを修正 | ||||||
|  |  | ||||||
|  | ## 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) | ## 13.5.4 (2023/02/09) | ||||||
|  |  | ||||||
| ### Improvements | ### Improvements | ||||||
|   | |||||||
| @@ -83,7 +83,7 @@ An actual domain will be assigned so you can test the federation. | |||||||
| 	- The title must be in the format `Release: x.y.z`. | 	- The title must be in the format `Release: x.y.z`. | ||||||
| 		- `x.y.z` is the new version you are trying to release. | 		- `x.y.z` is the new version you are trying to release. | ||||||
| 3. Deploy and perform a simple QA check. Also verify that the tests passed. | 3. Deploy and perform a simple QA check. Also verify that the tests passed. | ||||||
| 4. Merge it. | 4. Merge it. (Do not squash commit) | ||||||
| 5. Create a [release of GitHub](https://github.com/misskey-dev/misskey/releases) | 5. Create a [release of GitHub](https://github.com/misskey-dev/misskey/releases) | ||||||
| 	- The target branch must be `master` | 	- The target branch must be `master` | ||||||
| 	- The tag name must be the version | 	- The tag name must be the version | ||||||
| @@ -111,6 +111,26 @@ 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.   | ||||||
|  | **Note:** If you are using Windows, please clone the repository with WSL. Using Git for Windows will result in broken files due to the difference in how newlines are handled. | ||||||
|  |  | ||||||
|  | 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 [`/packages/backend/test`](/packages/backend/test). | - Test codes are located in [`/packages/backend/test`](/packages/backend/test). | ||||||
|  |  | ||||||
| @@ -258,9 +278,10 @@ SQLでは配列のインデックスは**1始まり**。 | |||||||
| ### null IN | ### null IN | ||||||
| nullが含まれる可能性のあるカラムにINするときは、そのままだとおかしくなるのでORなどでnullのハンドリングをしよう。 | nullが含まれる可能性のあるカラムにINするときは、そのままだとおかしくなるのでORなどでnullのハンドリングをしよう。 | ||||||
|  |  | ||||||
| ### `undefined`にご用心 | ### enumの削除は気をつける | ||||||
| MongoDBの時とは違い、findOneでレコードを取得する時に対象レコードが存在しない場合 **`undefined`** が返ってくるので注意。 | enumの列挙の内容の削除は、その値をもつレコードを全て削除しないといけない | ||||||
| MongoDBは`null`で返してきてたので、その感覚で`if (x === null)`とか書くとバグる。代わりに`if (x == null)`と書いてください |  | ||||||
|  | 削除が重たかったり不可能だったりする場合は、削除しないでおく | ||||||
|  |  | ||||||
| ### Migration作成方法 | ### Migration作成方法 | ||||||
| packages/backendで: | packages/backendで: | ||||||
|   | |||||||
							
								
								
									
										14
									
								
								Dockerfile
									
									
									
									
									
								
							
							
						
						
									
										14
									
								
								Dockerfile
									
									
									
									
									
								
							| @@ -1,3 +1,5 @@ | |||||||
|  | # syntax = docker/dockerfile:1.4 | ||||||
|  |  | ||||||
| ARG NODE_VERSION=18.13.0-bullseye | ARG NODE_VERSION=18.13.0-bullseye | ||||||
|  |  | ||||||
| FROM node:${NODE_VERSION} AS builder | FROM node:${NODE_VERSION} AS builder | ||||||
| @@ -14,16 +16,16 @@ RUN corepack enable | |||||||
|  |  | ||||||
| WORKDIR /misskey | WORKDIR /misskey | ||||||
|  |  | ||||||
| COPY ["pnpm-lock.yaml", "pnpm-workspace.yaml", "package.json", "./"] | COPY --link ["pnpm-lock.yaml", "pnpm-workspace.yaml", "package.json", "./"] | ||||||
| COPY ["scripts", "./scripts"] | COPY --link ["scripts", "./scripts"] | ||||||
| COPY ["packages/backend/package.json", "./packages/backend/"] | COPY --link ["packages/backend/package.json", "./packages/backend/"] | ||||||
| COPY ["packages/frontend/package.json", "./packages/frontend/"] | COPY --link ["packages/frontend/package.json", "./packages/frontend/"] | ||||||
| COPY ["packages/sw/package.json", "./packages/sw/"] | COPY --link ["packages/sw/package.json", "./packages/sw/"] | ||||||
|  |  | ||||||
| RUN --mount=type=cache,target=/root/.local/share/pnpm/store,sharing=locked \ | RUN --mount=type=cache,target=/root/.local/share/pnpm/store,sharing=locked \ | ||||||
| 	pnpm i --frozen-lockfile --aggregate-output | 	pnpm i --frozen-lockfile --aggregate-output | ||||||
|  |  | ||||||
| COPY . ./ | COPY --link . ./ | ||||||
|  |  | ||||||
| ARG NODE_ENV=production | ARG NODE_ENV=production | ||||||
|  |  | ||||||
|   | |||||||
| @@ -5,9 +5,9 @@ Also, the later tasks are more indefinite and are subject to change as developme | |||||||
| ## (1) Improve maintainability \<current phase\> | ## (1) Improve maintainability \<current phase\> | ||||||
| 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)~~ → Done ✔️ | ||||||
| - Improve CI | - Improve CI | ||||||
| 	- Fix tests | 	- ~~Fix tests~~ → Done ✔️ | ||||||
| 	- 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~~ → Done ✔️ | 		- ~~May need to implement a mechanism that allows for DI~~ → Done ✔️ | ||||||
|   | |||||||
| @@ -1,3 +1,4 @@ | |||||||
| apiVersion: v2 | apiVersion: v2 | ||||||
| name: misskey | name: misskey | ||||||
| version: 0.0.0 | version: 0.0.0 | ||||||
|  | description: This chart is created for the purpose of previewing Pull Requests. Do not use this for production use. | ||||||
|   | |||||||
							
								
								
									
										5
									
								
								codecov.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								codecov.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | |||||||
|  | coverage: | ||||||
|  |   status: | ||||||
|  |     project: | ||||||
|  |       default: | ||||||
|  |         only_pulls: true | ||||||
| @@ -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: "أظهر للمستخدمين البِعاد" | ||||||
|   | |||||||
| @@ -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: "রিমোট ব্যাবহারকারীদের জন্য দেখান" | ||||||
|   | |||||||
| @@ -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: | ||||||
|   | |||||||
| @@ -103,6 +103,8 @@ renoted: "Renote getätigt." | |||||||
| cantRenote: "Renote dieses Beitrags nicht möglich." | cantRenote: "Renote dieses Beitrags nicht möglich." | ||||||
| cantReRenote: "Renote einer Renote nicht möglich." | cantReRenote: "Renote einer Renote nicht möglich." | ||||||
| quote: "Zitieren" | quote: "Zitieren" | ||||||
|  | inChannelRenote: "Kanal-interner Renote" | ||||||
|  | inChannelQuote: "Kanal-internes Zitat" | ||||||
| pinnedNote: "Angeheftete Notiz" | pinnedNote: "Angeheftete Notiz" | ||||||
| pinned: "Angeheftet" | pinned: "Angeheftet" | ||||||
| you: "Du" | you: "Du" | ||||||
| @@ -129,6 +131,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 +259,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 +469,8 @@ 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" | enableAdvancedMfm: "Erweitertes MFM aktivieren" | ||||||
|  | enableAnimatedMfm: "Animiertes MFM aktivieren" | ||||||
| doing: "In Bearbeitung …" | doing: "In Bearbeitung …" | ||||||
| category: "Kategorie" | category: "Kategorie" | ||||||
| tags: "Schlagwörter" | tags: "Schlagwörter" | ||||||
| @@ -861,6 +867,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 +947,16 @@ 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." | ||||||
|  | thisPostMayBeAnnoying: "Dieser Beitrag stört eventuell andere Benutzer." | ||||||
|  | thisPostMayBeAnnoyingHome: "Zur Startseite schicken" | ||||||
|  | thisPostMayBeAnnoyingCancel: "Abbrechen" | ||||||
|  | thisPostMayBeAnnoyingIgnore: "Trotzdem schicken" | ||||||
|  | collapseRenotes: "Bereits gesehene Renotes verkürzt anzeigen" | ||||||
|  | internalServerError: "Serverinterner Fehler" | ||||||
|  | internalServerErrorDescription: "Im Server ist ein unerwarteter Fehler aufgetreten." | ||||||
|  | copyErrorInfo: "Fehlerdetails kopieren" | ||||||
| _achievements: | _achievements: | ||||||
|   earnedAt: "Freigeschaltet am" |   earnedAt: "Freigeschaltet am" | ||||||
|   _types: |   _types: | ||||||
| @@ -1323,72 +1341,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" | ||||||
| @@ -1593,12 +1545,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" | ||||||
| @@ -1869,6 +1824,6 @@ _deck: | |||||||
|     tl: "Chronik" |     tl: "Chronik" | ||||||
|     antenna: "Antennen" |     antenna: "Antennen" | ||||||
|     list: "Listen" |     list: "Listen" | ||||||
|     channel: "Kanäle" |     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: | ||||||
|   | |||||||
| @@ -103,6 +103,8 @@ renoted: "Renoted." | |||||||
| cantRenote: "This post can't be renoted." | cantRenote: "This post can't be renoted." | ||||||
| cantReRenote: "A renote can't be renoted." | cantReRenote: "A renote can't be renoted." | ||||||
| quote: "Quote" | quote: "Quote" | ||||||
|  | inChannelRenote: "Channel-only Renote" | ||||||
|  | inChannelQuote: "Channel-only Quote" | ||||||
| pinnedNote: "Pinned note" | pinnedNote: "Pinned note" | ||||||
| pinned: "Pin to profile" | pinned: "Pin to profile" | ||||||
| you: "You" | you: "You" | ||||||
| @@ -129,6 +131,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 +259,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 +469,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 animated MFM" | ||||||
| doing: "Processing..." | doing: "Processing..." | ||||||
| category: "Category" | category: "Category" | ||||||
| tags: "Tags" | tags: "Tags" | ||||||
| @@ -861,6 +867,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 +947,16 @@ 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" | ||||||
|  | collapseRenotes: "Collapse renotes you've already seen" | ||||||
|  | internalServerError: "Internal Server Error" | ||||||
|  | internalServerErrorDescription: "The server has run into an unexpected error." | ||||||
|  | copyErrorInfo: "Copy error details" | ||||||
| _achievements: | _achievements: | ||||||
|   earnedAt: "Unlocked at" |   earnedAt: "Unlocked at" | ||||||
|   _types: |   _types: | ||||||
| @@ -1323,72 +1341,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" | ||||||
| @@ -1593,12 +1545,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" | ||||||
| @@ -1869,6 +1824,6 @@ _deck: | |||||||
|     tl: "Timeline" |     tl: "Timeline" | ||||||
|     antenna: "Antennas" |     antenna: "Antennas" | ||||||
|     list: "List" |     list: "List" | ||||||
|     channel: "Channels" |     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" | ||||||
| @@ -257,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" | ||||||
| @@ -465,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" | ||||||
| @@ -940,6 +941,8 @@ cannotPerformTemporaryDescription: "Esta acción no se puede realizar porque se | |||||||
| preset: "Predefinido" | preset: "Predefinido" | ||||||
| selectFromPresets: "Escoger desde predefinidos" | selectFromPresets: "Escoger desde predefinidos" | ||||||
| achievements: "Logros" | 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: | _achievements: | ||||||
|   earnedAt: "Desbloqueado el" |   earnedAt: "Desbloqueado el" | ||||||
|   _types: |   _types: | ||||||
| @@ -1324,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" | ||||||
| @@ -1594,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" | ||||||
|   | |||||||
| @@ -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" | ||||||
|   | |||||||
| @@ -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" | ||||||
|   | |||||||
| @@ -84,12 +84,12 @@ error: "Errore" | |||||||
| somethingHappened: "Si è verificato un problema" | somethingHappened: "Si è verificato un problema" | ||||||
| retry: "Riprova" | retry: "Riprova" | ||||||
| pageLoadError: "Caricamento pagina non riuscito. " | pageLoadError: "Caricamento pagina non riuscito. " | ||||||
| pageLoadErrorDescription: "Questo viene normalmente causato dalla rete o dalla cache del browser. Si prega di pulire la cache, o di attendere e riprovare più tardi." | pageLoadErrorDescription: "Questo problema viene normalmente causato da errori di rete o dalla cache del browser. Si prega di pulire la cache, o di attendere e riprovare più tardi." | ||||||
| serverIsDead: "Il server non risponde. Si prega di attendere e riprovare più tardi." | serverIsDead: "Il server non risponde. Si prega di attendere e riprovare più tardi." | ||||||
| youShouldUpgradeClient: "Per visualizzare la pagina è necessario aggiornare il client alla nuova versione e ricaricare." | youShouldUpgradeClient: "Per visualizzare la pagina è necessario aggiornare il client alla nuova versione e ricaricare." | ||||||
| enterListName: "Nome della lista" | enterListName: "Nome della lista" | ||||||
| privacy: "Privacy" | privacy: "Privacy" | ||||||
| makeFollowManuallyApprove: "Richiedi di approvare i follower manualmente" | makeFollowManuallyApprove: "Approva i follower manualmente" | ||||||
| defaultNoteVisibility: "Privacy predefinita delle note" | defaultNoteVisibility: "Privacy predefinita delle note" | ||||||
| follow: "Segui" | follow: "Segui" | ||||||
| followRequest: "Richiesta di follow" | followRequest: "Richiesta di follow" | ||||||
| @@ -103,6 +103,8 @@ renoted: "Rinotato!" | |||||||
| cantRenote: "È impossibile rinotare questa nota." | cantRenote: "È impossibile rinotare questa nota." | ||||||
| cantReRenote: "È impossibile rinotare una Rinota." | cantReRenote: "È impossibile rinotare una Rinota." | ||||||
| quote: "Cita" | quote: "Cita" | ||||||
|  | inChannelRenote: "Rinota nel canale" | ||||||
|  | inChannelQuote: "Cita nel canale" | ||||||
| pinnedNote: "Nota fissata" | pinnedNote: "Nota fissata" | ||||||
| pinned: "Fissa sul profilo" | pinned: "Fissa sul profilo" | ||||||
| you: "Tu" | you: "Tu" | ||||||
| @@ -129,6 +131,7 @@ unblockConfirm: "Vuoi davvero sbloccare il profilo?" | |||||||
| suspendConfirm: "Vuoi sospendere questo profilo?" | suspendConfirm: "Vuoi sospendere questo profilo?" | ||||||
| unsuspendConfirm: "Vuoi revocare la sospensione si questo profilo?" | unsuspendConfirm: "Vuoi revocare la sospensione si questo profilo?" | ||||||
| selectList: "Seleziona una lista" | selectList: "Seleziona una lista" | ||||||
|  | selectChannel: "Seleziona canale" | ||||||
| selectAntenna: "Scegli un'antenna" | selectAntenna: "Scegli un'antenna" | ||||||
| selectWidget: "Seleziona il riquadro" | selectWidget: "Seleziona il riquadro" | ||||||
| editWidgets: "Modifica i riquadri" | editWidgets: "Modifica i riquadri" | ||||||
| @@ -256,6 +259,8 @@ noMoreHistory: "Non c'è più cronologia da visualizzare" | |||||||
| startMessaging: "Nuovo messaggio" | startMessaging: "Nuovo messaggio" | ||||||
| nUsersRead: "Letto da {n} persone" | nUsersRead: "Letto da {n} persone" | ||||||
| agreeTo: "Sono d'accordo con {0}" | agreeTo: "Sono d'accordo con {0}" | ||||||
|  | agreeBelow: "Accetto quanto riportato sotto" | ||||||
|  | basicNotesBeforeCreateAccount: "Note importanti" | ||||||
| tos: "Termini di servizio" | tos: "Termini di servizio" | ||||||
| start: "Inizia!" | start: "Inizia!" | ||||||
| home: "Home" | home: "Home" | ||||||
| @@ -464,7 +469,8 @@ 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" | enableAdvancedMfm: "Attiva MFM avanzati" | ||||||
|  | enableAnimatedMfm: "Attiva MFM animati" | ||||||
| doing: "In corso..." | doing: "In corso..." | ||||||
| category: "Categoria" | category: "Categoria" | ||||||
| tags: "Tag" | tags: "Tag" | ||||||
| @@ -861,6 +867,8 @@ failedToFetchAccountInformation: "Impossibile recuperare le informazioni sul pro | |||||||
| rateLimitExceeded: "Superato il limite di velocità." | rateLimitExceeded: "Superato il limite di velocità." | ||||||
| cropImage: "Ritaglio dell'immagine" | cropImage: "Ritaglio dell'immagine" | ||||||
| cropImageAsk: "Si desidera ritagliare l'immagine?" | cropImageAsk: "Si desidera ritagliare l'immagine?" | ||||||
|  | cropYes: "Ritaglia" | ||||||
|  | cropNo: "Non ritagliare" | ||||||
| file: "Allegati" | file: "Allegati" | ||||||
| recentNHours: "Ultime {n} ore" | recentNHours: "Ultime {n} ore" | ||||||
| recentNDays: "Ultimi {n} giorni" | recentNDays: "Ultimi {n} giorni" | ||||||
| @@ -939,6 +947,16 @@ cannotPerformTemporaryDescription: "L'attività non può essere svolta, poiché | |||||||
| preset: "Preimpostato" | preset: "Preimpostato" | ||||||
| selectFromPresets: "Seleziona preimpostato" | selectFromPresets: "Seleziona preimpostato" | ||||||
| achievements: "Obiettivi raggiunti" | achievements: "Obiettivi raggiunti" | ||||||
|  | gotInvalidResponseError: "Risposta del server non valida" | ||||||
|  | gotInvalidResponseErrorDescription: "Il server potrebbe essere irraggiungibile o in manutenzione. Riprova più tardi." | ||||||
|  | thisPostMayBeAnnoying: "Questa nota potrebbe essere offensiva" | ||||||
|  | thisPostMayBeAnnoyingHome: "Pubblica sulla timeline principale" | ||||||
|  | thisPostMayBeAnnoyingCancel: "Annulla" | ||||||
|  | thisPostMayBeAnnoyingIgnore: "Pubblica lo stesso" | ||||||
|  | collapseRenotes: "Comprimi i Rinota già letti" | ||||||
|  | internalServerError: "Errore interno del server" | ||||||
|  | internalServerErrorDescription: "Si è verificato un errore imprevisto all'interno del server" | ||||||
|  | copyErrorInfo: "Copia le informazioni sull'errore" | ||||||
| _achievements: | _achievements: | ||||||
|   earnedAt: "Data di conseguimento" |   earnedAt: "Data di conseguimento" | ||||||
|   _types: |   _types: | ||||||
| @@ -1323,72 +1341,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" | ||||||
| @@ -1593,12 +1545,15 @@ _permissions: | |||||||
|   "read:gallery-likes": "Visualizza i contenuti della galleria." |   "read:gallery-likes": "Visualizza i contenuti della galleria." | ||||||
|   "write:gallery-likes": "Manipolazione dei \"Mi piace\" della galleria." |   "write:gallery-likes": "Manipolazione dei \"Mi piace\" della galleria." | ||||||
| _auth: | _auth: | ||||||
|  |   shareAccessTitle: "Permessi dell'applicazione" | ||||||
|   shareAccess: "Vuoi autorizzare {name} ad accedere al tuo profilo?" |   shareAccess: "Vuoi autorizzare {name} ad accedere al tuo profilo?" | ||||||
|   shareAccessAsk: "Vuoi autorizzare questa App ad accedere al tuo profilo?" |   shareAccessAsk: "Vuoi autorizzare questa App ad accedere al tuo profilo?" | ||||||
|  |   permission: "{name} richiede i permessi seguenti" | ||||||
|   permissionAsk: "Questa app richiede le seguenti autorizzazioni:" |   permissionAsk: "Questa app richiede le seguenti autorizzazioni:" | ||||||
|   pleaseGoBack: "Si prega di ritornare sulla app" |   pleaseGoBack: "Si prega di ritornare sulla app" | ||||||
|   callback: "Ritornando sulla app" |   callback: "Ritornando sulla app" | ||||||
|   denied: "Accesso negato" |   denied: "Accesso negato" | ||||||
|  |   pleaseLogin: "Per favore accedi al tuo account per cambiare i permessi dell'applicazione" | ||||||
| _antennaSources: | _antennaSources: | ||||||
|   all: "Tutte le note" |   all: "Tutte le note" | ||||||
|   homeTimeline: "Note dagli utenti che segui" |   homeTimeline: "Note dagli utenti che segui" | ||||||
|   | |||||||
| @@ -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: "あなた" | ||||||
| @@ -257,6 +259,8 @@ noMoreHistory: "これより過去の履歴はありません" | |||||||
| startMessaging: "チャットを開始" | startMessaging: "チャットを開始" | ||||||
| nUsersRead: "{n}人が読みました" | nUsersRead: "{n}人が読みました" | ||||||
| agreeTo: "{0}に同意" | agreeTo: "{0}に同意" | ||||||
|  | agreeBelow: "下記に同意する" | ||||||
|  | basicNotesBeforeCreateAccount: "基本的な注意事項" | ||||||
| tos: "利用規約" | tos: "利用規約" | ||||||
| start: "始める" | start: "始める" | ||||||
| home: "ホーム" | home: "ホーム" | ||||||
| @@ -388,17 +392,20 @@ userList: "リスト" | |||||||
| about: "情報" | about: "情報" | ||||||
| aboutMisskey: "Misskeyについて" | aboutMisskey: "Misskeyについて" | ||||||
| administrator: "管理者" | administrator: "管理者" | ||||||
| token: "トークン" | token: "確認コード" | ||||||
| twoStepAuthentication: "二段階認証" | 2fa: "二要素認証" | ||||||
|  | totp: "認証アプリ" | ||||||
|  | totpDescription: "認証アプリを使ってワンタイムパスワードを入力" | ||||||
| moderator: "モデレーター" | moderator: "モデレーター" | ||||||
| moderation: "モデレーション" | moderation: "モデレーション" | ||||||
| nUsersMentioned: "{n}人が投稿" | nUsersMentioned: "{n}人が投稿" | ||||||
|  | securityKeyAndPasskey: "セキュリティキー・パスキー" | ||||||
| securityKey: "セキュリティキー" | securityKey: "セキュリティキー" | ||||||
| securityKeyName: "キーの名前" |  | ||||||
| registerSecurityKey: "セキュリティキーを登録する" |  | ||||||
| lastUsed: "最後の使用" | lastUsed: "最後の使用" | ||||||
|  | lastUsedAt: "最後の使用: {t}" | ||||||
| unregister: "登録を解除" | unregister: "登録を解除" | ||||||
| passwordLessLogin: "パスワード無しログイン" | passwordLessLogin: "パスワードレスログイン" | ||||||
|  | passwordLessLoginDescription: "パスワードを使用せず、セキュリティキーやパスキーなどのみでログインします" | ||||||
| resetPassword: "パスワードをリセット" | resetPassword: "パスワードをリセット" | ||||||
| newPasswordIs: "新しいパスワードは「{password}」です" | newPasswordIs: "新しいパスワードは「{password}」です" | ||||||
| reduceUiAnimation: "UIのアニメーションを減らす" | reduceUiAnimation: "UIのアニメーションを減らす" | ||||||
| @@ -413,24 +420,15 @@ markAsReadAllTalkMessages: "すべてのチャットを既読にする" | |||||||
| help: "ヘルプ" | help: "ヘルプ" | ||||||
| inputMessageHere: "ここにメッセージを入力" | inputMessageHere: "ここにメッセージを入力" | ||||||
| close: "閉じる" | close: "閉じる" | ||||||
| group: "グループ" |  | ||||||
| groups: "グループ" |  | ||||||
| createGroup: "グループを作成" |  | ||||||
| ownedGroups: "所有グループ" |  | ||||||
| joinedGroups: "参加しているグループ" |  | ||||||
| invites: "招待" | invites: "招待" | ||||||
| groupName: "グループ名" |  | ||||||
| members: "メンバー" | members: "メンバー" | ||||||
| transfer: "譲渡" | transfer: "譲渡" | ||||||
| messagingWithUser: "ユーザーとチャット" |  | ||||||
| messagingWithGroup: "グループでチャット" |  | ||||||
| title: "タイトル" | title: "タイトル" | ||||||
| text: "テキスト" | text: "テキスト" | ||||||
| enable: "有効にする" | enable: "有効にする" | ||||||
| next: "次" | next: "次" | ||||||
| retype: "再入力" | retype: "再入力" | ||||||
| noteOf: "{user}のノート" | noteOf: "{user}のノート" | ||||||
| inviteToGroup: "グループに招待" |  | ||||||
| quoteAttached: "引用付き" | quoteAttached: "引用付き" | ||||||
| quoteQuestion: "引用として添付しますか?" | quoteQuestion: "引用として添付しますか?" | ||||||
| noMessagesYet: "まだチャットはありません" | noMessagesYet: "まだチャットはありません" | ||||||
| @@ -452,20 +450,17 @@ passwordMatched: "一致しました" | |||||||
| passwordNotMatched: "一致していません" | passwordNotMatched: "一致していません" | ||||||
| signinWith: "{x}でログイン" | signinWith: "{x}でログイン" | ||||||
| signinFailed: "ログインできませんでした。ユーザー名とパスワードを確認してください。" | signinFailed: "ログインできませんでした。ユーザー名とパスワードを確認してください。" | ||||||
| tapSecurityKey: "セキュリティキーにタッチ" |  | ||||||
| or: "もしくは" | or: "もしくは" | ||||||
| language: "言語" | language: "言語" | ||||||
| uiLanguage: "UIの表示言語" | uiLanguage: "UIの表示言語" | ||||||
| groupInvited: "グループに招待されました" |  | ||||||
| aboutX: "{x}について" | aboutX: "{x}について" | ||||||
| emojiStyle: "絵文字のスタイル" | emojiStyle: "絵文字のスタイル" | ||||||
| native: "ネイティブ" | native: "ネイティブ" | ||||||
| disableDrawer: "メニューをドロワーで表示しない" | disableDrawer: "メニューをドロワーで表示しない" | ||||||
| youHaveNoGroups: "グループがありません" |  | ||||||
| joinOrCreateGroup: "既存のグループに招待してもらうか、新しくグループを作成してください。" |  | ||||||
| noHistory: "履歴はありません" | noHistory: "履歴はありません" | ||||||
| signinHistory: "ログイン履歴" | signinHistory: "ログイン履歴" | ||||||
| disableAnimatedMfm: "動きのあるMFMを無効にする" | enableAdvancedMfm: "高度なMFMを有効にする" | ||||||
|  | enableAnimatedMfm: "動きのあるMFMを有効にする" | ||||||
| doing: "やっています" | doing: "やっています" | ||||||
| category: "カテゴリ" | category: "カテゴリ" | ||||||
| tags: "タグ" | tags: "タグ" | ||||||
| @@ -784,6 +779,7 @@ popularPosts: "人気の投稿" | |||||||
| shareWithNote: "ノートで共有" | shareWithNote: "ノートで共有" | ||||||
| ads: "広告" | ads: "広告" | ||||||
| expiration: "期限" | expiration: "期限" | ||||||
|  | startingperiod: "開始期間" | ||||||
| memo: "メモ" | memo: "メモ" | ||||||
| priority: "優先度" | priority: "優先度" | ||||||
| high: "高" | high: "高" | ||||||
| @@ -835,8 +831,6 @@ deleteAccountConfirm: "アカウントが削除されます。よろしいです | |||||||
| incorrectPassword: "パスワードが間違っています。" | incorrectPassword: "パスワードが間違っています。" | ||||||
| voteConfirm: "「{choice}」に投票しますか?" | voteConfirm: "「{choice}」に投票しますか?" | ||||||
| hide: "隠す" | hide: "隠す" | ||||||
| leaveGroup: "グループから抜ける" |  | ||||||
| leaveGroupConfirm: "「{name}」から抜けますか?" |  | ||||||
| useDrawerReactionPickerForMobile: "モバイルデバイスのときドロワーで表示" | useDrawerReactionPickerForMobile: "モバイルデバイスのときドロワーで表示" | ||||||
| welcomeBackWithName: "おかえりなさい、{name}さん" | welcomeBackWithName: "おかえりなさい、{name}さん" | ||||||
| clickToFinishEmailVerification: "[{ok}]を押して、メールアドレスの確認を完了してください。" | clickToFinishEmailVerification: "[{ok}]を押して、メールアドレスの確認を完了してください。" | ||||||
| @@ -862,6 +856,8 @@ failedToFetchAccountInformation: "アカウント情報の取得に失敗しま | |||||||
| rateLimitExceeded: "レート制限を超えました" | rateLimitExceeded: "レート制限を超えました" | ||||||
| cropImage: "画像のクロップ" | cropImage: "画像のクロップ" | ||||||
| cropImageAsk: "画像をクロップしますか?" | cropImageAsk: "画像をクロップしますか?" | ||||||
|  | cropYes: "クロップする" | ||||||
|  | cropNo: "そのまま使う" | ||||||
| file: "ファイル" | file: "ファイル" | ||||||
| recentNHours: "直近{n}時間" | recentNHours: "直近{n}時間" | ||||||
| recentNDays: "直近{n}日" | recentNDays: "直近{n}日" | ||||||
| @@ -942,6 +938,17 @@ selectFromPresets: "プリセットから選択" | |||||||
| achievements: "実績" | achievements: "実績" | ||||||
| gotInvalidResponseError: "サーバーの応答が無効です" | gotInvalidResponseError: "サーバーの応答が無効です" | ||||||
| gotInvalidResponseErrorDescription: "サーバーがダウンまたはメンテナンスしている可能性があります。しばらくしてから再度お試しください。" | gotInvalidResponseErrorDescription: "サーバーがダウンまたはメンテナンスしている可能性があります。しばらくしてから再度お試しください。" | ||||||
|  | thisPostMayBeAnnoying: "この投稿は迷惑になる可能性があります。" | ||||||
|  | thisPostMayBeAnnoyingHome: "ホームに投稿" | ||||||
|  | thisPostMayBeAnnoyingCancel: "やめる" | ||||||
|  | thisPostMayBeAnnoyingIgnore: "このまま投稿" | ||||||
|  | collapseRenotes: "見たことのあるRenoteを省略して表示" | ||||||
|  | internalServerError: "サーバー内部エラー" | ||||||
|  | internalServerErrorDescription: "サーバー内部で予期しないエラーが発生しました。" | ||||||
|  | copyErrorInfo: "エラー情報をコピー" | ||||||
|  | joinThisServer: "このサーバーに登録する" | ||||||
|  | exploreOtherServers: "他のサーバーを探す" | ||||||
|  | letsLookAtTimeline: "タイムラインを見てみる" | ||||||
|  |  | ||||||
| _achievements: | _achievements: | ||||||
|   earnedAt: "獲得日時" |   earnedAt: "獲得日時" | ||||||
| @@ -1343,73 +1350,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: "リモートユーザーに表示" | ||||||
| @@ -1584,14 +1524,29 @@ _tutorial: | |||||||
|  |  | ||||||
| _2fa: | _2fa: | ||||||
|   alreadyRegistered: "既に設定は完了しています。" |   alreadyRegistered: "既に設定は完了しています。" | ||||||
|   registerDevice: "デバイスを登録" |   registerTOTP: "認証アプリの設定を開始" | ||||||
|   registerKey: "キーを登録" |   passwordToTOTP: "パスワードを入力してください" | ||||||
|   step1: "まず、{a}や{b}などの認証アプリをお使いのデバイスにインストールします。" |   step1: "まず、{a}や{b}などの認証アプリをお使いのデバイスにインストールします。" | ||||||
|   step2: "次に、表示されているQRコードをアプリでスキャンします。" |   step2: "次に、表示されているQRコードをアプリでスキャンします。" | ||||||
|   step2Url: "デスクトップアプリでは次のURLを入力します:" |   step2Click: "QRコードをクリックすると、お使いの端末にインストールされている認証アプリやキーリングに登録できます。" | ||||||
|   step3: "アプリに表示されているトークンを入力して完了です。" |   step2Url: "デスクトップアプリでは次のURIを入力します:" | ||||||
|   step4: "これからログインするときも、同じようにトークンを入力します。" |   step3Title: "確認コードを入力" | ||||||
|   securityKeyInfo: "FIDO2をサポートするハードウェアセキュリティキーもしくは端末の指紋認証やPINを使用してログインするように設定できます。" |   step3: "アプリに表示されている確認コード(トークン)を入力して完了です。" | ||||||
|  |   step4: "これからログインするときも、同じように確認コードを入力します。" | ||||||
|  |   securityKeyNotSupported: "お使いのブラウザはセキュリティキーに対応していません。" | ||||||
|  |   registerTOTPBeforeKey: "セキュリティキー・パスキーを登録するには、まず認証アプリの設定を行なってください。" | ||||||
|  |   securityKeyInfo: "FIDO2をサポートするハードウェアセキュリティキー、端末の生体認証やPINロック、パスキーといった、WebAuthn由来の鍵を登録します。" | ||||||
|  |   chromePasskeyNotSupported: "Chromeのパスキーは現在サポートしていません。" | ||||||
|  |   registerSecurityKey: "セキュリティキー・パスキーを登録する" | ||||||
|  |   securityKeyName: "キーの名前を入力" | ||||||
|  |   tapSecurityKey: "ブラウザの指示に従い、セキュリティキーやパスキーを登録してください" | ||||||
|  |   removeKey: "セキュリティキーを削除" | ||||||
|  |   removeKeyConfirm: "{name}を削除しますか?" | ||||||
|  |   whyTOTPOnlyRenew: "セキュリティキーが登録されている場合、認証アプリの設定は解除できません。" | ||||||
|  |   renewTOTP: "認証アプリを再設定" | ||||||
|  |   renewTOTPConfirm: "今までの認証アプリの確認コードは使用できなくなります" | ||||||
|  |   renewTOTPOk: "再設定する" | ||||||
|  |   renewTOTPCancel: "やめておく" | ||||||
|  |  | ||||||
| _permissions: | _permissions: | ||||||
|   "read:account": "アカウントの情報を見る" |   "read:account": "アカウントの情報を見る" | ||||||
| @@ -1628,19 +1583,21 @@ _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: "全てのノート" | ||||||
|   homeTimeline: "フォローしているユーザーのノート" |   homeTimeline: "フォローしているユーザーのノート" | ||||||
|   users: "指定した一人または複数のユーザーのノート" |   users: "指定した一人または複数のユーザーのノート" | ||||||
|   userList: "指定したリストのユーザーのノート" |   userList: "指定したリストのユーザーのノート" | ||||||
|   userGroup: "指定したグループのユーザーのノート" |  | ||||||
|  |  | ||||||
| _weekday: | _weekday: | ||||||
|   sunday: "日曜日" |   sunday: "日曜日" | ||||||
| @@ -1870,12 +1827,9 @@ _notification: | |||||||
|   youGotReply: "{name}からのリプライ" |   youGotReply: "{name}からのリプライ" | ||||||
|   youGotQuote: "{name}による引用" |   youGotQuote: "{name}による引用" | ||||||
|   youRenoted: "{name}がRenoteしました" |   youRenoted: "{name}がRenoteしました" | ||||||
|   youGotMessagingMessageFromUser: "{name}からのチャットがあります" |  | ||||||
|   youGotMessagingMessageFromGroup: "{name}のチャットがあります" |  | ||||||
|   youWereFollowed: "フォローされました" |   youWereFollowed: "フォローされました" | ||||||
|   youReceivedFollowRequest: "フォローリクエストが来ました" |   youReceivedFollowRequest: "フォローリクエストが来ました" | ||||||
|   yourFollowRequestAccepted: "フォローリクエストが承認されました" |   yourFollowRequestAccepted: "フォローリクエストが承認されました" | ||||||
|   youWereInvitedToGroup: "{userName}があなたをグループに招待しました" |  | ||||||
|   pollEnded: "アンケートの結果が出ました" |   pollEnded: "アンケートの結果が出ました" | ||||||
|   unreadAntennaNote: "アンテナ {name}" |   unreadAntennaNote: "アンテナ {name}" | ||||||
|   emptyPushNotificationMessage: "プッシュ通知の更新をしました" |   emptyPushNotificationMessage: "プッシュ通知の更新をしました" | ||||||
| @@ -1892,7 +1846,7 @@ _notification: | |||||||
|     pollEnded: "アンケートが終了" |     pollEnded: "アンケートが終了" | ||||||
|     receiveFollowRequest: "フォロー申請を受け取った" |     receiveFollowRequest: "フォロー申請を受け取った" | ||||||
|     followRequestAccepted: "フォローが受理された" |     followRequestAccepted: "フォローが受理された" | ||||||
|     groupInvited: "グループに招待された" |     achievementEarned: "実績の獲得" | ||||||
|     app: "連携アプリからの通知" |     app: "連携アプリからの通知" | ||||||
|  |  | ||||||
|   _actions: |   _actions: | ||||||
| @@ -1928,3 +1882,7 @@ _deck: | |||||||
|     channel: "チャンネル" |     channel: "チャンネル" | ||||||
|     mentions: "あなた宛て" |     mentions: "あなた宛て" | ||||||
|     direct: "ダイレクト" |     direct: "ダイレクト" | ||||||
|  |  | ||||||
|  | _dialog: | ||||||
|  |   charactersExceeded: "最大文字数を超えています! 現在 {current} / 制限 {max}" | ||||||
|  |   charactersBelow: "最小文字数を下回っています! 現在 {current} / 制限 {min}" | ||||||
|   | |||||||
| @@ -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: "フォローを頼む" | ||||||
| @@ -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: "ホーム" | ||||||
| @@ -300,7 +305,7 @@ avatar: "アイコン" | |||||||
| banner: "バナー" | banner: "バナー" | ||||||
| nsfw: "閲覧注意" | nsfw: "閲覧注意" | ||||||
| whenServerDisconnected: "サーバーとの接続が切れたとき" | whenServerDisconnected: "サーバーとの接続が切れたとき" | ||||||
| disconnectedFromServer: "サーバーとの通信が切れたで" | disconnectedFromServer: "サーバーが機嫌悪いねん" | ||||||
| reload: "リロード" | reload: "リロード" | ||||||
| doNothing: "何もせんとく" | doNothing: "何もせんとく" | ||||||
| reloadConfirm: "リロードしてええか?" | reloadConfirm: "リロードしてええか?" | ||||||
| @@ -464,7 +469,8 @@ youHaveNoGroups: "グループがあらへんねぇ。" | |||||||
| joinOrCreateGroup: "既存のグループに招待してもらうか、新しくグループ作ってからやってな" | joinOrCreateGroup: "既存のグループに招待してもらうか、新しくグループ作ってからやってな" | ||||||
| noHistory: "履歴はあらへんねぇ。" | noHistory: "履歴はあらへんねぇ。" | ||||||
| signinHistory: "ログイン履歴" | signinHistory: "ログイン履歴" | ||||||
| disableAnimatedMfm: "動きがやかましいMFMを止める" | enableAdvancedMfm: "ややこしいMFMもありにする" | ||||||
|  | enableAnimatedMfm: "動きがやかましいMFMも許したる" | ||||||
| doing: "やっとるがな" | doing: "やっとるがな" | ||||||
| category: "カテゴリ" | category: "カテゴリ" | ||||||
| tags: "タグ" | tags: "タグ" | ||||||
| @@ -673,8 +679,8 @@ sentReactionsCount: "リアクションした数やで" | |||||||
| receivedReactionsCount: "リアクションされた数" | receivedReactionsCount: "リアクションされた数" | ||||||
| pollVotesCount: "アンケートに投票した数" | pollVotesCount: "アンケートに投票した数" | ||||||
| pollVotedCount: "アンケートに投票された数" | pollVotedCount: "アンケートに投票された数" | ||||||
| yes: "はい" | yes: "ええで" | ||||||
| no: "いいえ" | no: "あかんで" | ||||||
| driveFilesCount: "ドライブのファイル数" | driveFilesCount: "ドライブのファイル数" | ||||||
| driveUsage: "ドライブ使用量やで" | driveUsage: "ドライブ使用量やで" | ||||||
| noCrawle: "クローラーによるインデックスを拒否するで" | noCrawle: "クローラーによるインデックスを拒否するで" | ||||||
| @@ -861,6 +867,8 @@ failedToFetchAccountInformation: "アカウントの取得に失敗したみた | |||||||
| rateLimitExceeded: "レート制限が超えたみたいやで" | rateLimitExceeded: "レート制限が超えたみたいやで" | ||||||
| cropImage: "画像のクロップ" | cropImage: "画像のクロップ" | ||||||
| cropImageAsk: "画像をクロップしたってええか?" | cropImageAsk: "画像をクロップしたってええか?" | ||||||
|  | cropYes: "切り抜いたる" | ||||||
|  | cropNo: "切り抜かへん" | ||||||
| file: "ファイル" | file: "ファイル" | ||||||
| recentNHours: "直近{n}時間" | recentNHours: "直近{n}時間" | ||||||
| recentNDays: "直近{n}日" | recentNDays: "直近{n}日" | ||||||
| @@ -938,6 +946,172 @@ cannotPerformTemporary: "一時的に利用できへんで" | |||||||
| cannotPerformTemporaryDescription: "操作回数が制限を超えたから一時的に利用できへんくなったで。ちょっと時間置いてからもう一回やってやー。" | cannotPerformTemporaryDescription: "操作回数が制限を超えたから一時的に利用できへんくなったで。ちょっと時間置いてからもう一回やってやー。" | ||||||
| preset: "プリセット" | preset: "プリセット" | ||||||
| selectFromPresets: "プリセットから選ぶ" | selectFromPresets: "プリセットから選ぶ" | ||||||
|  | achievements: "実績" | ||||||
|  | gotInvalidResponseError: "サーバー黙っとるわ、知らんけど" | ||||||
|  | gotInvalidResponseErrorDescription: "サーバーいま日曜日。またきて月曜日。" | ||||||
|  | thisPostMayBeAnnoying: "この投稿は迷惑かもしらんで。" | ||||||
|  | thisPostMayBeAnnoyingHome: "ホームに投稿" | ||||||
|  | thisPostMayBeAnnoyingCancel: "やめとく" | ||||||
|  | thisPostMayBeAnnoyingIgnore: "このまま投稿" | ||||||
|  | collapseRenotes: "見たことあるRenoteは省略やで" | ||||||
|  | internalServerError: "サーバー内部エラー" | ||||||
|  | internalServerErrorDescription: "サーバー内部でよう分からんエラーやわ" | ||||||
|  | copyErrorInfo: "エラー情報をコピー" | ||||||
|  | _achievements: | ||||||
|  |   earnedAt: "貰った日ぃ" | ||||||
|  |   _types: | ||||||
|  |     _notes1: | ||||||
|  |       title: "まいど!" | ||||||
|  |       description: "初めてノート投稿したった" | ||||||
|  |       flavor: "Misskeyを楽しんでな~" | ||||||
|  |     _notes10: | ||||||
|  |       title: "ノートの天保山" | ||||||
|  |       description: "ノートを10回投稿した" | ||||||
|  |     _notes100: | ||||||
|  |       title: "ノートの真田山" | ||||||
|  |       description: "ノートを100回投稿した" | ||||||
|  |     _notes500: | ||||||
|  |       title: "ノートの生駒山" | ||||||
|  |       description: "ノートを500回投稿した" | ||||||
|  |     _notes1000: | ||||||
|  |       title: "ノートの山" | ||||||
|  |       description: "ノートを1,000回投稿した" | ||||||
|  |     _notes5000: | ||||||
|  |       title: "箕面の滝からノート" | ||||||
|  |       description: "ノートを5,000回投稿した" | ||||||
|  |     _notes10000: | ||||||
|  |       title: "スーパーノート" | ||||||
|  |       description: "ノートを10,000回投稿した" | ||||||
|  |     _notes20000: | ||||||
|  |       title: "ニードモアノート" | ||||||
|  |       description: "ノートを20,000回投稿した" | ||||||
|  |     _notes30000: | ||||||
|  |       title: "ノートノートノート" | ||||||
|  |       description: "ノートを30,000回投稿した" | ||||||
|  |     _notes40000: | ||||||
|  |       title: "ノート工場" | ||||||
|  |       description: "ノートを40,000回投稿した" | ||||||
|  |     _notes50000: | ||||||
|  |       title: "ノートの惑星" | ||||||
|  |       description: "ノートを50,000回投稿した" | ||||||
|  |     _notes60000: | ||||||
|  |       title: "ノートクエーサー" | ||||||
|  |       description: "ノートを60,000回投稿した" | ||||||
|  |     _notes70000: | ||||||
|  |       title: "ブラックノートホール" | ||||||
|  |       description: "ノートを70,000回投稿した" | ||||||
|  |     _notes80000: | ||||||
|  |       title: "ノートギャラクシー" | ||||||
|  |       description: "ノートを80,000回投稿した" | ||||||
|  |     _notes90000: | ||||||
|  |       title: "ノートバース" | ||||||
|  |       description: "ノートを90,000回投稿した" | ||||||
|  |     _notes100000: | ||||||
|  |       title: "ALL YOUR NOTE ARE BELONG TO US" | ||||||
|  |       description: "ノートを100,000回投稿した" | ||||||
|  |       flavor: "そんなに書くことあるんか?" | ||||||
|  |     _login3: | ||||||
|  |       title: "ビギナーⅠ" | ||||||
|  |       description: "通算ログイン日数が3日" | ||||||
|  |       flavor: "今日からワシはミスキストやで" | ||||||
|  |     _login7: | ||||||
|  |       title: "ビギナーⅡ" | ||||||
|  |       description: "通算ログイン日数が7日" | ||||||
|  |       flavor: "慣れてきたんちゃう?" | ||||||
|  |     _login15: | ||||||
|  |       title: "ビギナーⅢ" | ||||||
|  |       description: "通算ログイン日数が15日" | ||||||
|  |     _login30: | ||||||
|  |       title: "ミスキストⅠ" | ||||||
|  |       description: "通算ログイン日数が30日" | ||||||
|  |     _login60: | ||||||
|  |       title: "ミスキストⅡ" | ||||||
|  |       description: "通算ログイン日数が60日" | ||||||
|  |     _login100: | ||||||
|  |       title: "ミスキストⅢ" | ||||||
|  |       description: "通算ログイン日数が100日" | ||||||
|  |       flavor: "そのユーザー、ミスキストにつき" | ||||||
|  |     _login200: | ||||||
|  |       title: "常連Ⅰ" | ||||||
|  |     _followers500: | ||||||
|  |       title: "基地局" | ||||||
|  |       description: "フォロワーが500人を超した" | ||||||
|  |     _followers1000: | ||||||
|  |       title: "インフルエンサー" | ||||||
|  |       description: "フォロワーが1,000人を超した" | ||||||
|  |     _collectAchievements30: | ||||||
|  |       title: "実績コレクター" | ||||||
|  |       description: "実績を30個以上獲得した" | ||||||
|  |     _viewAchievements3min: | ||||||
|  |       title: "実績好き" | ||||||
|  |       description: "実績一覧を3分以上眺め続けた" | ||||||
|  |     _iLoveMisskey: | ||||||
|  |       title: "Misskey好きやねん" | ||||||
|  |       description: "\"I ❤ #Misskey\"を投稿した" | ||||||
|  |       flavor: "Misskeyを使ってくれてありがとうな~ by 開発チーム" | ||||||
|  |     _foundTreasure: | ||||||
|  |       title: "なんでも鑑定団" | ||||||
|  |       description: "隠されたお宝を発見した" | ||||||
|  |     _client30min: | ||||||
|  |       title: "ねんね" | ||||||
|  |       description: "クライアントを起動してから30分以上経過した" | ||||||
|  |     _noteDeletedWithin1min: | ||||||
|  |       title: "*おおっと*" | ||||||
|  |       description: "投稿してから1分以内にその投稿を消した" | ||||||
|  |     _postedAtLateNight: | ||||||
|  |       title: "夜行性" | ||||||
|  |       description: "深夜にノートを投稿した" | ||||||
|  |       flavor: "そろそろ寝よか" | ||||||
|  |     _postedAt0min0sec: | ||||||
|  |       title: "時報" | ||||||
|  |       description: "0分0秒にノートを投稿した" | ||||||
|  |       flavor: "ポッ ポッ ポッ ピーン" | ||||||
|  |     _selfQuote: | ||||||
|  |       title: "自己言及" | ||||||
|  |       description: "自分のノートを引用した" | ||||||
|  |     _htl20npm: | ||||||
|  |       title: "流れるTL" | ||||||
|  |       description: "ホームタイムラインの流速が20npmを超す" | ||||||
|  |     _viewInstanceChart: | ||||||
|  |       title: "アナリスト" | ||||||
|  |       description: "インスタンスのチャートを表示した" | ||||||
|  |     _outputHelloWorldOnScratchpad: | ||||||
|  |       title: "Hello, world!" | ||||||
|  |       description: "スクラッチパッドで hello worldを出力した" | ||||||
|  |     _open3windows: | ||||||
|  |       title: "マド開けすぎ" | ||||||
|  |       description: "ウィンドウを3つ以上開いた状態にした" | ||||||
|  |     _driveFolderCircularReference: | ||||||
|  |       title: "環状線" | ||||||
|  |       description: "ドライブのフォルダを再帰的な入れ子にしようとした" | ||||||
|  |     _reactWithoutRead: | ||||||
|  |       title: "ちゃんと読んだんか?" | ||||||
|  |       description: "100文字以上のテキストを含むノートに投稿されてから3秒以内にリアクションした" | ||||||
|  |     _clickedClickHere: | ||||||
|  |       title: "ここをクリック" | ||||||
|  |       description: "ここをクリックした" | ||||||
|  |     _justPlainLucky: | ||||||
|  |       title: "単なるラッキー" | ||||||
|  |       description: "10秒ごとに0.005%の確率で獲得" | ||||||
|  |     _setNameToSyuilo: | ||||||
|  |       title: "神様コンプレックス" | ||||||
|  |       description: "名前を syuilo に設定した" | ||||||
|  |     _passedSinceAccountCreated1: | ||||||
|  |       title: "一周年" | ||||||
|  |       description: "アカウント作成から1年経過した" | ||||||
|  |     _passedSinceAccountCreated2: | ||||||
|  |       title: "二周年" | ||||||
|  |       description: "アカウント作成から2年経過した" | ||||||
|  |     _passedSinceAccountCreated3: | ||||||
|  |       title: "三周年" | ||||||
|  |       description: "アカウント作成から3年経過した" | ||||||
|  |     _loggedInOnBirthday: | ||||||
|  |       title: "ハッピーバースデー!" | ||||||
|  |       description: "誕生日にログインした" | ||||||
|  |     _loggedInOnNewYearsDay: | ||||||
|  |       title: "あけましておめでとうございます!" | ||||||
|  |       description: "元旦にログインした" | ||||||
|  |       flavor: "今年も弊インスタンスをよろしくお願いします" | ||||||
| _role: | _role: | ||||||
|   new: "ロールの作成" |   new: "ロールの作成" | ||||||
|   edit: "ロールの編集" |   edit: "ロールの編集" | ||||||
| @@ -1083,72 +1257,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 +1463,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 +1697,7 @@ _notification: | |||||||
|   pollEnded: "アンケートの結果が出たみたいや" |   pollEnded: "アンケートの結果が出たみたいや" | ||||||
|   unreadAntennaNote: "アンテナ {name}" |   unreadAntennaNote: "アンテナ {name}" | ||||||
|   emptyPushNotificationMessage: "プッシュ通知の更新をしといたで" |   emptyPushNotificationMessage: "プッシュ通知の更新をしといたで" | ||||||
|  |   achievementEarned: "実績を獲得しとるで" | ||||||
|   _types: |   _types: | ||||||
|     all: "すべて" |     all: "すべて" | ||||||
|     follow: "フォロー" |     follow: "フォロー" | ||||||
|   | |||||||
| @@ -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: "팔로우중인 유저의 노트" | ||||||
|   | |||||||
| @@ -49,24 +49,191 @@ deleteAndEdit: "ລົບແລະແກ້ໄຂ" | |||||||
| deleteAndEditConfirm: "ເຈົ້າແນ່ໃຈບໍ່? ທີ່ທ່ານຕ້ອງການທີ່ຈະລຶບບັນທຶກນີ້ແລະແກ້ໄຂມັນ ທ່ານອາດຈະສູນເສຍການໂຕ້ຕອບ, ບັນທຶກ, ແລະການຕອບກັບທັງໝົດ" | deleteAndEditConfirm: "ເຈົ້າແນ່ໃຈບໍ່? ທີ່ທ່ານຕ້ອງການທີ່ຈະລຶບບັນທຶກນີ້ແລະແກ້ໄຂມັນ ທ່ານອາດຈະສູນເສຍການໂຕ້ຕອບ, ບັນທຶກ, ແລະການຕອບກັບທັງໝົດ" | ||||||
| addToList: "ເພີ່ມໃສ່ລາຍຊື່" | addToList: "ເພີ່ມໃສ່ລາຍຊື່" | ||||||
| sendMessage: "ສົ່ງຂໍ້ຄວາມ" | 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: "ປັກໝຸດໄປຫາໂປຣໄຟລ໌" | 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: "ອີນສະແຕນ" | instances: "ອີນສະແຕນ" | ||||||
|  | instanceInfo: "ອີນສະແຕນ" | ||||||
|  | statistics: "ສະຖິຕິ" | ||||||
|  | clearQueue: "ລ້າງຄິວ" | ||||||
|  | clearCachedFiles: "ລຶບລ້າງແຄສ" | ||||||
|  | editProfile: "ແກ້ໄຂໂປຣໄຟລ໌" | ||||||
|  | done: "ສຳເລັດ" | ||||||
|  | processing: "ກຳລັງປະມວນຜົນ" | ||||||
|  | preview: "ສະແດງເປັນຕົວຢ່າງ" | ||||||
|  | default: "ຄ່າເລີ່ມຕົ້ນ" | ||||||
|  | blocked: "ບລັອກແລ້ວ " | ||||||
|  | all: "ທັງໝົດ" | ||||||
|  | subscribing: "ສະໝັກສະມາຊິກແລັວ" | ||||||
|  | publishing: "ການພິມເຜີຍແຜ່" | ||||||
|  | notResponding: "ບໍ່ຕອບສະໜອງ" | ||||||
|  | instanceFollowing: "ກຳລັງຕິດຕາມສຸດຕົວຢ່າງ" | ||||||
|  | instanceFollowers: "ຜູ້ຕິດຕາມຕົວຢ່າງ" | ||||||
|  | instanceUsers: "ຜູ້ຊົມໃຊ້ຂອງຕົວຢ່າງນີ້" | ||||||
|  | changePassword: "ປ່ຽນລະຫັດຜ່ານ" | ||||||
|  | featured: "ໄຮໄລທ໌" | ||||||
|  | announcements: "ປະກາດ" | ||||||
| remove: "ລຶບ" | remove: "ລຶບ" | ||||||
|  | messaging: "ແຊ໋ດ" | ||||||
|  | tos: "ເງື່ອນໄຂການໃຫ້ບໍລິການ" | ||||||
|  | start: "ເລີ່ມຕົ້ນນຳໃຊ້ເລີຍ" | ||||||
|  | home: "ໜ້າຫຼັກ" | ||||||
|  | images: "ຮູບພາບ" | ||||||
|  | birthday: "ວັນເກີດ" | ||||||
|  | registeredDate: "ວັນທີ່ເປັນສະມາຊິກ" | ||||||
|  | location: "ທີ່ຕັ້ງ" | ||||||
|  | theme: "ແທ໋ມ" | ||||||
|  | light: "ສະຫວ່າງ" | ||||||
|  | dark: "ມືດ" | ||||||
|  | lightThemes: "ຊຸດຮູບແບບສະຫວ່າງ" | ||||||
|  | darkThemes: "ຮູບແບບສີສັນມືດ" | ||||||
|  | fileName: "ຊື່ໄຟລ໌" | ||||||
|  | selectFile: "ເລືອກໄຟລ໌" | ||||||
|  | selectFiles: "ເລືອກໄຟລ໌" | ||||||
|  | nsfw: "NSFW" | ||||||
|  | accept: "ອະນຸຍາດ" | ||||||
|  | pinnedNotes: "ບັນທຶກທີ່ປັກໝຸດໄວ້" | ||||||
|  | userList: "ລາຍການ" | ||||||
| smtpUser: "ຊື່ຜູ້ໃຊ້" | smtpUser: "ຊື່ຜູ້ໃຊ້" | ||||||
| smtpPass: "ລະຫັດຜ່ານ" | smtpPass: "ລະຫັດຜ່ານ" | ||||||
|  | clearCache: "ລຶບລ້າງແຄສ" | ||||||
| user: "ຜູ້ໃຊ້ຕ່າງໆ" | user: "ຜູ້ໃຊ້ຕ່າງໆ" | ||||||
| searchByGoogle: "ຄົ້ນຫາ" | searchByGoogle: "ຄົ້ນຫາ" | ||||||
| _mfm: | file: "ໄຟລ໌" | ||||||
|   search: "ຄົ້ນຫາ" | _email: | ||||||
|  |   _follow: | ||||||
|  |     title: "ໄດ້ຕິດຕາມທ່ານ" | ||||||
|  | _theme: | ||||||
|  |   keys: | ||||||
|  |     mention: "ໄດ້ກ່າວມາ" | ||||||
|  |     renote: "Renote" | ||||||
| _sfx: | _sfx: | ||||||
|  |   note: "ບັນທຶກ" | ||||||
|   notification: "ການແຈ້ງເຕືອນ" |   notification: "ການແຈ້ງເຕືອນ" | ||||||
|  |   chat: "ແຊ໋ດ" | ||||||
| _widgets: | _widgets: | ||||||
|   profile: "ໂພຼຟາຍ" |   profile: "ໂພຼຟາຍ" | ||||||
|  |   instanceInfo: "ອີນສະແຕນ" | ||||||
|   notifications: "ການແຈ້ງເຕືອນ" |   notifications: "ການແຈ້ງເຕືອນ" | ||||||
|   timeline: "ເສັ້ນກຳນົດເວລາ" |   timeline: "ເສັ້ນກຳນົດເວລາ" | ||||||
|  |   _userList: | ||||||
|  |     chooseList: "ເລືອກບັນຊີລາຍການ" | ||||||
|  | _cw: | ||||||
|  |   show: "ໂຫຼດເພີ່ມເຕີມ" | ||||||
|  | _visibility: | ||||||
|  |   home: "ໜ້າຫຼັກ" | ||||||
|  |   followers: "ຜູ້ຕິດຕາມ" | ||||||
| _profile: | _profile: | ||||||
|   username: "ຊື່ຜູ້ໃຊ້" |   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: | _deck: | ||||||
|   _columns: |   _columns: | ||||||
|     notifications: "ການແຈ້ງເຕືອນ" |     notifications: "ການແຈ້ງເຕືອນ" | ||||||
|     tl: "ເສັ້ນກຳນົດເວລາ" |     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" | ||||||
|   | |||||||
| @@ -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: | ||||||
|   | |||||||
| @@ -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: "Только для других сайтов" | ||||||
|   | |||||||
| @@ -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" | ||||||
|   | |||||||
| @@ -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: | ||||||
| @@ -1323,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: "แสดงสำหรับผู้ใช้ระยะไกล" | ||||||
| @@ -1593,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: "โน้ตจากผู้ใช้ที่ติดตาม" | ||||||
|   | |||||||
| @@ -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: "Теги" | ||||||
| @@ -1087,6 +1086,9 @@ _achievements: | |||||||
|     _outputHelloWorldOnScratchpad: |     _outputHelloWorldOnScratchpad: | ||||||
|       title: "Hello, world!" |       title: "Hello, world!" | ||||||
|       description: "Вивести \"hello world\" у Скретчпаді" |       description: "Вивести \"hello world\" у Скретчпаді" | ||||||
|  |     _reactWithoutRead: | ||||||
|  |       title: "Прочитали як слід?" | ||||||
|  |       description: "Реакція на нотатку, що містить понад 100 символів, протягом 3 секунд після її публікації" | ||||||
|     _clickedClickHere: |     _clickedClickHere: | ||||||
|       title: "Натисніть тут" |       title: "Натисніть тут" | ||||||
|       description: "Натиснуто тут" |       description: "Натиснуто тут" | ||||||
| @@ -1111,6 +1113,8 @@ _achievements: | |||||||
|     _loggedInOnNewYearsDay: |     _loggedInOnNewYearsDay: | ||||||
|       title: "З Новим роком!" |       title: "З Новим роком!" | ||||||
|       description: "Увійшли в перший день року" |       description: "Увійшли в перший день року" | ||||||
|  |     _cookieClicked: | ||||||
|  |       flavor: "Чекайте, це вірний сайт?" | ||||||
|     _brainDiver: |     _brainDiver: | ||||||
|       title: "Brain Diver" |       title: "Brain Diver" | ||||||
|       description: "Відправити посилання на \"Brain Diver\"" |       description: "Відправити посилання на \"Brain Diver\"" | ||||||
| @@ -1191,65 +1195,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: "Відображати для віддалених користувачів" | ||||||
|   | |||||||
| @@ -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" | ||||||
|   | |||||||
| @@ -103,6 +103,8 @@ renoted: "已转发。" | |||||||
| cantRenote: "该帖无法转发。" | cantRenote: "该帖无法转发。" | ||||||
| cantReRenote: "转发无法被再次转发。" | cantReRenote: "转发无法被再次转发。" | ||||||
| quote: "引用" | quote: "引用" | ||||||
|  | inChannelRenote: "在频道内转发" | ||||||
|  | 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: "省略显示已经看过的转发内容" | ||||||
|  | internalServerError: "内部服务器错误" | ||||||
|  | internalServerErrorDescription: "内部服务器发生了预期外的错误" | ||||||
|  | copyErrorInfo: "复制错误信息" | ||||||
| _achievements: | _achievements: | ||||||
|   earnedAt: "达成时间" |   earnedAt: "达成时间" | ||||||
|   _types: |   _types: | ||||||
| @@ -1122,7 +1140,7 @@ _achievements: | |||||||
|       description: "在0点发布一篇帖子" |       description: "在0点发布一篇帖子" | ||||||
|       flavor: "嘣 嘣 嘣 Biu——!" |       flavor: "嘣 嘣 嘣 Biu——!" | ||||||
|     _selfQuote: |     _selfQuote: | ||||||
|       title: "自我提及" |       title: "自我引用" | ||||||
|       description: "引用了自己的帖子" |       description: "引用了自己的帖子" | ||||||
|     _htl20npm: |     _htl20npm: | ||||||
|       title: "流动的时间线" |       title: "流动的时间线" | ||||||
| @@ -1323,72 +1341,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: "仅远程用户" | ||||||
| @@ -1593,12 +1545,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: "已关注用户的帖子" | ||||||
|   | |||||||
| @@ -103,6 +103,8 @@ renoted: "轉傳成功" | |||||||
| cantRenote: "無法轉發此貼文。" | cantRenote: "無法轉發此貼文。" | ||||||
| cantReRenote: "無法轉傳之前已經轉傳過的內容。" | cantReRenote: "無法轉傳之前已經轉傳過的內容。" | ||||||
| quote: "引用" | quote: "引用" | ||||||
|  | inChannelRenote: "在頻道內轉發" | ||||||
|  | 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: "省略顯示已看過的轉發貼文" | ||||||
|  | internalServerError: "內部伺服器錯誤" | ||||||
|  | internalServerErrorDescription: "內部伺服器發生了非預期的錯誤。" | ||||||
|  | copyErrorInfo: "複製錯誤資訊" | ||||||
| _achievements: | _achievements: | ||||||
|   earnedAt: "獲得日期" |   earnedAt: "獲得日期" | ||||||
|   _types: |   _types: | ||||||
| @@ -1323,72 +1341,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: "向遠端使用者顯示" | ||||||
| @@ -1593,12 +1545,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: "來自已追隨使用者的貼文" | ||||||
|   | |||||||
							
								
								
									
										14
									
								
								package.json
									
									
									
									
									
								
							
							
						
						
									
										14
									
								
								package.json
									
									
									
									
									
								
							| @@ -1,12 +1,12 @@ | |||||||
| { | { | ||||||
| 	"name": "misskey", | 	"name": "misskey", | ||||||
| 	"version": "13.5.4", | 	"version": "13.7.3", | ||||||
| 	"codename": "nasubi", | 	"codename": "nasubi", | ||||||
| 	"repository": { | 	"repository": { | ||||||
| 		"type": "git", | 		"type": "git", | ||||||
| 		"url": "https://github.com/misskey-dev/misskey.git" | 		"url": "https://github.com/misskey-dev/misskey.git" | ||||||
| 	}, | 	}, | ||||||
| 	"packageManager": "pnpm@7.24.3", | 	"packageManager": "pnpm@7.27.0", | ||||||
| 	"workspaces": [ | 	"workspaces": [ | ||||||
| 		"packages/frontend", | 		"packages/frontend", | ||||||
| 		"packages/backend", | 		"packages/backend", | ||||||
| @@ -54,12 +54,12 @@ | |||||||
| 	"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.51.0", | 		"@typescript-eslint/eslint-plugin": "5.52.0", | ||||||
| 		"@typescript-eslint/parser": "5.51.0", | 		"@typescript-eslint/parser": "5.52.0", | ||||||
| 		"cross-env": "7.0.3", | 		"cross-env": "7.0.3", | ||||||
| 		"cypress": "12.5.1", | 		"cypress": "12.6.0", | ||||||
| 		"eslint": "8.33.0", | 		"eslint": "8.34.0", | ||||||
| 		"start-server-and-test": "1.15.3" | 		"start-server-and-test": "1.15.4" | ||||||
| 	}, | 	}, | ||||||
| 	"optionalDependencies": { | 	"optionalDependencies": { | ||||||
| 		"@tensorflow/tfjs-core": "4.2.0" | 		"@tensorflow/tfjs-core": "4.2.0" | ||||||
|   | |||||||
| @@ -1,25 +1,23 @@ | |||||||
| { | { | ||||||
|   "$schema": "https://json.schemastore.org/swcrc", | 	"$schema": "https://json.schemastore.org/swcrc", | ||||||
|   "jsc": { | 	"jsc": { | ||||||
|     "parser": { | 		"parser": { | ||||||
|       "syntax": "typescript", | 			"syntax": "typescript", | ||||||
|       "dynamicImport": true, | 			"dynamicImport": true, | ||||||
|       "decorators": true | 			"decorators": true | ||||||
|     }, | 		}, | ||||||
|     "transform": { | 		"transform": { | ||||||
|       "legacyDecorator": true, | 			"legacyDecorator": true, | ||||||
|       "decoratorMetadata": true | 			"decoratorMetadata": true | ||||||
|     }, | 		}, | ||||||
| 		"experimental": { | 		"experimental": { | ||||||
| 			"keepImportAssertions": true | 			"keepImportAssertions": true | ||||||
| 		}, | 		}, | ||||||
| 		"baseUrl": ".", | 		"baseUrl": "src", | ||||||
| 		"paths": { | 		"paths": { | ||||||
| 			"@/*": [ | 			"@/*": ["*"] | ||||||
| 				"./src/*" |  | ||||||
| 			] |  | ||||||
| 		}, | 		}, | ||||||
| 		"target": "es2021" | 		"target": "es2021" | ||||||
|   }, | 	}, | ||||||
|   "minify": false | 	"minify": false | ||||||
| } | } | ||||||
|   | |||||||
										
											Binary file not shown.
										
									
								
							| Before Width: | Height: | Size: 7.5 KiB After Width: | Height: | Size: 26 KiB | 
										
											Binary file not shown.
										
									
								
							| Before Width: | Height: | Size: 20 KiB After Width: | Height: | Size: 162 KiB | 
| @@ -1,14 +0,0 @@ | |||||||
| // https://github.com/facebook/jest/issues/12270#issuecomment-1194746382 |  | ||||||
|  |  | ||||||
| const nativeModule = require('node:module'); |  | ||||||
|  |  | ||||||
| function resolver(module, options) { |  | ||||||
|   const { basedir, defaultResolver } = options; |  | ||||||
|   try { |  | ||||||
|     return defaultResolver(module, options); |  | ||||||
|   } catch (error) { |  | ||||||
|     return nativeModule.createRequire(basedir).resolve(module); |  | ||||||
|   } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| module.exports = resolver; |  | ||||||
| @@ -83,7 +83,14 @@ module.exports = { | |||||||
|  |  | ||||||
| 	// A map from regular expressions to module names or to arrays of module names that allow to stub out resources with a single module | 	// A map from regular expressions to module names or to arrays of module names that allow to stub out resources with a single module | ||||||
| 	moduleNameMapper: { | 	moduleNameMapper: { | ||||||
| 		"^@/(.*?).js": "<rootDir>/src/$1.ts", | 		// Do not resolve .wasm.js to .wasm by the rule below | ||||||
|  | 		'^(.+)\\.wasm\\.js$': '$1.wasm.js', | ||||||
|  | 		// SWC converts @/foo/bar.js to `../../src/foo/bar.js`, and then this rule | ||||||
|  | 		// converts it again to `../../src/foo/bar` which then can be resolved to | ||||||
|  | 		// `.ts` files. | ||||||
|  | 		// See https://github.com/swc-project/jest/issues/64#issuecomment-1029753225 | ||||||
|  | 		// TODO: Use `--allowImportingTsExtensions` on TypeScript 5.0 so that we can | ||||||
|  | 		// directly import `.ts` files without this hack. | ||||||
| 		'^(\\.{1,2}/.*)\\.js$': '$1', | 		'^(\\.{1,2}/.*)\\.js$': '$1', | ||||||
| 	}, | 	}, | ||||||
|  |  | ||||||
| @@ -112,7 +119,7 @@ module.exports = { | |||||||
| 	// resetModules: false, | 	// resetModules: false, | ||||||
|  |  | ||||||
| 	// A path to a custom resolver | 	// A path to a custom resolver | ||||||
| 	resolver: './jest-resolver.cjs', | 	// resolver: './jest-resolver.cjs', | ||||||
|  |  | ||||||
| 	// Automatically restore mock state between every test | 	// Automatically restore mock state between every test | ||||||
| 	restoreMocks: true, | 	restoreMocks: true, | ||||||
|   | |||||||
							
								
								
									
										27
									
								
								packages/backend/migration/1676434944993-drop-group.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								packages/backend/migration/1676434944993-drop-group.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,27 @@ | |||||||
|  | export class dropGroup1676434944993 { | ||||||
|  |     name = 'dropGroup1676434944993' | ||||||
|  |  | ||||||
|  |     async up(queryRunner) { | ||||||
|  |         await queryRunner.query(`ALTER TABLE "antenna" DROP CONSTRAINT "FK_ccbf5a8c0be4511133dcc50ddeb"`); | ||||||
|  |         await queryRunner.query(`ALTER TABLE "notification" DROP CONSTRAINT "FK_8fe87814e978053a53b1beb7e98"`); | ||||||
|  |         await queryRunner.query(`ALTER TABLE "antenna" DROP COLUMN "userGroupJoiningId"`); | ||||||
|  |         await queryRunner.query(`ALTER TABLE "notification" DROP COLUMN "userGroupInvitationId"`); | ||||||
|  |         await queryRunner.query(`ALTER TYPE "public"."antenna_src_enum" RENAME TO "antenna_src_enum_old"`); | ||||||
|  |         await queryRunner.query(`CREATE TYPE "public"."antenna_src_enum" AS ENUM('home', 'all', 'users', 'list')`); | ||||||
|  |         await queryRunner.query(`ALTER TABLE "antenna" ALTER COLUMN "src" TYPE "public"."antenna_src_enum" USING "src"::"text"::"public"."antenna_src_enum"`); | ||||||
|  |         await queryRunner.query(`DROP TYPE "public"."antenna_src_enum_old"`); | ||||||
|  |         await queryRunner.query(`ALTER TABLE "user_profile" ALTER COLUMN "emailNotificationTypes" SET DEFAULT '["follow","receiveFollowRequest"]'`); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     async down(queryRunner) { | ||||||
|  |         await queryRunner.query(`ALTER TABLE "user_profile" ALTER COLUMN "emailNotificationTypes" SET DEFAULT '["follow", "receiveFollowRequest", "groupInvited"]'`); | ||||||
|  |         await queryRunner.query(`CREATE TYPE "public"."antenna_src_enum_old" AS ENUM('home', 'all', 'users', 'list', 'group')`); | ||||||
|  |         await queryRunner.query(`ALTER TABLE "antenna" ALTER COLUMN "src" TYPE "public"."antenna_src_enum_old" USING "src"::"text"::"public"."antenna_src_enum_old"`); | ||||||
|  |         await queryRunner.query(`DROP TYPE "public"."antenna_src_enum"`); | ||||||
|  |         await queryRunner.query(`ALTER TYPE "public"."antenna_src_enum_old" RENAME TO "antenna_src_enum"`); | ||||||
|  |         await queryRunner.query(`ALTER TABLE "notification" ADD "userGroupInvitationId" character varying(32)`); | ||||||
|  |         await queryRunner.query(`ALTER TABLE "antenna" ADD "userGroupJoiningId" character varying(32)`); | ||||||
|  |         await queryRunner.query(`ALTER TABLE "notification" ADD CONSTRAINT "FK_8fe87814e978053a53b1beb7e98" FOREIGN KEY ("userGroupInvitationId") REFERENCES "user_group_invitation"("id") ON DELETE CASCADE ON UPDATE NO ACTION`); | ||||||
|  |         await queryRunner.query(`ALTER TABLE "antenna" ADD CONSTRAINT "FK_ccbf5a8c0be4511133dcc50ddeb" FOREIGN KEY ("userGroupJoiningId") REFERENCES "user_group_joining"("id") ON DELETE CASCADE ON UPDATE NO ACTION`); | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										9
									
								
								packages/backend/migration/1676438468213-ad3.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								packages/backend/migration/1676438468213-ad3.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,9 @@ | |||||||
|  | export class ad1676438468213 { | ||||||
|  | 	name = 'ad1676438468213'; | ||||||
|  | 	async up(queryRunner) { | ||||||
|  | 			await queryRunner.query(`ALTER TABLE "ad" ADD "startsAt" TIMESTAMP WITH TIME ZONE NOT NULL DEFAULT now()`); | ||||||
|  | 	} | ||||||
|  | 	async down(queryRunner) { | ||||||
|  | 		await queryRunner.query(`ALTER TABLE "ad" DROP COLUMN "startsAt"`); | ||||||
|  | 	} | ||||||
|  | } | ||||||
| @@ -7,11 +7,13 @@ | |||||||
| 		"start": "node ./built/index.js", | 		"start": "node ./built/index.js", | ||||||
| 		"start:test": "NODE_ENV=test node ./built/index.js", | 		"start:test": "NODE_ENV=test node ./built/index.js", | ||||||
| 		"migrate": "pnpm typeorm migration:run -d ormconfig.js", | 		"migrate": "pnpm typeorm migration:run -d ormconfig.js", | ||||||
| 		"build:swc": "swc src -d built -D", | 		"build": "swc src -d built -D", | ||||||
| 		"watch:swc": "swc src -d built -D -w", | 		"watch:swc": "swc src -d built -D -w", | ||||||
| 		"build": "tsc -p tsconfig.json || echo done. && tsc-alias -p tsconfig.json", | 		"build:tsc": "tsc -p tsconfig.json && tsc-alias -p tsconfig.json", | ||||||
| 		"watch": "node watch.mjs", | 		"watch": "node watch.mjs", | ||||||
| 		"lint": "tsc --noEmit && eslint --quiet \"src/**/*.ts\"", | 		"typecheck": "tsc --noEmit", | ||||||
|  | 		"eslint": "eslint --quiet \"src/**/*.ts\"", | ||||||
|  | 		"lint": "pnpm typecheck && pnpm eslint", | ||||||
| 		"jest": "cross-env NODE_ENV=test node --experimental-vm-modules --experimental-import-meta-resolve node_modules/jest/bin/jest.js --forceExit --runInBand", | 		"jest": "cross-env NODE_ENV=test node --experimental-vm-modules --experimental-import-meta-resolve node_modules/jest/bin/jest.js --forceExit --runInBand", | ||||||
| 		"jest-and-coverage": "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": "cross-env NODE_ENV=test node --experimental-vm-modules --experimental-import-meta-resolve node_modules/jest/bin/jest.js --coverage --forceExit --runInBand", | ||||||
| 		"jest-clear": "cross-env NODE_ENV=test node --experimental-vm-modules --experimental-import-meta-resolve node_modules/jest/bin/jest.js --clearCache", | 		"jest-clear": "cross-env NODE_ENV=test node --experimental-vm-modules --experimental-import-meta-resolve node_modules/jest/bin/jest.js --clearCache", | ||||||
| @@ -19,34 +21,47 @@ | |||||||
| 		"test-and-coverage": "pnpm jest-and-coverage" | 		"test-and-coverage": "pnpm jest-and-coverage" | ||||||
| 	}, | 	}, | ||||||
| 	"optionalDependencies": { | 	"optionalDependencies": { | ||||||
|  | 		"@swc/core-android-arm64": "^1.3.11", | ||||||
|  | 		"@swc/core-darwin-arm64": "^1.3.36", | ||||||
|  | 		"@swc/core-darwin-x64": "^1.3.36", | ||||||
|  | 		"@swc/core-linux-arm-gnueabihf": "^1.3.36", | ||||||
|  | 		"@swc/core-linux-arm64-gnu": "^1.3.36", | ||||||
|  | 		"@swc/core-linux-arm64-musl": "^1.3.36", | ||||||
|  | 		"@swc/core-linux-x64-gnu": "^1.3.36", | ||||||
|  | 		"@swc/core-linux-x64-musl": "^1.3.36", | ||||||
|  | 		"@swc/core-win32-arm64-msvc": "^1.3.36", | ||||||
|  | 		"@swc/core-win32-ia32-msvc": "^1.3.36", | ||||||
|  | 		"@swc/core-win32-x64-msvc": "^1.3.36", | ||||||
| 		"@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.1", | 		"@bull-board/api": "4.12.1", | ||||||
| 		"@bull-board/fastify": "4.11.1", | 		"@bull-board/fastify": "4.12.1", | ||||||
| 		"@bull-board/ui": "4.11.1", | 		"@bull-board/ui": "4.12.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.1", | ||||||
| 		"@fastify/static": "6.8.0", | 		"@fastify/static": "6.9.0", | ||||||
| 		"@fastify/view": "7.4.1", | 		"@fastify/view": "7.4.1", | ||||||
| 		"@nestjs/common": "9.3.7", | 		"@nestjs/common": "9.3.9", | ||||||
| 		"@nestjs/core": "9.3.7", | 		"@nestjs/core": "9.3.9", | ||||||
| 		"@nestjs/testing": "9.3.7", | 		"@nestjs/testing": "9.3.9", | ||||||
| 		"@peertube/http-signature": "1.7.0", | 		"@peertube/http-signature": "1.7.0", | ||||||
| 		"@sinonjs/fake-timers": "10.0.2", | 		"@sinonjs/fake-timers": "10.0.2", | ||||||
|  | 		"@swc/cli": "0.1.62", | ||||||
|  | 		"@swc/core": "1.3.35", | ||||||
| 		"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.1318.0", | ||||||
| 		"bcryptjs": "2.4.3", | 		"bcryptjs": "2.4.3", | ||||||
| 		"blurhash": "2.0.4", | 		"blurhash": "2.0.5", | ||||||
| 		"bull": "4.10.3", | 		"bull": "4.10.4", | ||||||
| 		"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", | ||||||
| @@ -58,12 +73,13 @@ | |||||||
| 		"date-fns": "2.29.3", | 		"date-fns": "2.29.3", | ||||||
| 		"deep-email-validator": "0.1.21", | 		"deep-email-validator": "0.1.21", | ||||||
| 		"escape-regexp": "0.0.1", | 		"escape-regexp": "0.0.1", | ||||||
| 		"fastify": "4.12.0", | 		"fastify": "4.13.0", | ||||||
| 		"feed": "4.2.2", | 		"feed": "4.2.2", | ||||||
| 		"file-type": "18.2.0", | 		"file-type": "18.2.1", | ||||||
| 		"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", | ||||||
|  | 		"happy-dom": "^8.7.0", | ||||||
| 		"hpagent": "1.2.0", | 		"hpagent": "1.2.0", | ||||||
| 		"ioredis": "4.28.5", | 		"ioredis": "4.28.5", | ||||||
| 		"ip-cidr": "3.1.0", | 		"ip-cidr": "3.1.0", | ||||||
| @@ -83,6 +99,7 @@ | |||||||
| 		"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", | ||||||
|  | 		"otpauth": "^9.0.2", | ||||||
| 		"parse5": "7.1.2", | 		"parse5": "7.1.2", | ||||||
| 		"pg": "8.9.0", | 		"pg": "8.9.0", | ||||||
| 		"private-ip": "3.0.0", | 		"private-ip": "3.0.0", | ||||||
| @@ -102,21 +119,20 @@ | |||||||
| 		"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.9.0", | 		"sanitize-html": "2.10.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", |  | ||||||
| 		"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": "github:misskey-dev/summaly", | ||||||
| 		"systeminformation": "5.17.8", | 		"systeminformation": "5.17.9", | ||||||
| 		"tinycolor2": "1.6.0", | 		"tinycolor2": "1.6.0", | ||||||
| 		"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.12", | 		"typeorm": "0.3.11", | ||||||
| 		"typescript": "4.9.5", | 		"typescript": "4.9.5", | ||||||
| 		"ulid": "2.3.0", | 		"ulid": "2.3.0", | ||||||
| 		"unzipper": "0.10.11", | 		"unzipper": "0.10.11", | ||||||
| @@ -124,14 +140,12 @@ | |||||||
| 		"vary": "1.1.2", | 		"vary": "1.1.2", | ||||||
| 		"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.1", | ||||||
| 		"xev": "3.0.2" | 		"xev": "3.0.2" | ||||||
| 	}, | 	}, | ||||||
| 	"devDependencies": { | 	"devDependencies": { | ||||||
| 		"@jest/globals": "29.4.2", | 		"@jest/globals": "29.4.3", | ||||||
| 		"@redocly/openapi-core": "1.0.0-beta.123", | 		"@redocly/openapi-core": "1.0.0-beta.123", | ||||||
| 		"@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", | ||||||
| @@ -149,7 +163,7 @@ | |||||||
| 		"@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.13.0", | 		"@types/node": "18.14.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", | ||||||
| @@ -165,7 +179,6 @@ | |||||||
| 		"@types/semver": "7.3.13", | 		"@types/semver": "7.3.13", | ||||||
| 		"@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/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", | ||||||
| @@ -174,13 +187,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.51.0", | 		"@typescript-eslint/eslint-plugin": "5.52.0", | ||||||
| 		"@typescript-eslint/parser": "5.51.0", | 		"@typescript-eslint/parser": "5.52.0", | ||||||
| 		"cross-env": "7.0.3", | 		"cross-env": "7.0.3", | ||||||
| 		"eslint": "8.33.0", | 		"eslint": "8.34.0", | ||||||
| 		"eslint-plugin-import": "2.27.5", | 		"eslint-plugin-import": "2.27.5", | ||||||
| 		"execa": "6.1.0", | 		"execa": "6.1.0", | ||||||
| 		"jest": "29.4.2", | 		"jest": "29.4.3", | ||||||
| 		"jest-mock": "29.4.2" | 		"jest-mock": "29.4.3" | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
							
								
								
									
										8
									
								
								packages/backend/src/@types/redis-lock.d.ts
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								packages/backend/src/@types/redis-lock.d.ts
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,8 @@ | |||||||
|  | declare module 'redis-lock' { | ||||||
|  | 	import type Redis from 'ioredis'; | ||||||
|  |  | ||||||
|  | 	type Lock = (lockName: string, timeout?: number, taskToPerform?: () => Promise<void>) => void; | ||||||
|  | 	function redisLock(client: Redis.Redis, retryDelay: number): Lock; | ||||||
|  |  | ||||||
|  | 	export = redisLock; | ||||||
|  | } | ||||||
| @@ -67,6 +67,7 @@ export type Source = { | |||||||
|  |  | ||||||
| 	mediaProxy?: string; | 	mediaProxy?: string; | ||||||
| 	proxyRemoteFiles?: boolean; | 	proxyRemoteFiles?: boolean; | ||||||
|  | 	videoThumbnailGenerator?: string; | ||||||
|  |  | ||||||
| 	signToActivityPubGet?: boolean; | 	signToActivityPubGet?: boolean; | ||||||
| }; | }; | ||||||
| @@ -89,6 +90,7 @@ export type Mixin = { | |||||||
| 	clientManifestExists: boolean; | 	clientManifestExists: boolean; | ||||||
| 	mediaProxy: string; | 	mediaProxy: string; | ||||||
| 	externalMediaProxyEnabled: boolean; | 	externalMediaProxyEnabled: boolean; | ||||||
|  | 	videoThumbnailGenerator: string | null; | ||||||
| }; | }; | ||||||
|  |  | ||||||
| export type Config = Source & Mixin; | export type Config = Source & Mixin; | ||||||
| @@ -144,6 +146,10 @@ export function loadConfig() { | |||||||
| 	mixin.mediaProxy = externalMediaProxy ?? internalMediaProxy; | 	mixin.mediaProxy = externalMediaProxy ?? internalMediaProxy; | ||||||
| 	mixin.externalMediaProxyEnabled = externalMediaProxy !== null && 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); | ||||||
|   | |||||||
| @@ -32,7 +32,7 @@ export class AccountUpdateService { | |||||||
| 	 | 	 | ||||||
| 		// フォロワーがリモートユーザーかつ投稿者がローカルユーザーならUpdateを配信 | 		// フォロワーがリモートユーザーかつ投稿者がローカルユーザーならUpdateを配信 | ||||||
| 		if (this.userEntityService.isLocalUser(user)) { | 		if (this.userEntityService.isLocalUser(user)) { | ||||||
| 			const content = this.apRendererService.renderActivity(this.apRendererService.renderUpdate(await this.apRendererService.renderPerson(user), user)); | 			const content = this.apRendererService.addContext(this.apRendererService.renderUpdate(await this.apRendererService.renderPerson(user), user)); | ||||||
| 			this.apDeliverManagerService.deliverToFollowers(user, content); | 			this.apDeliverManagerService.deliverToFollowers(user, content); | ||||||
| 			this.relayService.deliverToRelays(user, content); | 			this.relayService.deliverToRelays(user, content); | ||||||
| 		} | 		} | ||||||
|   | |||||||
| @@ -12,7 +12,7 @@ import { PushNotificationService } from '@/core/PushNotificationService.js'; | |||||||
| import * as Acct from '@/misc/acct.js'; | import * as Acct from '@/misc/acct.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, NotesRepository, AntennaNotesRepository, AntennasRepository, UserGroupJoiningsRepository, UserListJoiningsRepository } from '@/models/index.js'; | import type { MutingsRepository, NotesRepository, AntennaNotesRepository, AntennasRepository, 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'; | ||||||
| @@ -39,9 +39,6 @@ export class AntennaService implements OnApplicationShutdown { | |||||||
| 		@Inject(DI.antennasRepository) | 		@Inject(DI.antennasRepository) | ||||||
| 		private antennasRepository: AntennasRepository, | 		private antennasRepository: AntennasRepository, | ||||||
|  |  | ||||||
| 		@Inject(DI.userGroupJoiningsRepository) |  | ||||||
| 		private userGroupJoiningsRepository: UserGroupJoiningsRepository, |  | ||||||
|  |  | ||||||
| 		@Inject(DI.userListJoiningsRepository) | 		@Inject(DI.userListJoiningsRepository) | ||||||
| 		private userListJoiningsRepository: UserListJoiningsRepository, | 		private userListJoiningsRepository: UserListJoiningsRepository, | ||||||
|  |  | ||||||
| @@ -160,14 +157,6 @@ export class AntennaService implements OnApplicationShutdown { | |||||||
| 			})).map(x => x.userId); | 			})).map(x => x.userId); | ||||||
| 	 | 	 | ||||||
| 			if (!listUsers.includes(note.userId)) return false; | 			if (!listUsers.includes(note.userId)) return false; | ||||||
| 		} else if (antenna.src === 'group') { |  | ||||||
| 			const joining = await this.userGroupJoiningsRepository.findOneByOrFail({ id: antenna.userGroupJoiningId! }); |  | ||||||
| 	 |  | ||||||
| 			const groupUsers = (await this.userGroupJoiningsRepository.findBy({ |  | ||||||
| 				userGroupId: joining.userGroupId, |  | ||||||
| 			})).map(x => x.userId); |  | ||||||
| 	 |  | ||||||
| 			if (!groupUsers.includes(note.userId)) return false; |  | ||||||
| 		} else if (antenna.src === 'users') { | 		} else if (antenna.src === 'users') { | ||||||
| 			const accts = antenna.users.map(x => { | 			const accts = antenna.users.map(x => { | ||||||
| 				const { username, host } = Acct.parse(x); | 				const { username, host } = Acct.parse(x); | ||||||
|   | |||||||
| @@ -12,7 +12,7 @@ const retryDelay = 100; | |||||||
|  |  | ||||||
| @Injectable() | @Injectable() | ||||||
| export class AppLockService { | export class AppLockService { | ||||||
| 	private lock: (key: string, timeout?: number) => Promise<() => void>; | 	private lock: (key: string, timeout?: number, _?: (() => Promise<void>) | undefined) => Promise<() => void>; | ||||||
|  |  | ||||||
| 	constructor( | 	constructor( | ||||||
| 		@Inject(DI.redis) | 		@Inject(DI.redis) | ||||||
|   | |||||||
| @@ -1,5 +1,4 @@ | |||||||
| import { Module } from '@nestjs/common'; | import { Module } from '@nestjs/common'; | ||||||
| import { DI } from '../di-symbols.js'; |  | ||||||
| import { AccountUpdateService } from './AccountUpdateService.js'; | import { AccountUpdateService } from './AccountUpdateService.js'; | ||||||
| import { AiService } from './AiService.js'; | import { AiService } from './AiService.js'; | ||||||
| import { AntennaService } from './AntennaService.js'; | import { AntennaService } from './AntennaService.js'; | ||||||
| @@ -22,7 +21,6 @@ import { IdService } from './IdService.js'; | |||||||
| import { ImageProcessingService } from './ImageProcessingService.js'; | import { ImageProcessingService } from './ImageProcessingService.js'; | ||||||
| import { InstanceActorService } from './InstanceActorService.js'; | import { InstanceActorService } from './InstanceActorService.js'; | ||||||
| import { InternalStorageService } from './InternalStorageService.js'; | import { InternalStorageService } from './InternalStorageService.js'; | ||||||
| import { MessagingService } from './MessagingService.js'; |  | ||||||
| import { MetaService } from './MetaService.js'; | import { MetaService } from './MetaService.js'; | ||||||
| import { MfmService } from './MfmService.js'; | import { MfmService } from './MfmService.js'; | ||||||
| import { ModerationLogService } from './ModerationLogService.js'; | import { ModerationLogService } from './ModerationLogService.js'; | ||||||
| @@ -82,7 +80,6 @@ import { GalleryLikeEntityService } from './entities/GalleryLikeEntityService.js | |||||||
| import { GalleryPostEntityService } from './entities/GalleryPostEntityService.js'; | import { GalleryPostEntityService } from './entities/GalleryPostEntityService.js'; | ||||||
| import { HashtagEntityService } from './entities/HashtagEntityService.js'; | import { HashtagEntityService } from './entities/HashtagEntityService.js'; | ||||||
| import { InstanceEntityService } from './entities/InstanceEntityService.js'; | import { InstanceEntityService } from './entities/InstanceEntityService.js'; | ||||||
| import { MessagingMessageEntityService } from './entities/MessagingMessageEntityService.js'; |  | ||||||
| import { ModerationLogEntityService } from './entities/ModerationLogEntityService.js'; | import { ModerationLogEntityService } from './entities/ModerationLogEntityService.js'; | ||||||
| import { MutingEntityService } from './entities/MutingEntityService.js'; | import { MutingEntityService } from './entities/MutingEntityService.js'; | ||||||
| import { NoteEntityService } from './entities/NoteEntityService.js'; | import { NoteEntityService } from './entities/NoteEntityService.js'; | ||||||
| @@ -93,8 +90,6 @@ import { PageEntityService } from './entities/PageEntityService.js'; | |||||||
| import { PageLikeEntityService } from './entities/PageLikeEntityService.js'; | import { PageLikeEntityService } from './entities/PageLikeEntityService.js'; | ||||||
| import { SigninEntityService } from './entities/SigninEntityService.js'; | import { SigninEntityService } from './entities/SigninEntityService.js'; | ||||||
| import { UserEntityService } from './entities/UserEntityService.js'; | import { UserEntityService } from './entities/UserEntityService.js'; | ||||||
| import { UserGroupEntityService } from './entities/UserGroupEntityService.js'; |  | ||||||
| import { UserGroupInvitationEntityService } from './entities/UserGroupInvitationEntityService.js'; |  | ||||||
| import { UserListEntityService } from './entities/UserListEntityService.js'; | import { UserListEntityService } from './entities/UserListEntityService.js'; | ||||||
| import { FlashEntityService } from './entities/FlashEntityService.js'; | import { FlashEntityService } from './entities/FlashEntityService.js'; | ||||||
| import { FlashLikeEntityService } from './entities/FlashLikeEntityService.js'; | import { FlashLikeEntityService } from './entities/FlashLikeEntityService.js'; | ||||||
| @@ -146,7 +141,6 @@ const $IdService: Provider = { provide: 'IdService', useExisting: IdService }; | |||||||
| const $ImageProcessingService: Provider = { provide: 'ImageProcessingService', useExisting: ImageProcessingService }; | const $ImageProcessingService: Provider = { provide: 'ImageProcessingService', useExisting: ImageProcessingService }; | ||||||
| const $InstanceActorService: Provider = { provide: 'InstanceActorService', useExisting: InstanceActorService }; | const $InstanceActorService: Provider = { provide: 'InstanceActorService', useExisting: InstanceActorService }; | ||||||
| const $InternalStorageService: Provider = { provide: 'InternalStorageService', useExisting: InternalStorageService }; | const $InternalStorageService: Provider = { provide: 'InternalStorageService', useExisting: InternalStorageService }; | ||||||
| const $MessagingService: Provider = { provide: 'MessagingService', useExisting: MessagingService }; |  | ||||||
| const $MetaService: Provider = { provide: 'MetaService', useExisting: MetaService }; | const $MetaService: Provider = { provide: 'MetaService', useExisting: MetaService }; | ||||||
| const $MfmService: Provider = { provide: 'MfmService', useExisting: MfmService }; | const $MfmService: Provider = { provide: 'MfmService', useExisting: MfmService }; | ||||||
| const $ModerationLogService: Provider = { provide: 'ModerationLogService', useExisting: ModerationLogService }; | const $ModerationLogService: Provider = { provide: 'ModerationLogService', useExisting: ModerationLogService }; | ||||||
| @@ -207,7 +201,6 @@ const $GalleryLikeEntityService: Provider = { provide: 'GalleryLikeEntityService | |||||||
| const $GalleryPostEntityService: Provider = { provide: 'GalleryPostEntityService', useExisting: GalleryPostEntityService }; | const $GalleryPostEntityService: Provider = { provide: 'GalleryPostEntityService', useExisting: GalleryPostEntityService }; | ||||||
| const $HashtagEntityService: Provider = { provide: 'HashtagEntityService', useExisting: HashtagEntityService }; | const $HashtagEntityService: Provider = { provide: 'HashtagEntityService', useExisting: HashtagEntityService }; | ||||||
| const $InstanceEntityService: Provider = { provide: 'InstanceEntityService', useExisting: InstanceEntityService }; | const $InstanceEntityService: Provider = { provide: 'InstanceEntityService', useExisting: InstanceEntityService }; | ||||||
| const $MessagingMessageEntityService: Provider = { provide: 'MessagingMessageEntityService', useExisting: MessagingMessageEntityService }; |  | ||||||
| const $ModerationLogEntityService: Provider = { provide: 'ModerationLogEntityService', useExisting: ModerationLogEntityService }; | const $ModerationLogEntityService: Provider = { provide: 'ModerationLogEntityService', useExisting: ModerationLogEntityService }; | ||||||
| const $MutingEntityService: Provider = { provide: 'MutingEntityService', useExisting: MutingEntityService }; | const $MutingEntityService: Provider = { provide: 'MutingEntityService', useExisting: MutingEntityService }; | ||||||
| const $NoteEntityService: Provider = { provide: 'NoteEntityService', useExisting: NoteEntityService }; | const $NoteEntityService: Provider = { provide: 'NoteEntityService', useExisting: NoteEntityService }; | ||||||
| @@ -218,8 +211,6 @@ const $PageEntityService: Provider = { provide: 'PageEntityService', useExisting | |||||||
| const $PageLikeEntityService: Provider = { provide: 'PageLikeEntityService', useExisting: PageLikeEntityService }; | const $PageLikeEntityService: Provider = { provide: 'PageLikeEntityService', useExisting: PageLikeEntityService }; | ||||||
| const $SigninEntityService: Provider = { provide: 'SigninEntityService', useExisting: SigninEntityService }; | const $SigninEntityService: Provider = { provide: 'SigninEntityService', useExisting: SigninEntityService }; | ||||||
| const $UserEntityService: Provider = { provide: 'UserEntityService', useExisting: UserEntityService }; | const $UserEntityService: Provider = { provide: 'UserEntityService', useExisting: UserEntityService }; | ||||||
| const $UserGroupEntityService: Provider = { provide: 'UserGroupEntityService', useExisting: UserGroupEntityService }; |  | ||||||
| const $UserGroupInvitationEntityService: Provider = { provide: 'UserGroupInvitationEntityService', useExisting: UserGroupInvitationEntityService }; |  | ||||||
| const $UserListEntityService: Provider = { provide: 'UserListEntityService', useExisting: UserListEntityService }; | const $UserListEntityService: Provider = { provide: 'UserListEntityService', useExisting: UserListEntityService }; | ||||||
| const $FlashEntityService: Provider = { provide: 'FlashEntityService', useExisting: FlashEntityService }; | const $FlashEntityService: Provider = { provide: 'FlashEntityService', useExisting: FlashEntityService }; | ||||||
| const $FlashLikeEntityService: Provider = { provide: 'FlashLikeEntityService', useExisting: FlashLikeEntityService }; | const $FlashLikeEntityService: Provider = { provide: 'FlashLikeEntityService', useExisting: FlashLikeEntityService }; | ||||||
| @@ -273,7 +264,6 @@ const $ApQuestionService: Provider = { provide: 'ApQuestionService', useExisting | |||||||
| 		ImageProcessingService, | 		ImageProcessingService, | ||||||
| 		InstanceActorService, | 		InstanceActorService, | ||||||
| 		InternalStorageService, | 		InternalStorageService, | ||||||
| 		MessagingService, |  | ||||||
| 		MetaService, | 		MetaService, | ||||||
| 		MfmService, | 		MfmService, | ||||||
| 		ModerationLogService, | 		ModerationLogService, | ||||||
| @@ -333,7 +323,6 @@ const $ApQuestionService: Provider = { provide: 'ApQuestionService', useExisting | |||||||
| 		GalleryPostEntityService, | 		GalleryPostEntityService, | ||||||
| 		HashtagEntityService, | 		HashtagEntityService, | ||||||
| 		InstanceEntityService, | 		InstanceEntityService, | ||||||
| 		MessagingMessageEntityService, |  | ||||||
| 		ModerationLogEntityService, | 		ModerationLogEntityService, | ||||||
| 		MutingEntityService, | 		MutingEntityService, | ||||||
| 		NoteEntityService, | 		NoteEntityService, | ||||||
| @@ -344,8 +333,6 @@ const $ApQuestionService: Provider = { provide: 'ApQuestionService', useExisting | |||||||
| 		PageLikeEntityService, | 		PageLikeEntityService, | ||||||
| 		SigninEntityService, | 		SigninEntityService, | ||||||
| 		UserEntityService, | 		UserEntityService, | ||||||
| 		UserGroupEntityService, |  | ||||||
| 		UserGroupInvitationEntityService, |  | ||||||
| 		UserListEntityService, | 		UserListEntityService, | ||||||
| 		FlashEntityService, | 		FlashEntityService, | ||||||
| 		FlashLikeEntityService, | 		FlashLikeEntityService, | ||||||
| @@ -394,7 +381,6 @@ const $ApQuestionService: Provider = { provide: 'ApQuestionService', useExisting | |||||||
| 		$ImageProcessingService, | 		$ImageProcessingService, | ||||||
| 		$InstanceActorService, | 		$InstanceActorService, | ||||||
| 		$InternalStorageService, | 		$InternalStorageService, | ||||||
| 		$MessagingService, |  | ||||||
| 		$MetaService, | 		$MetaService, | ||||||
| 		$MfmService, | 		$MfmService, | ||||||
| 		$ModerationLogService, | 		$ModerationLogService, | ||||||
| @@ -454,7 +440,6 @@ const $ApQuestionService: Provider = { provide: 'ApQuestionService', useExisting | |||||||
| 		$GalleryPostEntityService, | 		$GalleryPostEntityService, | ||||||
| 		$HashtagEntityService, | 		$HashtagEntityService, | ||||||
| 		$InstanceEntityService, | 		$InstanceEntityService, | ||||||
| 		$MessagingMessageEntityService, |  | ||||||
| 		$ModerationLogEntityService, | 		$ModerationLogEntityService, | ||||||
| 		$MutingEntityService, | 		$MutingEntityService, | ||||||
| 		$NoteEntityService, | 		$NoteEntityService, | ||||||
| @@ -465,8 +450,6 @@ const $ApQuestionService: Provider = { provide: 'ApQuestionService', useExisting | |||||||
| 		$PageLikeEntityService, | 		$PageLikeEntityService, | ||||||
| 		$SigninEntityService, | 		$SigninEntityService, | ||||||
| 		$UserEntityService, | 		$UserEntityService, | ||||||
| 		$UserGroupEntityService, |  | ||||||
| 		$UserGroupInvitationEntityService, |  | ||||||
| 		$UserListEntityService, | 		$UserListEntityService, | ||||||
| 		$FlashEntityService, | 		$FlashEntityService, | ||||||
| 		$FlashLikeEntityService, | 		$FlashLikeEntityService, | ||||||
| @@ -516,7 +499,6 @@ const $ApQuestionService: Provider = { provide: 'ApQuestionService', useExisting | |||||||
| 		ImageProcessingService, | 		ImageProcessingService, | ||||||
| 		InstanceActorService, | 		InstanceActorService, | ||||||
| 		InternalStorageService, | 		InternalStorageService, | ||||||
| 		MessagingService, |  | ||||||
| 		MetaService, | 		MetaService, | ||||||
| 		MfmService, | 		MfmService, | ||||||
| 		ModerationLogService, | 		ModerationLogService, | ||||||
| @@ -575,7 +557,6 @@ const $ApQuestionService: Provider = { provide: 'ApQuestionService', useExisting | |||||||
| 		GalleryPostEntityService, | 		GalleryPostEntityService, | ||||||
| 		HashtagEntityService, | 		HashtagEntityService, | ||||||
| 		InstanceEntityService, | 		InstanceEntityService, | ||||||
| 		MessagingMessageEntityService, |  | ||||||
| 		ModerationLogEntityService, | 		ModerationLogEntityService, | ||||||
| 		MutingEntityService, | 		MutingEntityService, | ||||||
| 		NoteEntityService, | 		NoteEntityService, | ||||||
| @@ -586,8 +567,6 @@ const $ApQuestionService: Provider = { provide: 'ApQuestionService', useExisting | |||||||
| 		PageLikeEntityService, | 		PageLikeEntityService, | ||||||
| 		SigninEntityService, | 		SigninEntityService, | ||||||
| 		UserEntityService, | 		UserEntityService, | ||||||
| 		UserGroupEntityService, |  | ||||||
| 		UserGroupInvitationEntityService, |  | ||||||
| 		UserListEntityService, | 		UserListEntityService, | ||||||
| 		FlashEntityService, | 		FlashEntityService, | ||||||
| 		FlashLikeEntityService, | 		FlashLikeEntityService, | ||||||
| @@ -636,7 +615,6 @@ const $ApQuestionService: Provider = { provide: 'ApQuestionService', useExisting | |||||||
| 		$ImageProcessingService, | 		$ImageProcessingService, | ||||||
| 		$InstanceActorService, | 		$InstanceActorService, | ||||||
| 		$InternalStorageService, | 		$InternalStorageService, | ||||||
| 		$MessagingService, |  | ||||||
| 		$MetaService, | 		$MetaService, | ||||||
| 		$MfmService, | 		$MfmService, | ||||||
| 		$ModerationLogService, | 		$ModerationLogService, | ||||||
| @@ -695,7 +673,6 @@ const $ApQuestionService: Provider = { provide: 'ApQuestionService', useExisting | |||||||
| 		$GalleryPostEntityService, | 		$GalleryPostEntityService, | ||||||
| 		$HashtagEntityService, | 		$HashtagEntityService, | ||||||
| 		$InstanceEntityService, | 		$InstanceEntityService, | ||||||
| 		$MessagingMessageEntityService, |  | ||||||
| 		$ModerationLogEntityService, | 		$ModerationLogEntityService, | ||||||
| 		$MutingEntityService, | 		$MutingEntityService, | ||||||
| 		$NoteEntityService, | 		$NoteEntityService, | ||||||
| @@ -706,8 +683,6 @@ const $ApQuestionService: Provider = { provide: 'ApQuestionService', useExisting | |||||||
| 		$PageLikeEntityService, | 		$PageLikeEntityService, | ||||||
| 		$SigninEntityService, | 		$SigninEntityService, | ||||||
| 		$UserEntityService, | 		$UserEntityService, | ||||||
| 		$UserGroupEntityService, |  | ||||||
| 		$UserGroupInvitationEntityService, |  | ||||||
| 		$UserListEntityService, | 		$UserListEntityService, | ||||||
| 		$FlashEntityService, | 		$FlashEntityService, | ||||||
| 		$FlashLikeEntityService, | 		$FlashLikeEntityService, | ||||||
|   | |||||||
| @@ -61,7 +61,7 @@ export class CustomEmojiService { | |||||||
| 			await this.db.queryResultCache!.remove(['meta_emojis']); | 			await this.db.queryResultCache!.remove(['meta_emojis']); | ||||||
|  |  | ||||||
| 			this.globalEventService.publishBroadcastStream('emojiAdded', { | 			this.globalEventService.publishBroadcastStream('emojiAdded', { | ||||||
| 				emoji: await this.emojiEntityService.pack(emoji.id), | 				emoji: await this.emojiEntityService.packDetailed(emoji.id), | ||||||
| 			}); | 			}); | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
|   | |||||||
| @@ -7,7 +7,7 @@ import { DI } from '@/di-symbols.js'; | |||||||
| import type { DriveFilesRepository, UsersRepository, DriveFoldersRepository, UserProfilesRepository } from '@/models/index.js'; | import type { DriveFilesRepository, UsersRepository, DriveFoldersRepository, UserProfilesRepository } from '@/models/index.js'; | ||||||
| import type { Config } from '@/config.js'; | import type { Config } from '@/config.js'; | ||||||
| import Logger from '@/logger.js'; | import Logger from '@/logger.js'; | ||||||
| import type { IRemoteUser, User } from '@/models/entities/User.js'; | import type { RemoteUser, User } from '@/models/entities/User.js'; | ||||||
| import { MetaService } from '@/core/MetaService.js'; | import { MetaService } from '@/core/MetaService.js'; | ||||||
| import { DriveFile } from '@/models/entities/DriveFile.js'; | import { DriveFile } from '@/models/entities/DriveFile.js'; | ||||||
| import { IdService } from '@/core/IdService.js'; | import { IdService } from '@/core/IdService.js'; | ||||||
| @@ -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 { | ||||||
| @@ -391,7 +399,7 @@ export class DriveService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	private async deleteOldFile(user: IRemoteUser) { | 	private async deleteOldFile(user: RemoteUser) { | ||||||
| 		const q = this.driveFilesRepository.createQueryBuilder('file') | 		const q = this.driveFilesRepository.createQueryBuilder('file') | ||||||
| 			.where('file.userId = :userId', { userId: user.id }) | 			.where('file.userId = :userId', { userId: user.id }) | ||||||
| 			.andWhere('file.isLink = FALSE'); | 			.andWhere('file.isLink = FALSE'); | ||||||
| @@ -492,7 +500,7 @@ export class DriveService { | |||||||
| 					throw new IdentifiableError('c6244ed2-a39a-4e1c-bf93-f0fbd7764fa6', 'No free space.'); | 					throw new IdentifiableError('c6244ed2-a39a-4e1c-bf93-f0fbd7764fa6', 'No free space.'); | ||||||
| 				} else { | 				} else { | ||||||
| 				// (アバターまたはバナーを含まず)最も古いファイルを削除する | 				// (アバターまたはバナーを含まず)最も古いファイルを削除する | ||||||
| 					this.deleteOldFile(await this.usersRepository.findOneByOrFail({ id: user.id }) as IRemoteUser); | 					this.deleteOldFile(await this.usersRepository.findOneByOrFail({ id: user.id }) as RemoteUser); | ||||||
| 				} | 				} | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
|   | |||||||
| @@ -2,7 +2,6 @@ import { URL } from 'node:url'; | |||||||
| import { Inject, Injectable } from '@nestjs/common'; | import { Inject, Injectable } from '@nestjs/common'; | ||||||
| import { JSDOM } from 'jsdom'; | import { JSDOM } from 'jsdom'; | ||||||
| import tinycolor from 'tinycolor2'; | import tinycolor from 'tinycolor2'; | ||||||
| import fetch from 'node-fetch'; |  | ||||||
| import type { Instance } from '@/models/entities/Instance.js'; | import type { Instance } from '@/models/entities/Instance.js'; | ||||||
| import type { InstancesRepository } from '@/models/index.js'; | import type { InstancesRepository } from '@/models/index.js'; | ||||||
| import { AppLockService } from '@/core/AppLockService.js'; | import { AppLockService } from '@/core/AppLockService.js'; | ||||||
|   | |||||||
| @@ -3,7 +3,7 @@ import * as crypto from 'node:crypto'; | |||||||
| import { join } from 'node:path'; | import { join } from 'node:path'; | ||||||
| import * as stream from 'node:stream'; | import * as stream from 'node:stream'; | ||||||
| import * as util from 'node:util'; | import * as util from 'node:util'; | ||||||
| import { Inject, Injectable } from '@nestjs/common'; | import { Injectable } from '@nestjs/common'; | ||||||
| import { FSWatcher } from 'chokidar'; | import { FSWatcher } from 'chokidar'; | ||||||
| import { fileTypeFromFile } from 'file-type'; | import { fileTypeFromFile } from 'file-type'; | ||||||
| import FFmpeg from 'fluent-ffmpeg'; | import FFmpeg from 'fluent-ffmpeg'; | ||||||
|   | |||||||
| @@ -3,21 +3,15 @@ import Redis from 'ioredis'; | |||||||
| import type { User } from '@/models/entities/User.js'; | import type { User } from '@/models/entities/User.js'; | ||||||
| import type { Note } from '@/models/entities/Note.js'; | import type { Note } from '@/models/entities/Note.js'; | ||||||
| import type { UserList } from '@/models/entities/UserList.js'; | import type { UserList } from '@/models/entities/UserList.js'; | ||||||
| import type { UserGroup } from '@/models/entities/UserGroup.js'; |  | ||||||
| import type { Antenna } from '@/models/entities/Antenna.js'; | import type { Antenna } from '@/models/entities/Antenna.js'; | ||||||
| import type { Channel } from '@/models/entities/Channel.js'; |  | ||||||
| import type { | import type { | ||||||
| 	StreamChannels, | 	StreamChannels, | ||||||
| 	AdminStreamTypes, | 	AdminStreamTypes, | ||||||
| 	AntennaStreamTypes, | 	AntennaStreamTypes, | ||||||
| 	BroadcastTypes, | 	BroadcastTypes, | ||||||
| 	ChannelStreamTypes, |  | ||||||
| 	DriveStreamTypes, | 	DriveStreamTypes, | ||||||
| 	GroupMessagingStreamTypes, |  | ||||||
| 	InternalStreamTypes, | 	InternalStreamTypes, | ||||||
| 	MainStreamTypes, | 	MainStreamTypes, | ||||||
| 	MessagingIndexStreamTypes, |  | ||||||
| 	MessagingStreamTypes, |  | ||||||
| 	NoteStreamTypes, | 	NoteStreamTypes, | ||||||
| 	UserListStreamTypes, | 	UserListStreamTypes, | ||||||
| 	UserStreamTypes, | 	UserStreamTypes, | ||||||
| @@ -83,11 +77,6 @@ export class GlobalEventService { | |||||||
| 		}); | 		}); | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis |  | ||||||
| 	public publishChannelStream<K extends keyof ChannelStreamTypes>(channelId: Channel['id'], type: K, value?: ChannelStreamTypes[K]): void { |  | ||||||
| 		this.publish(`channelStream:${channelId}`, type, typeof value === 'undefined' ? null : value); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public publishUserListStream<K extends keyof UserListStreamTypes>(listId: UserList['id'], type: K, value?: UserListStreamTypes[K]): void { | 	public publishUserListStream<K extends keyof UserListStreamTypes>(listId: UserList['id'], type: K, value?: UserListStreamTypes[K]): void { | ||||||
| 		this.publish(`userListStream:${listId}`, type, typeof value === 'undefined' ? null : value); | 		this.publish(`userListStream:${listId}`, type, typeof value === 'undefined' ? null : value); | ||||||
| @@ -98,21 +87,6 @@ export class GlobalEventService { | |||||||
| 		this.publish(`antennaStream:${antennaId}`, type, typeof value === 'undefined' ? null : value); | 		this.publish(`antennaStream:${antennaId}`, type, typeof value === 'undefined' ? null : value); | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis |  | ||||||
| 	public publishMessagingStream<K extends keyof MessagingStreamTypes>(userId: User['id'], otherpartyId: User['id'], type: K, value?: MessagingStreamTypes[K]): void { |  | ||||||
| 		this.publish(`messagingStream:${userId}-${otherpartyId}`, type, typeof value === 'undefined' ? null : value); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	@bindThis |  | ||||||
| 	public publishGroupMessagingStream<K extends keyof GroupMessagingStreamTypes>(groupId: UserGroup['id'], type: K, value?: GroupMessagingStreamTypes[K]): void { |  | ||||||
| 		this.publish(`messagingStream:${groupId}`, type, typeof value === 'undefined' ? null : value); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	@bindThis |  | ||||||
| 	public publishMessagingIndexStream<K extends keyof MessagingIndexStreamTypes>(userId: User['id'], type: K, value?: MessagingIndexStreamTypes[K]): void { |  | ||||||
| 		this.publish(`messagingIndexStream:${userId}`, type, typeof value === 'undefined' ? null : value); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public publishNotesStream(note: Packed<'Note'>): void { | 	public publishNotesStream(note: Packed<'Note'>): void { | ||||||
| 		this.publish('notesStream', null, note); | 		this.publish('notesStream', null, note); | ||||||
|   | |||||||
| @@ -99,7 +99,6 @@ export class HttpRequestService { | |||||||
| 		const res = await this.send(url, { | 		const res = await this.send(url, { | ||||||
| 			method: 'GET', | 			method: 'GET', | ||||||
| 			headers: Object.assign({ | 			headers: Object.assign({ | ||||||
| 				'User-Agent': this.config.userAgent, |  | ||||||
| 				Accept: accept, | 				Accept: accept, | ||||||
| 			}, headers ?? {}), | 			}, headers ?? {}), | ||||||
| 			timeout: 5000, | 			timeout: 5000, | ||||||
| @@ -114,7 +113,6 @@ export class HttpRequestService { | |||||||
| 		const res = await this.send(url, { | 		const res = await this.send(url, { | ||||||
| 			method: 'GET', | 			method: 'GET', | ||||||
| 			headers: Object.assign({ | 			headers: Object.assign({ | ||||||
| 				'User-Agent': this.config.userAgent, |  | ||||||
| 				Accept: accept, | 				Accept: accept, | ||||||
| 			}, headers ?? {}), | 			}, headers ?? {}), | ||||||
| 			timeout: 5000, | 			timeout: 5000, | ||||||
| @@ -144,7 +142,10 @@ export class HttpRequestService { | |||||||
|  |  | ||||||
| 		const res = await fetch(url, { | 		const res = await fetch(url, { | ||||||
| 			method: args.method ?? 'GET', | 			method: args.method ?? 'GET', | ||||||
| 			headers: args.headers, | 			headers: { | ||||||
|  | 				'User-Agent': this.config.userAgent, | ||||||
|  | 				...(args.headers ?? {}) | ||||||
|  | 			}, | ||||||
| 			body: args.body, | 			body: args.body, | ||||||
| 			size: args.size ?? 10 * 1024 * 1024, | 			size: args.size ?? 10 * 1024 * 1024, | ||||||
| 			agent: (url) => this.getAgentByUrl(url), | 			agent: (url) => this.getAgentByUrl(url), | ||||||
|   | |||||||
| @@ -107,7 +107,7 @@ export class ImageProcessingService { | |||||||
| 				withoutEnlargement: true, | 				withoutEnlargement: true, | ||||||
| 			}) | 			}) | ||||||
| 			.rotate() | 			.rotate() | ||||||
| 			.webp(options) | 			.webp(options); | ||||||
|  |  | ||||||
| 		return { | 		return { | ||||||
| 			data, | 			data, | ||||||
|   | |||||||
| @@ -1,6 +1,6 @@ | |||||||
| import { Inject, Injectable } from '@nestjs/common'; | import { Inject, Injectable } from '@nestjs/common'; | ||||||
| import { IsNull } from 'typeorm'; | import { IsNull } from 'typeorm'; | ||||||
| import type { ILocalUser } from '@/models/entities/User.js'; | import type { LocalUser } from '@/models/entities/User.js'; | ||||||
| import type { UsersRepository } from '@/models/index.js'; | import type { UsersRepository } from '@/models/index.js'; | ||||||
| import { Cache } from '@/misc/cache.js'; | import { Cache } from '@/misc/cache.js'; | ||||||
| import { DI } from '@/di-symbols.js'; | import { DI } from '@/di-symbols.js'; | ||||||
| @@ -11,7 +11,7 @@ const ACTOR_USERNAME = 'instance.actor' as const; | |||||||
|  |  | ||||||
| @Injectable() | @Injectable() | ||||||
| export class InstanceActorService { | export class InstanceActorService { | ||||||
| 	private cache: Cache<ILocalUser>; | 	private cache: Cache<LocalUser>; | ||||||
|  |  | ||||||
| 	constructor( | 	constructor( | ||||||
| 		@Inject(DI.usersRepository) | 		@Inject(DI.usersRepository) | ||||||
| @@ -19,24 +19,24 @@ export class InstanceActorService { | |||||||
|  |  | ||||||
| 		private createSystemUserService: CreateSystemUserService, | 		private createSystemUserService: CreateSystemUserService, | ||||||
| 	) { | 	) { | ||||||
| 		this.cache = new Cache<ILocalUser>(Infinity); | 		this.cache = new Cache<LocalUser>(Infinity); | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public async getInstanceActor(): Promise<ILocalUser> { | 	public async getInstanceActor(): Promise<LocalUser> { | ||||||
| 		const cached = this.cache.get(null); | 		const cached = this.cache.get(null); | ||||||
| 		if (cached) return cached; | 		if (cached) return cached; | ||||||
| 	 | 	 | ||||||
| 		const user = await this.usersRepository.findOneBy({ | 		const user = await this.usersRepository.findOneBy({ | ||||||
| 			host: IsNull(), | 			host: IsNull(), | ||||||
| 			username: ACTOR_USERNAME, | 			username: ACTOR_USERNAME, | ||||||
| 		}) as ILocalUser | undefined; | 		}) as LocalUser | undefined; | ||||||
| 	 | 	 | ||||||
| 		if (user) { | 		if (user) { | ||||||
| 			this.cache.set(null, user); | 			this.cache.set(null, user); | ||||||
| 			return user; | 			return user; | ||||||
| 		} else { | 		} else { | ||||||
| 			const created = await this.createSystemUserService.createSystemUser(ACTOR_USERNAME) as ILocalUser; | 			const created = await this.createSystemUserService.createSystemUser(ACTOR_USERNAME) as LocalUser; | ||||||
| 			this.cache.set(null, created); | 			this.cache.set(null, created); | ||||||
| 			return created; | 			return created; | ||||||
| 		} | 		} | ||||||
|   | |||||||
| @@ -1,307 +0,0 @@ | |||||||
| import { Inject, Injectable } from '@nestjs/common'; |  | ||||||
| import { In, Not } from 'typeorm'; |  | ||||||
| import { DI } from '@/di-symbols.js'; |  | ||||||
| import type { Config } from '@/config.js'; |  | ||||||
| import type { DriveFile } from '@/models/entities/DriveFile.js'; |  | ||||||
| import type { MessagingMessage } from '@/models/entities/MessagingMessage.js'; |  | ||||||
| import type { Note } from '@/models/entities/Note.js'; |  | ||||||
| import type { User, CacheableUser, IRemoteUser } from '@/models/entities/User.js'; |  | ||||||
| import type { UserGroup } from '@/models/entities/UserGroup.js'; |  | ||||||
| import { QueueService } from '@/core/QueueService.js'; |  | ||||||
| import { toArray } from '@/misc/prelude/array.js'; |  | ||||||
| import { IdentifiableError } from '@/misc/identifiable-error.js'; |  | ||||||
| import type { MessagingMessagesRepository, MutingsRepository, UserGroupJoiningsRepository, UsersRepository } from '@/models/index.js'; |  | ||||||
| import { IdService } from '@/core/IdService.js'; |  | ||||||
| import { GlobalEventService } from '@/core/GlobalEventService.js'; |  | ||||||
| import { UserEntityService } from '@/core/entities/UserEntityService.js'; |  | ||||||
| import { ApRendererService } from '@/core/activitypub/ApRendererService.js'; |  | ||||||
| import { MessagingMessageEntityService } from '@/core/entities/MessagingMessageEntityService.js'; |  | ||||||
| import { PushNotificationService } from '@/core/PushNotificationService.js'; |  | ||||||
| import { bindThis } from '@/decorators.js'; |  | ||||||
|  |  | ||||||
| @Injectable() |  | ||||||
| export class MessagingService { |  | ||||||
| 	constructor( |  | ||||||
| 		@Inject(DI.config) |  | ||||||
| 		private config: Config, |  | ||||||
|  |  | ||||||
| 		@Inject(DI.usersRepository) |  | ||||||
| 		private usersRepository: UsersRepository, |  | ||||||
|  |  | ||||||
| 		@Inject(DI.messagingMessagesRepository) |  | ||||||
| 		private messagingMessagesRepository: MessagingMessagesRepository, |  | ||||||
|  |  | ||||||
| 		@Inject(DI.userGroupJoiningsRepository) |  | ||||||
| 		private userGroupJoiningsRepository: UserGroupJoiningsRepository, |  | ||||||
|  |  | ||||||
| 		@Inject(DI.mutingsRepository) |  | ||||||
| 		private mutingsRepository: MutingsRepository, |  | ||||||
|  |  | ||||||
| 		private userEntityService: UserEntityService, |  | ||||||
| 		private messagingMessageEntityService: MessagingMessageEntityService, |  | ||||||
| 		private idService: IdService, |  | ||||||
| 		private globalEventService: GlobalEventService, |  | ||||||
| 		private apRendererService: ApRendererService, |  | ||||||
| 		private queueService: QueueService, |  | ||||||
| 		private pushNotificationService: PushNotificationService, |  | ||||||
| 	) { |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	@bindThis |  | ||||||
| 	public async createMessage(user: { id: User['id']; host: User['host']; }, recipientUser: CacheableUser | undefined, recipientGroup: UserGroup | undefined, text: string | null | undefined, file: DriveFile | null, uri?: string) { |  | ||||||
| 		const message = { |  | ||||||
| 			id: this.idService.genId(), |  | ||||||
| 			createdAt: new Date(), |  | ||||||
| 			fileId: file ? file.id : null, |  | ||||||
| 			recipientId: recipientUser ? recipientUser.id : null, |  | ||||||
| 			groupId: recipientGroup ? recipientGroup.id : null, |  | ||||||
| 			text: text ? text.trim() : null, |  | ||||||
| 			userId: user.id, |  | ||||||
| 			isRead: false, |  | ||||||
| 			reads: [] as any[], |  | ||||||
| 			uri, |  | ||||||
| 		} as MessagingMessage; |  | ||||||
| 	 |  | ||||||
| 		await this.messagingMessagesRepository.insert(message); |  | ||||||
| 	 |  | ||||||
| 		const messageObj = await this.messagingMessageEntityService.pack(message); |  | ||||||
| 	 |  | ||||||
| 		if (recipientUser) { |  | ||||||
| 			if (this.userEntityService.isLocalUser(user)) { |  | ||||||
| 				// 自分のストリーム |  | ||||||
| 				this.globalEventService.publishMessagingStream(message.userId, recipientUser.id, 'message', messageObj); |  | ||||||
| 				this.globalEventService.publishMessagingIndexStream(message.userId, 'message', messageObj); |  | ||||||
| 				this.globalEventService.publishMainStream(message.userId, 'messagingMessage', messageObj); |  | ||||||
| 			} |  | ||||||
| 	 |  | ||||||
| 			if (this.userEntityService.isLocalUser(recipientUser)) { |  | ||||||
| 				// 相手のストリーム |  | ||||||
| 				this.globalEventService.publishMessagingStream(recipientUser.id, message.userId, 'message', messageObj); |  | ||||||
| 				this.globalEventService.publishMessagingIndexStream(recipientUser.id, 'message', messageObj); |  | ||||||
| 				this.globalEventService.publishMainStream(recipientUser.id, 'messagingMessage', messageObj); |  | ||||||
| 			} |  | ||||||
| 		} else if (recipientGroup) { |  | ||||||
| 			// グループのストリーム |  | ||||||
| 			this.globalEventService.publishGroupMessagingStream(recipientGroup.id, 'message', messageObj); |  | ||||||
| 	 |  | ||||||
| 			// メンバーのストリーム |  | ||||||
| 			const joinings = await this.userGroupJoiningsRepository.findBy({ userGroupId: recipientGroup.id }); |  | ||||||
| 			for (const joining of joinings) { |  | ||||||
| 				this.globalEventService.publishMessagingIndexStream(joining.userId, 'message', messageObj); |  | ||||||
| 				this.globalEventService.publishMainStream(joining.userId, 'messagingMessage', messageObj); |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 	 |  | ||||||
| 		// 2秒経っても(今回作成した)メッセージが既読にならなかったら「未読のメッセージがありますよ」イベントを発行する |  | ||||||
| 		setTimeout(async () => { |  | ||||||
| 			const freshMessage = await this.messagingMessagesRepository.findOneBy({ id: message.id }); |  | ||||||
| 			if (freshMessage == null) return; // メッセージが削除されている場合もある |  | ||||||
| 	 |  | ||||||
| 			if (recipientUser && this.userEntityService.isLocalUser(recipientUser)) { |  | ||||||
| 				if (freshMessage.isRead) return; // 既読 |  | ||||||
| 	 |  | ||||||
| 				//#region ただしミュートされているなら発行しない |  | ||||||
| 				const mute = await this.mutingsRepository.findBy({ |  | ||||||
| 					muterId: recipientUser.id, |  | ||||||
| 				}); |  | ||||||
| 				if (mute.map(m => m.muteeId).includes(user.id)) return; |  | ||||||
| 				//#endregion |  | ||||||
| 	 |  | ||||||
| 				this.globalEventService.publishMainStream(recipientUser.id, 'unreadMessagingMessage', messageObj); |  | ||||||
| 				this.pushNotificationService.pushNotification(recipientUser.id, 'unreadMessagingMessage', messageObj); |  | ||||||
| 			} else if (recipientGroup) { |  | ||||||
| 				const joinings = await this.userGroupJoiningsRepository.findBy({ userGroupId: recipientGroup.id, userId: Not(user.id) }); |  | ||||||
| 				for (const joining of joinings) { |  | ||||||
| 					if (freshMessage.reads.includes(joining.userId)) return; // 既読 |  | ||||||
| 					this.globalEventService.publishMainStream(joining.userId, 'unreadMessagingMessage', messageObj); |  | ||||||
| 					this.pushNotificationService.pushNotification(joining.userId, 'unreadMessagingMessage', messageObj); |  | ||||||
| 				} |  | ||||||
| 			} |  | ||||||
| 		}, 2000); |  | ||||||
| 	 |  | ||||||
| 		if (recipientUser && this.userEntityService.isLocalUser(user) && this.userEntityService.isRemoteUser(recipientUser)) { |  | ||||||
| 			const note = { |  | ||||||
| 				id: message.id, |  | ||||||
| 				createdAt: message.createdAt, |  | ||||||
| 				fileIds: message.fileId ? [message.fileId] : [], |  | ||||||
| 				text: message.text, |  | ||||||
| 				userId: message.userId, |  | ||||||
| 				visibility: 'specified', |  | ||||||
| 				mentions: [recipientUser].map(u => u.id), |  | ||||||
| 				mentionedRemoteUsers: JSON.stringify([recipientUser].map(u => ({ |  | ||||||
| 					uri: u.uri, |  | ||||||
| 					username: u.username, |  | ||||||
| 					host: u.host, |  | ||||||
| 				}))), |  | ||||||
| 			} as Note; |  | ||||||
| 	 |  | ||||||
| 			const activity = this.apRendererService.renderActivity(this.apRendererService.renderCreate(await this.apRendererService.renderNote(note, false, true), note)); |  | ||||||
| 	 |  | ||||||
| 			this.queueService.deliver(user, activity, recipientUser.inbox); |  | ||||||
| 		} |  | ||||||
| 		return messageObj; |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	@bindThis |  | ||||||
| 	public async deleteMessage(message: MessagingMessage) { |  | ||||||
| 		await this.messagingMessagesRepository.delete(message.id); |  | ||||||
| 		this.postDeleteMessage(message); |  | ||||||
| 	} |  | ||||||
| 	 |  | ||||||
| 	@bindThis |  | ||||||
| 	private async postDeleteMessage(message: MessagingMessage) { |  | ||||||
| 		if (message.recipientId) { |  | ||||||
| 			const user = await this.usersRepository.findOneByOrFail({ id: message.userId }); |  | ||||||
| 			const recipient = await this.usersRepository.findOneByOrFail({ id: message.recipientId }); |  | ||||||
| 	 |  | ||||||
| 			if (this.userEntityService.isLocalUser(user)) this.globalEventService.publishMessagingStream(message.userId, message.recipientId, 'deleted', message.id); |  | ||||||
| 			if (this.userEntityService.isLocalUser(recipient)) this.globalEventService.publishMessagingStream(message.recipientId, message.userId, 'deleted', message.id); |  | ||||||
| 	 |  | ||||||
| 			if (this.userEntityService.isLocalUser(user) && this.userEntityService.isRemoteUser(recipient)) { |  | ||||||
| 				const activity = this.apRendererService.renderActivity(this.apRendererService.renderDelete(this.apRendererService.renderTombstone(`${this.config.url}/notes/${message.id}`), user)); |  | ||||||
| 				this.queueService.deliver(user, activity, recipient.inbox); |  | ||||||
| 			} |  | ||||||
| 		} else if (message.groupId) { |  | ||||||
| 			this.globalEventService.publishGroupMessagingStream(message.groupId, 'deleted', message.id); |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	/** |  | ||||||
| 	 * Mark messages as read |  | ||||||
| 	 */ |  | ||||||
| 	@bindThis |  | ||||||
| 	public async readUserMessagingMessage( |  | ||||||
| 		userId: User['id'], |  | ||||||
| 		otherpartyId: User['id'], |  | ||||||
| 		messageIds: MessagingMessage['id'][], |  | ||||||
| 	) { |  | ||||||
| 		if (messageIds.length === 0) return; |  | ||||||
|  |  | ||||||
| 		const messages = await this.messagingMessagesRepository.findBy({ |  | ||||||
| 			id: In(messageIds), |  | ||||||
| 		}); |  | ||||||
|  |  | ||||||
| 		for (const message of messages) { |  | ||||||
| 			if (message.recipientId !== userId) { |  | ||||||
| 				throw new IdentifiableError('e140a4bf-49ce-4fb6-b67c-b78dadf6b52f', 'Access denied (user).'); |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		// Update documents |  | ||||||
| 		await this.messagingMessagesRepository.update({ |  | ||||||
| 			id: In(messageIds), |  | ||||||
| 			userId: otherpartyId, |  | ||||||
| 			recipientId: userId, |  | ||||||
| 			isRead: false, |  | ||||||
| 		}, { |  | ||||||
| 			isRead: true, |  | ||||||
| 		}); |  | ||||||
|  |  | ||||||
| 		// Publish event |  | ||||||
| 		this.globalEventService.publishMessagingStream(otherpartyId, userId, 'read', messageIds); |  | ||||||
| 		this.globalEventService.publishMessagingIndexStream(userId, 'read', messageIds); |  | ||||||
|  |  | ||||||
| 		if (!await this.userEntityService.getHasUnreadMessagingMessage(userId)) { |  | ||||||
| 		// 全ての(いままで未読だった)自分宛てのメッセージを(これで)読みましたよというイベントを発行 |  | ||||||
| 			this.globalEventService.publishMainStream(userId, 'readAllMessagingMessages'); |  | ||||||
| 			this.pushNotificationService.pushNotification(userId, 'readAllMessagingMessages', undefined); |  | ||||||
| 		} else { |  | ||||||
| 		// そのユーザーとのメッセージで未読がなければイベント発行 |  | ||||||
| 			const count = await this.messagingMessagesRepository.count({ |  | ||||||
| 				where: { |  | ||||||
| 					userId: otherpartyId, |  | ||||||
| 					recipientId: userId, |  | ||||||
| 					isRead: false, |  | ||||||
| 				}, |  | ||||||
| 				take: 1, |  | ||||||
| 			}); |  | ||||||
|  |  | ||||||
| 			if (!count) { |  | ||||||
| 				this.pushNotificationService.pushNotification(userId, 'readAllMessagingMessagesOfARoom', { userId: otherpartyId }); |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	/** |  | ||||||
| 	 * Mark messages as read |  | ||||||
| 	 */ |  | ||||||
| 	@bindThis |  | ||||||
| 	public async readGroupMessagingMessage( |  | ||||||
| 		userId: User['id'], |  | ||||||
| 		groupId: UserGroup['id'], |  | ||||||
| 		messageIds: MessagingMessage['id'][], |  | ||||||
| 	) { |  | ||||||
| 		if (messageIds.length === 0) return; |  | ||||||
|  |  | ||||||
| 		// check joined |  | ||||||
| 		const joining = await this.userGroupJoiningsRepository.findOneBy({ |  | ||||||
| 			userId: userId, |  | ||||||
| 			userGroupId: groupId, |  | ||||||
| 		}); |  | ||||||
|  |  | ||||||
| 		if (joining == null) { |  | ||||||
| 			throw new IdentifiableError('930a270c-714a-46b2-b776-ad27276dc569', 'Access denied (group).'); |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		const messages = await this.messagingMessagesRepository.findBy({ |  | ||||||
| 			id: In(messageIds), |  | ||||||
| 		}); |  | ||||||
|  |  | ||||||
| 		const reads: MessagingMessage['id'][] = []; |  | ||||||
|  |  | ||||||
| 		for (const message of messages) { |  | ||||||
| 			if (message.userId === userId) continue; |  | ||||||
| 			if (message.reads.includes(userId)) continue; |  | ||||||
|  |  | ||||||
| 			// Update document |  | ||||||
| 			await this.messagingMessagesRepository.createQueryBuilder().update() |  | ||||||
| 				.set({ |  | ||||||
| 					reads: (() => `array_append("reads", '${joining.userId}')`) as any, |  | ||||||
| 				}) |  | ||||||
| 				.where('id = :id', { id: message.id }) |  | ||||||
| 				.execute(); |  | ||||||
|  |  | ||||||
| 			reads.push(message.id); |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		// Publish event |  | ||||||
| 		this.globalEventService.publishGroupMessagingStream(groupId, 'read', { |  | ||||||
| 			ids: reads, |  | ||||||
| 			userId: userId, |  | ||||||
| 		}); |  | ||||||
| 		this.globalEventService.publishMessagingIndexStream(userId, 'read', reads); |  | ||||||
|  |  | ||||||
| 		if (!await this.userEntityService.getHasUnreadMessagingMessage(userId)) { |  | ||||||
| 		// 全ての(いままで未読だった)自分宛てのメッセージを(これで)読みましたよというイベントを発行 |  | ||||||
| 			this.globalEventService.publishMainStream(userId, 'readAllMessagingMessages'); |  | ||||||
| 			this.pushNotificationService.pushNotification(userId, 'readAllMessagingMessages', undefined); |  | ||||||
| 		} else { |  | ||||||
| 		// そのグループにおいて未読がなければイベント発行 |  | ||||||
| 			const unreadExist = await this.messagingMessagesRepository.createQueryBuilder('message') |  | ||||||
| 				.where('message.groupId = :groupId', { groupId: groupId }) |  | ||||||
| 				.andWhere('message.userId != :userId', { userId: userId }) |  | ||||||
| 				.andWhere('NOT (:userId = ANY(message.reads))', { userId: userId }) |  | ||||||
| 				.andWhere('message.createdAt > :joinedAt', { joinedAt: joining.createdAt }) // 自分が加入する前の会話については、未読扱いしない |  | ||||||
| 				.getOne().then(x => x != null); |  | ||||||
|  |  | ||||||
| 			if (!unreadExist) { |  | ||||||
| 				this.pushNotificationService.pushNotification(userId, 'readAllMessagingMessagesOfARoom', { groupId }); |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	@bindThis |  | ||||||
| 	public async deliverReadActivity(user: { id: User['id']; host: null; }, recipient: IRemoteUser, messages: MessagingMessage | MessagingMessage[]) { |  | ||||||
| 		messages = toArray(messages).filter(x => x.uri); |  | ||||||
| 		const contents = messages.map(x => this.apRendererService.renderRead(user, x)); |  | ||||||
|  |  | ||||||
| 		if (contents.length > 1) { |  | ||||||
| 			const collection = this.apRendererService.renderOrderedCollection(null, contents.length, undefined, undefined, contents); |  | ||||||
| 			this.queueService.deliver(user, this.apRendererService.renderActivity(collection), recipient.inbox); |  | ||||||
| 		} else { |  | ||||||
| 			for (const content of contents) { |  | ||||||
| 				this.queueService.deliver(user, this.apRendererService.renderActivity(content), recipient.inbox); |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
| @@ -1,9 +1,8 @@ | |||||||
| import { URL } from 'node:url'; | import { URL } from 'node:url'; | ||||||
| import { Inject, Injectable } from '@nestjs/common'; | import { Inject, Injectable } from '@nestjs/common'; | ||||||
| import * as parse5 from 'parse5'; | import * as parse5 from 'parse5'; | ||||||
| import { JSDOM } from 'jsdom'; | import { Window } from 'happy-dom'; | ||||||
| import { DI } from '@/di-symbols.js'; | import { DI } from '@/di-symbols.js'; | ||||||
| import type { UsersRepository } from '@/models/index.js'; |  | ||||||
| import type { Config } from '@/config.js'; | import type { Config } from '@/config.js'; | ||||||
| import { intersperse } from '@/misc/prelude/array.js'; | import { intersperse } from '@/misc/prelude/array.js'; | ||||||
| import type { IMentionedRemoteUsers } from '@/models/entities/Note.js'; | import type { IMentionedRemoteUsers } from '@/models/entities/Note.js'; | ||||||
| @@ -236,7 +235,7 @@ export class MfmService { | |||||||
| 			return null; | 			return null; | ||||||
| 		} | 		} | ||||||
| 	 | 	 | ||||||
| 		const { window } = new JSDOM(''); | 		const { window } = new Window(); | ||||||
| 	 | 	 | ||||||
| 		const doc = window.document; | 		const doc = window.document; | ||||||
| 	 | 	 | ||||||
| @@ -301,7 +300,7 @@ export class MfmService { | |||||||
| 	 | 	 | ||||||
| 			hashtag: (node) => { | 			hashtag: (node) => { | ||||||
| 				const a = doc.createElement('a'); | 				const a = doc.createElement('a'); | ||||||
| 				a.href = `${this.config.url}/tags/${node.props.hashtag}`; | 				a.setAttribute('href', `${this.config.url}/tags/${node.props.hashtag}`); | ||||||
| 				a.textContent = `#${node.props.hashtag}`; | 				a.textContent = `#${node.props.hashtag}`; | ||||||
| 				a.setAttribute('rel', 'tag'); | 				a.setAttribute('rel', 'tag'); | ||||||
| 				return a; | 				return a; | ||||||
| @@ -327,7 +326,7 @@ export class MfmService { | |||||||
| 	 | 	 | ||||||
| 			link: (node) => { | 			link: (node) => { | ||||||
| 				const a = doc.createElement('a'); | 				const a = doc.createElement('a'); | ||||||
| 				a.href = node.props.url; | 				a.setAttribute('href', node.props.url); | ||||||
| 				appendChildren(node.children, a); | 				appendChildren(node.children, a); | ||||||
| 				return a; | 				return a; | ||||||
| 			}, | 			}, | ||||||
| @@ -336,7 +335,7 @@ export class MfmService { | |||||||
| 				const a = doc.createElement('a'); | 				const a = doc.createElement('a'); | ||||||
| 				const { username, host, acct } = node.props; | 				const { username, host, acct } = node.props; | ||||||
| 				const remoteUserInfo = mentionedRemoteUsers.find(remoteUser => remoteUser.username === username && remoteUser.host === host); | 				const remoteUserInfo = mentionedRemoteUsers.find(remoteUser => remoteUser.username === username && remoteUser.host === host); | ||||||
| 				a.href = remoteUserInfo ? (remoteUserInfo.url ? remoteUserInfo.url : remoteUserInfo.uri) : `${this.config.url}/${acct}`; | 				a.setAttribute('href', remoteUserInfo ? (remoteUserInfo.url ? remoteUserInfo.url : remoteUserInfo.uri) : `${this.config.url}/${acct}`); | ||||||
| 				a.className = 'u-url mention'; | 				a.className = 'u-url mention'; | ||||||
| 				a.textContent = acct; | 				a.textContent = acct; | ||||||
| 				return a; | 				return a; | ||||||
| @@ -361,14 +360,14 @@ export class MfmService { | |||||||
| 	 | 	 | ||||||
| 			url: (node) => { | 			url: (node) => { | ||||||
| 				const a = doc.createElement('a'); | 				const a = doc.createElement('a'); | ||||||
| 				a.href = node.props.url; | 				a.setAttribute('href', node.props.url); | ||||||
| 				a.textContent = node.props.url; | 				a.textContent = node.props.url; | ||||||
| 				return a; | 				return a; | ||||||
| 			}, | 			}, | ||||||
| 	 | 	 | ||||||
| 			search: (node) => { | 			search: (node) => { | ||||||
| 				const a = doc.createElement('a'); | 				const a = doc.createElement('a'); | ||||||
| 				a.href = `https://www.google.com/search?q=${node.props.query}`; | 				a.setAttribute('href', `https://www.google.com/search?q=${node.props.query}`); | ||||||
| 				a.textContent = node.props.content; | 				a.textContent = node.props.content; | ||||||
| 				return a; | 				return a; | ||||||
| 			}, | 			}, | ||||||
|   | |||||||
| @@ -1,5 +1,5 @@ | |||||||
| import * as mfm from 'mfm-js'; | import * as mfm from 'mfm-js'; | ||||||
| import { Not, In, DataSource } from 'typeorm'; | import { In, DataSource } from 'typeorm'; | ||||||
| import { Inject, Injectable } from '@nestjs/common'; | import { Inject, Injectable } from '@nestjs/common'; | ||||||
| import { extractMentions } from '@/misc/extract-mentions.js'; | import { extractMentions } from '@/misc/extract-mentions.js'; | ||||||
| import { extractCustomEmojisFromMfm } from '@/misc/extract-custom-emojis-from-mfm.js'; | import { extractCustomEmojisFromMfm } from '@/misc/extract-custom-emojis-from-mfm.js'; | ||||||
| @@ -11,7 +11,7 @@ import type { DriveFile } from '@/models/entities/DriveFile.js'; | |||||||
| import type { App } from '@/models/entities/App.js'; | import type { App } from '@/models/entities/App.js'; | ||||||
| import { concat } from '@/misc/prelude/array.js'; | import { concat } from '@/misc/prelude/array.js'; | ||||||
| import { IdService } from '@/core/IdService.js'; | import { IdService } from '@/core/IdService.js'; | ||||||
| import type { User, ILocalUser, IRemoteUser } from '@/models/entities/User.js'; | import type { User, LocalUser, RemoteUser } from '@/models/entities/User.js'; | ||||||
| import type { IPoll } from '@/models/entities/Poll.js'; | import type { IPoll } from '@/models/entities/Poll.js'; | ||||||
| import { Poll } from '@/models/entities/Poll.js'; | import { Poll } from '@/models/entities/Poll.js'; | ||||||
| import { isDuplicateKeyValueError } from '@/misc/is-duplicate-key-value-error.js'; | import { isDuplicateKeyValueError } from '@/misc/is-duplicate-key-value-error.js'; | ||||||
| @@ -52,7 +52,7 @@ class NotificationManager { | |||||||
| 	private notifier: { id: User['id']; }; | 	private notifier: { id: User['id']; }; | ||||||
| 	private note: Note; | 	private note: Note; | ||||||
| 	private queue: { | 	private queue: { | ||||||
| 		target: ILocalUser['id']; | 		target: LocalUser['id']; | ||||||
| 		reason: NotificationType; | 		reason: NotificationType; | ||||||
| 	}[]; | 	}[]; | ||||||
|  |  | ||||||
| @@ -68,7 +68,7 @@ class NotificationManager { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public push(notifiee: ILocalUser['id'], reason: NotificationType) { | 	public push(notifiee: LocalUser['id'], reason: NotificationType) { | ||||||
| 		// 自分自身へは通知しない | 		// 自分自身へは通知しない | ||||||
| 		if (this.notifier.id === notifiee) return; | 		if (this.notifier.id === notifiee) return; | ||||||
|  |  | ||||||
| @@ -605,7 +605,7 @@ export class NoteCreateService { | |||||||
|  |  | ||||||
| 					// メンションされたリモートユーザーに配送 | 					// メンションされたリモートユーザーに配送 | ||||||
| 					for (const u of mentionedUsers.filter(u => this.userEntityService.isRemoteUser(u))) { | 					for (const u of mentionedUsers.filter(u => this.userEntityService.isRemoteUser(u))) { | ||||||
| 						dm.addDirectRecipe(u as IRemoteUser); | 						dm.addDirectRecipe(u as RemoteUser); | ||||||
| 					} | 					} | ||||||
|  |  | ||||||
| 					// 投稿がリプライかつ投稿者がローカルユーザーかつリプライ先の投稿の投稿者がリモートユーザーなら配送 | 					// 投稿がリプライかつ投稿者がローカルユーザーかつリプライ先の投稿の投稿者がリモートユーザーなら配送 | ||||||
| @@ -711,7 +711,7 @@ export class NoteCreateService { | |||||||
| 			? this.apRendererService.renderAnnounce(data.renote.uri ? data.renote.uri : `${this.config.url}/notes/${data.renote.id}`, note) | 			? this.apRendererService.renderAnnounce(data.renote.uri ? data.renote.uri : `${this.config.url}/notes/${data.renote.id}`, note) | ||||||
| 			: this.apRendererService.renderCreate(await this.apRendererService.renderNote(note, false), note); | 			: this.apRendererService.renderCreate(await this.apRendererService.renderNote(note, false), note); | ||||||
|  |  | ||||||
| 		return this.apRendererService.renderActivity(content); | 		return this.apRendererService.addContext(content); | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
|   | |||||||
| @@ -1,6 +1,6 @@ | |||||||
| import { Brackets, In } from 'typeorm'; | import { Brackets, In } from 'typeorm'; | ||||||
| import { Injectable, Inject } from '@nestjs/common'; | import { Injectable, Inject } from '@nestjs/common'; | ||||||
| import type { User, ILocalUser, IRemoteUser } from '@/models/entities/User.js'; | import type { User, LocalUser, RemoteUser } from '@/models/entities/User.js'; | ||||||
| import type { Note, IMentionedRemoteUsers } from '@/models/entities/Note.js'; | import type { Note, IMentionedRemoteUsers } from '@/models/entities/Note.js'; | ||||||
| import type { InstancesRepository, NotesRepository, UsersRepository } from '@/models/index.js'; | import type { InstancesRepository, NotesRepository, UsersRepository } from '@/models/index.js'; | ||||||
| import { RelayService } from '@/core/RelayService.js'; | import { RelayService } from '@/core/RelayService.js'; | ||||||
| @@ -78,7 +78,7 @@ export class NoteDeleteService { | |||||||
| 					}); | 					}); | ||||||
| 				} | 				} | ||||||
|  |  | ||||||
| 				const content = this.apRendererService.renderActivity(renote | 				const content = this.apRendererService.addContext(renote | ||||||
| 					? this.apRendererService.renderUndo(this.apRendererService.renderAnnounce(renote.uri ?? `${this.config.url}/notes/${renote.id}`, note), user) | 					? this.apRendererService.renderUndo(this.apRendererService.renderAnnounce(renote.uri ?? `${this.config.url}/notes/${renote.id}`, note), user) | ||||||
| 					: this.apRendererService.renderDelete(this.apRendererService.renderTombstone(`${this.config.url}/notes/${note.id}`), user)); | 					: this.apRendererService.renderDelete(this.apRendererService.renderTombstone(`${this.config.url}/notes/${note.id}`), user)); | ||||||
|  |  | ||||||
| @@ -90,7 +90,7 @@ export class NoteDeleteService { | |||||||
| 			for (const cascadingNote of cascadingNotes) { | 			for (const cascadingNote of cascadingNotes) { | ||||||
| 				if (!cascadingNote.user) continue; | 				if (!cascadingNote.user) continue; | ||||||
| 				if (!this.userEntityService.isLocalUser(cascadingNote.user)) continue; | 				if (!this.userEntityService.isLocalUser(cascadingNote.user)) continue; | ||||||
| 				const content = this.apRendererService.renderActivity(this.apRendererService.renderDelete(this.apRendererService.renderTombstone(`${this.config.url}/notes/${cascadingNote.id}`), cascadingNote.user)); | 				const content = this.apRendererService.addContext(this.apRendererService.renderDelete(this.apRendererService.renderTombstone(`${this.config.url}/notes/${cascadingNote.id}`), cascadingNote.user)); | ||||||
| 				this.deliverToConcerned(cascadingNote.user, cascadingNote, content); | 				this.deliverToConcerned(cascadingNote.user, cascadingNote, content); | ||||||
| 			} | 			} | ||||||
| 			//#endregion | 			//#endregion | ||||||
| @@ -159,11 +159,11 @@ export class NoteDeleteService { | |||||||
|  |  | ||||||
| 		return await this.usersRepository.find({ | 		return await this.usersRepository.find({ | ||||||
| 			where, | 			where, | ||||||
| 		}) as IRemoteUser[]; | 		}) as RemoteUser[]; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	private async deliverToConcerned(user: { id: ILocalUser['id']; host: null; }, note: Note, content: any) { | 	private async deliverToConcerned(user: { id: LocalUser['id']; host: null; }, note: Note, content: any) { | ||||||
| 		this.apDeliverManagerService.deliverToFollowers(user, content); | 		this.apDeliverManagerService.deliverToFollowers(user, content); | ||||||
| 		this.relayService.deliverToRelays(user, content); | 		this.relayService.deliverToRelays(user, content); | ||||||
| 		const remoteUsers = await this.getMentionedRemoteUsers(note); | 		const remoteUsers = await this.getMentionedRemoteUsers(note); | ||||||
|   | |||||||
| @@ -115,7 +115,7 @@ export class NotePiningService { | |||||||
|  |  | ||||||
| 		const target = `${this.config.url}/users/${user.id}/collections/featured`; | 		const target = `${this.config.url}/users/${user.id}/collections/featured`; | ||||||
| 		const item = `${this.config.url}/notes/${noteId}`; | 		const item = `${this.config.url}/notes/${noteId}`; | ||||||
| 		const content = this.apRendererService.renderActivity(isAddition ? this.apRendererService.renderAdd(user, target, item) : this.apRendererService.renderRemove(user, target, item)); | 		const content = this.apRendererService.addContext(isAddition ? this.apRendererService.renderAdd(user, target, item) : this.apRendererService.renderRemove(user, target, item)); | ||||||
|  |  | ||||||
| 		this.apDeliverManagerService.deliverToFollowers(user, content); | 		this.apDeliverManagerService.deliverToFollowers(user, content); | ||||||
| 		this.relayService.deliverToRelays(user, content); | 		this.relayService.deliverToRelays(user, content); | ||||||
|   | |||||||
| @@ -2,13 +2,12 @@ import { Inject, Injectable } from '@nestjs/common'; | |||||||
| import { In } from 'typeorm'; | import { In } from 'typeorm'; | ||||||
| import { DI } from '@/di-symbols.js'; | import { DI } from '@/di-symbols.js'; | ||||||
| import type { NotificationsRepository } from '@/models/index.js'; | import type { NotificationsRepository } from '@/models/index.js'; | ||||||
| import type { UsersRepository } from '@/models/index.js'; |  | ||||||
| import type { User } from '@/models/entities/User.js'; | import type { User } from '@/models/entities/User.js'; | ||||||
| import type { Notification } from '@/models/entities/Notification.js'; | import type { Notification } from '@/models/entities/Notification.js'; | ||||||
| import { UserEntityService } from '@/core/entities/UserEntityService.js'; | import { UserEntityService } from '@/core/entities/UserEntityService.js'; | ||||||
|  | import { bindThis } from '@/decorators.js'; | ||||||
| import { GlobalEventService } from './GlobalEventService.js'; | import { GlobalEventService } from './GlobalEventService.js'; | ||||||
| import { PushNotificationService } from './PushNotificationService.js'; | import { PushNotificationService } from './PushNotificationService.js'; | ||||||
| import { bindThis } from '@/decorators.js'; |  | ||||||
|  |  | ||||||
| @Injectable() | @Injectable() | ||||||
| export class NotificationService { | export class NotificationService { | ||||||
| @@ -66,7 +65,6 @@ export class NotificationService { | |||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	private postReadNotifications(userId: User['id'], notificationIds: Notification['id'][]) { | 	private postReadNotifications(userId: User['id'], notificationIds: Notification['id'][]) { | ||||||
| 		this.globalEventService.publishMainStream(userId, 'readNotifications', notificationIds); |  | ||||||
| 		return this.pushNotificationService.pushNotification(userId, 'readNotifications', { notificationIds }); | 		return this.pushNotificationService.pushNotification(userId, 'readNotifications', { notificationIds }); | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,10 +1,8 @@ | |||||||
| import { Inject, Injectable } from '@nestjs/common'; | import { Inject, Injectable } from '@nestjs/common'; | ||||||
| import { Not } from 'typeorm'; |  | ||||||
| import { DI } from '@/di-symbols.js'; | import { DI } from '@/di-symbols.js'; | ||||||
| import type { NotesRepository, UsersRepository, PollsRepository, PollVotesRepository } from '@/models/index.js'; | import type { NotesRepository, UsersRepository, PollsRepository, PollVotesRepository, User } 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 { IdService } from '@/core/IdService.js'; | import { IdService } from '@/core/IdService.js'; | ||||||
| import { GlobalEventService } from '@/core/GlobalEventService.js'; | import { GlobalEventService } from '@/core/GlobalEventService.js'; | ||||||
| import { ApRendererService } from '@/core/activitypub/ApRendererService.js'; | import { ApRendererService } from '@/core/activitypub/ApRendererService.js'; | ||||||
| @@ -39,7 +37,7 @@ export class PollService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public async vote(user: CacheableUser, note: Note, choice: number) { | 	public async vote(user: User, note: Note, choice: number) { | ||||||
| 		const poll = await this.pollsRepository.findOneBy({ noteId: note.id }); | 		const poll = await this.pollsRepository.findOneBy({ noteId: note.id }); | ||||||
| 	 | 	 | ||||||
| 		if (poll == null) throw new Error('poll not found'); | 		if (poll == null) throw new Error('poll not found'); | ||||||
| @@ -97,7 +95,7 @@ export class PollService { | |||||||
| 		if (user == null) throw new Error('note not found'); | 		if (user == null) throw new Error('note not found'); | ||||||
| 	 | 	 | ||||||
| 		if (this.userEntityService.isLocalUser(user)) { | 		if (this.userEntityService.isLocalUser(user)) { | ||||||
| 			const content = this.apRendererService.renderActivity(this.apRendererService.renderUpdate(await this.apRendererService.renderNote(note, false), user)); | 			const content = this.apRendererService.addContext(this.apRendererService.renderUpdate(await this.apRendererService.renderNote(note, false), user)); | ||||||
| 			this.apDeliverManagerService.deliverToFollowers(user, content); | 			this.apDeliverManagerService.deliverToFollowers(user, content); | ||||||
| 			this.relayService.deliverToRelays(user, content); | 			this.relayService.deliverToRelays(user, content); | ||||||
| 		} | 		} | ||||||
|   | |||||||
| @@ -1,6 +1,6 @@ | |||||||
| import { Inject, Injectable } from '@nestjs/common'; | import { Inject, Injectable } from '@nestjs/common'; | ||||||
| import type { UsersRepository } from '@/models/index.js'; | import type { UsersRepository } from '@/models/index.js'; | ||||||
| import type { ILocalUser, User } from '@/models/entities/User.js'; | import type { LocalUser } from '@/models/entities/User.js'; | ||||||
| import { DI } from '@/di-symbols.js'; | import { DI } from '@/di-symbols.js'; | ||||||
| import { MetaService } from '@/core/MetaService.js'; | import { MetaService } from '@/core/MetaService.js'; | ||||||
| import { bindThis } from '@/decorators.js'; | import { bindThis } from '@/decorators.js'; | ||||||
| @@ -16,9 +16,9 @@ export class ProxyAccountService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public async fetch(): Promise<ILocalUser | null> { | 	public async fetch(): Promise<LocalUser | null> { | ||||||
| 		const meta = await this.metaService.fetch(); | 		const meta = await this.metaService.fetch(); | ||||||
| 		if (meta.proxyAccountId == null) return null; | 		if (meta.proxyAccountId == null) return null; | ||||||
| 		return await this.usersRepository.findOneByOrFail({ id: meta.proxyAccountId }) as ILocalUser; | 		return await this.usersRepository.findOneByOrFail({ id: meta.proxyAccountId }) as LocalUser; | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|   | |||||||
| @@ -9,24 +9,21 @@ import { MetaService } from '@/core/MetaService.js'; | |||||||
| import { bindThis } from '@/decorators.js'; | import { bindThis } from '@/decorators.js'; | ||||||
|  |  | ||||||
| // Defined also packages/sw/types.ts#L13 | // Defined also packages/sw/types.ts#L13 | ||||||
| type pushNotificationsTypes = { | type PushNotificationsTypes = { | ||||||
| 	'notification': Packed<'Notification'>; | 	'notification': Packed<'Notification'>; | ||||||
| 	'unreadMessagingMessage': Packed<'MessagingMessage'>; |  | ||||||
| 	'unreadAntennaNote': { | 	'unreadAntennaNote': { | ||||||
| 		antenna: { id: string, name: string }; | 		antenna: { id: string, name: string }; | ||||||
| 		note: Packed<'Note'>; | 		note: Packed<'Note'>; | ||||||
| 	}; | 	}; | ||||||
| 	'readNotifications': { notificationIds: string[] }; | 	'readNotifications': { notificationIds: string[] }; | ||||||
| 	'readAllNotifications': undefined; | 	'readAllNotifications': undefined; | ||||||
| 	'readAllMessagingMessages': undefined; |  | ||||||
| 	'readAllMessagingMessagesOfARoom': { userId: string } | { groupId: string }; |  | ||||||
| 	'readAntenna': { antennaId: string }; | 	'readAntenna': { antennaId: string }; | ||||||
| 	'readAllAntennas': undefined; | 	'readAllAntennas': undefined; | ||||||
| }; | }; | ||||||
|  |  | ||||||
| // Reduce length because push message servers have character limits | // Reduce length because push message servers have character limits | ||||||
| function truncateBody<T extends keyof pushNotificationsTypes>(type: T, body: pushNotificationsTypes[T]): pushNotificationsTypes[T] { | function truncateBody<T extends keyof PushNotificationsTypes>(type: T, body: PushNotificationsTypes[T]): PushNotificationsTypes[T] { | ||||||
| 	if (body === undefined) return body; | 	if (typeof body !== 'object') return body; | ||||||
|  |  | ||||||
| 	return { | 	return { | ||||||
| 		...body, | 		...body, | ||||||
| @@ -40,11 +37,9 @@ function truncateBody<T extends keyof pushNotificationsTypes>(type: T, body: pus | |||||||
| 				reply: undefined, | 				reply: undefined, | ||||||
| 				renote: undefined, | 				renote: undefined, | ||||||
| 				user: type === 'notification' ? undefined as any : body.note.user, | 				user: type === 'notification' ? undefined as any : body.note.user, | ||||||
| 			} | 			}, | ||||||
| 		} : {}), | 		} : {}), | ||||||
| 	}; | 	}; | ||||||
|  |  | ||||||
| 	return body; |  | ||||||
| } | } | ||||||
|  |  | ||||||
| @Injectable() | @Injectable() | ||||||
| @@ -61,7 +56,7 @@ export class PushNotificationService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public async pushNotification<T extends keyof pushNotificationsTypes>(userId: string, type: T, body: pushNotificationsTypes[T]) { | 	public async pushNotification<T extends keyof PushNotificationsTypes>(userId: string, type: T, body: PushNotificationsTypes[T]) { | ||||||
| 		const meta = await this.metaService.fetch(); | 		const meta = await this.metaService.fetch(); | ||||||
| 	 | 	 | ||||||
| 		if (!meta.enableServiceWorker || meta.swPublicKey == null || meta.swPrivateKey == null) return; | 		if (!meta.enableServiceWorker || meta.swPublicKey == null || meta.swPrivateKey == null) return; | ||||||
| @@ -81,8 +76,6 @@ export class PushNotificationService { | |||||||
| 			if ([ | 			if ([ | ||||||
| 				'readNotifications', | 				'readNotifications', | ||||||
| 				'readAllNotifications', | 				'readAllNotifications', | ||||||
| 				'readAllMessagingMessages', |  | ||||||
| 				'readAllMessagingMessagesOfARoom', |  | ||||||
| 				'readAntenna', | 				'readAntenna', | ||||||
| 				'readAllAntennas', | 				'readAllAntennas', | ||||||
| 			].includes(type) && !subscription.sendReadMessage) continue; | 			].includes(type) && !subscription.sendReadMessage) continue; | ||||||
|   | |||||||
| @@ -3,7 +3,7 @@ import { IsNull } from 'typeorm'; | |||||||
| import { DI } from '@/di-symbols.js'; | import { DI } from '@/di-symbols.js'; | ||||||
| import type { EmojisRepository, BlockingsRepository, NoteReactionsRepository, UsersRepository, NotesRepository } from '@/models/index.js'; | import type { EmojisRepository, BlockingsRepository, NoteReactionsRepository, UsersRepository, NotesRepository } from '@/models/index.js'; | ||||||
| import { IdentifiableError } from '@/misc/identifiable-error.js'; | import { IdentifiableError } from '@/misc/identifiable-error.js'; | ||||||
| import type { IRemoteUser, User } from '@/models/entities/User.js'; | import type { RemoteUser, User } from '@/models/entities/User.js'; | ||||||
| import type { Note } from '@/models/entities/Note.js'; | import type { Note } from '@/models/entities/Note.js'; | ||||||
| import { IdService } from '@/core/IdService.js'; | import { IdService } from '@/core/IdService.js'; | ||||||
| import type { NoteReaction } from '@/models/entities/NoteReaction.js'; | import type { NoteReaction } from '@/models/entities/NoteReaction.js'; | ||||||
| @@ -85,7 +85,7 @@ export class ReactionService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	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 | null) { | ||||||
| 		// Check blocking | 		// Check blocking | ||||||
| 		if (note.userId !== user.id) { | 		if (note.userId !== user.id) { | ||||||
| 			const blocked = await this.userBlockingService.checkBlocked(note.userId, user.id); | 			const blocked = await this.userBlockingService.checkBlocked(note.userId, user.id); | ||||||
| @@ -177,11 +177,11 @@ export class ReactionService { | |||||||
| 	 | 	 | ||||||
| 		//#region 配信 | 		//#region 配信 | ||||||
| 		if (this.userEntityService.isLocalUser(user) && !note.localOnly) { | 		if (this.userEntityService.isLocalUser(user) && !note.localOnly) { | ||||||
| 			const content = this.apRendererService.renderActivity(await this.apRendererService.renderLike(record, note)); | 			const content = this.apRendererService.addContext(await this.apRendererService.renderLike(record, note)); | ||||||
| 			const dm = this.apDeliverManagerService.createDeliverManager(user, content); | 			const dm = this.apDeliverManagerService.createDeliverManager(user, content); | ||||||
| 			if (note.userHost !== null) { | 			if (note.userHost !== null) { | ||||||
| 				const reactee = await this.usersRepository.findOneBy({ id: note.userId }); | 				const reactee = await this.usersRepository.findOneBy({ id: note.userId }); | ||||||
| 				dm.addDirectRecipe(reactee as IRemoteUser); | 				dm.addDirectRecipe(reactee as RemoteUser); | ||||||
| 			} | 			} | ||||||
| 	 | 	 | ||||||
| 			if (['public', 'home', 'followers'].includes(note.visibility)) { | 			if (['public', 'home', 'followers'].includes(note.visibility)) { | ||||||
| @@ -189,7 +189,7 @@ export class ReactionService { | |||||||
| 			} else if (note.visibility === 'specified') { | 			} else if (note.visibility === 'specified') { | ||||||
| 				const visibleUsers = await Promise.all(note.visibleUserIds.map(id => this.usersRepository.findOneBy({ id }))); | 				const visibleUsers = await Promise.all(note.visibleUserIds.map(id => this.usersRepository.findOneBy({ id }))); | ||||||
| 				for (const u of visibleUsers.filter(u => u && this.userEntityService.isRemoteUser(u))) { | 				for (const u of visibleUsers.filter(u => u && this.userEntityService.isRemoteUser(u))) { | ||||||
| 					dm.addDirectRecipe(u as IRemoteUser); | 					dm.addDirectRecipe(u as RemoteUser); | ||||||
| 				} | 				} | ||||||
| 			} | 			} | ||||||
| 	 | 	 | ||||||
| @@ -235,11 +235,11 @@ export class ReactionService { | |||||||
| 	 | 	 | ||||||
| 		//#region 配信 | 		//#region 配信 | ||||||
| 		if (this.userEntityService.isLocalUser(user) && !note.localOnly) { | 		if (this.userEntityService.isLocalUser(user) && !note.localOnly) { | ||||||
| 			const content = this.apRendererService.renderActivity(this.apRendererService.renderUndo(await this.apRendererService.renderLike(exist, note), user)); | 			const content = this.apRendererService.addContext(this.apRendererService.renderUndo(await this.apRendererService.renderLike(exist, note), user)); | ||||||
| 			const dm = this.apDeliverManagerService.createDeliverManager(user, content); | 			const dm = this.apDeliverManagerService.createDeliverManager(user, content); | ||||||
| 			if (note.userHost !== null) { | 			if (note.userHost !== null) { | ||||||
| 				const reactee = await this.usersRepository.findOneBy({ id: note.userId }); | 				const reactee = await this.usersRepository.findOneBy({ id: note.userId }); | ||||||
| 				dm.addDirectRecipe(reactee as IRemoteUser); | 				dm.addDirectRecipe(reactee as RemoteUser); | ||||||
| 			} | 			} | ||||||
| 			dm.addFollowersRecipe(); | 			dm.addFollowersRecipe(); | ||||||
| 			dm.execute(); | 			dm.execute(); | ||||||
|   | |||||||
| @@ -1,6 +1,6 @@ | |||||||
| import { Inject, Injectable } from '@nestjs/common'; | import { Inject, Injectable } from '@nestjs/common'; | ||||||
| import { IsNull } from 'typeorm'; | import { IsNull } from 'typeorm'; | ||||||
| import type { ILocalUser, User } from '@/models/entities/User.js'; | import type { LocalUser, User } from '@/models/entities/User.js'; | ||||||
| import type { RelaysRepository, UsersRepository } from '@/models/index.js'; | import type { RelaysRepository, UsersRepository } from '@/models/index.js'; | ||||||
| import { IdService } from '@/core/IdService.js'; | import { IdService } from '@/core/IdService.js'; | ||||||
| import { Cache } from '@/misc/cache.js'; | import { Cache } from '@/misc/cache.js'; | ||||||
| @@ -34,16 +34,16 @@ export class RelayService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	private async getRelayActor(): Promise<ILocalUser> { | 	private async getRelayActor(): Promise<LocalUser> { | ||||||
| 		const user = await this.usersRepository.findOneBy({ | 		const user = await this.usersRepository.findOneBy({ | ||||||
| 			host: IsNull(), | 			host: IsNull(), | ||||||
| 			username: ACTOR_USERNAME, | 			username: ACTOR_USERNAME, | ||||||
| 		}); | 		}); | ||||||
| 	 | 	 | ||||||
| 		if (user) return user as ILocalUser; | 		if (user) return user as LocalUser; | ||||||
| 	 | 	 | ||||||
| 		const created = await this.createSystemUserService.createSystemUser(ACTOR_USERNAME); | 		const created = await this.createSystemUserService.createSystemUser(ACTOR_USERNAME); | ||||||
| 		return created as ILocalUser; | 		return created as LocalUser; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| @@ -56,7 +56,7 @@ export class RelayService { | |||||||
| 	 | 	 | ||||||
| 		const relayActor = await this.getRelayActor(); | 		const relayActor = await this.getRelayActor(); | ||||||
| 		const follow = await this.apRendererService.renderFollowRelay(relay, relayActor); | 		const follow = await this.apRendererService.renderFollowRelay(relay, relayActor); | ||||||
| 		const activity = this.apRendererService.renderActivity(follow); | 		const activity = this.apRendererService.addContext(follow); | ||||||
| 		this.queueService.deliver(relayActor, activity, relay.inbox); | 		this.queueService.deliver(relayActor, activity, relay.inbox); | ||||||
| 	 | 	 | ||||||
| 		return relay; | 		return relay; | ||||||
| @@ -75,7 +75,7 @@ export class RelayService { | |||||||
| 		const relayActor = await this.getRelayActor(); | 		const relayActor = await this.getRelayActor(); | ||||||
| 		const follow = this.apRendererService.renderFollowRelay(relay, relayActor); | 		const follow = this.apRendererService.renderFollowRelay(relay, relayActor); | ||||||
| 		const undo = this.apRendererService.renderUndo(follow, relayActor); | 		const undo = this.apRendererService.renderUndo(follow, relayActor); | ||||||
| 		const activity = this.apRendererService.renderActivity(undo); | 		const activity = this.apRendererService.addContext(undo); | ||||||
| 		this.queueService.deliver(relayActor, activity, relay.inbox); | 		this.queueService.deliver(relayActor, activity, relay.inbox); | ||||||
| 	 | 	 | ||||||
| 		await this.relaysRepository.delete(relay.id); | 		await this.relaysRepository.delete(relay.id); | ||||||
|   | |||||||
| @@ -1,7 +1,6 @@ | |||||||
| import { Inject, Injectable } from '@nestjs/common'; | import { Injectable } from '@nestjs/common'; | ||||||
| import type Logger from '@/logger.js'; | import type Logger from '@/logger.js'; | ||||||
| import { LoggerService } from '@/core/LoggerService.js'; | import { LoggerService } from '@/core/LoggerService.js'; | ||||||
| import { bindThis } from '@/decorators.js'; |  | ||||||
|  |  | ||||||
| @Injectable() | @Injectable() | ||||||
| export class RemoteLoggerService { | export class RemoteLoggerService { | ||||||
|   | |||||||
| @@ -4,7 +4,7 @@ import chalk from 'chalk'; | |||||||
| import { IsNull } from 'typeorm'; | import { IsNull } from 'typeorm'; | ||||||
| import { DI } from '@/di-symbols.js'; | import { DI } from '@/di-symbols.js'; | ||||||
| import type { UsersRepository } from '@/models/index.js'; | import type { UsersRepository } from '@/models/index.js'; | ||||||
| import type { IRemoteUser, User } from '@/models/entities/User.js'; | import type { RemoteUser, User } from '@/models/entities/User.js'; | ||||||
| import type { Config } from '@/config.js'; | import type { Config } from '@/config.js'; | ||||||
| import type Logger from '@/logger.js'; | import type Logger from '@/logger.js'; | ||||||
| import { UtilityService } from '@/core/UtilityService.js'; | import { UtilityService } from '@/core/UtilityService.js'; | ||||||
| @@ -60,7 +60,7 @@ export class RemoteUserResolveService { | |||||||
| 			}); | 			}); | ||||||
| 		} | 		} | ||||||
| 	 | 	 | ||||||
| 		const user = await this.usersRepository.findOneBy({ usernameLower, host }) as IRemoteUser | null; | 		const user = await this.usersRepository.findOneBy({ usernameLower, host }) as RemoteUser | null; | ||||||
| 	 | 	 | ||||||
| 		const acctLower = `${usernameLower}@${host}`; | 		const acctLower = `${usernameLower}@${host}`; | ||||||
| 	 | 	 | ||||||
| @@ -82,7 +82,7 @@ export class RemoteUserResolveService { | |||||||
| 			const self = await this.resolveSelf(acctLower); | 			const self = await this.resolveSelf(acctLower); | ||||||
| 	 | 	 | ||||||
| 			if (user.uri !== self.href) { | 			if (user.uri !== self.href) { | ||||||
| 				// if uri mismatch, Fix (user@host <=> AP's Person id(IRemoteUser.uri)) mapping. | 				// if uri mismatch, Fix (user@host <=> AP's Person id(RemoteUser.uri)) mapping. | ||||||
| 				this.logger.info(`uri missmatch: ${acctLower}`); | 				this.logger.info(`uri missmatch: ${acctLower}`); | ||||||
| 				this.logger.info(`recovery missmatch uri for (username=${username}, host=${host}) from ${user.uri} to ${self.href}`); | 				this.logger.info(`recovery missmatch uri for (username=${username}, host=${host}) from ${user.uri} to ${self.href}`); | ||||||
| 	 | 	 | ||||||
|   | |||||||
| @@ -3,7 +3,7 @@ import Redis from 'ioredis'; | |||||||
| import { In } from 'typeorm'; | import { In } from 'typeorm'; | ||||||
| import type { Role, RoleAssignment, RoleAssignmentsRepository, RolesRepository, UsersRepository } from '@/models/index.js'; | import type { Role, RoleAssignment, RoleAssignmentsRepository, RolesRepository, UsersRepository } from '@/models/index.js'; | ||||||
| import { Cache } from '@/misc/cache.js'; | import { Cache } from '@/misc/cache.js'; | ||||||
| import type { CacheableLocalUser, CacheableUser, ILocalUser, User } from '@/models/entities/User.js'; | import type { User } from '@/models/entities/User.js'; | ||||||
| import { DI } from '@/di-symbols.js'; | import { DI } from '@/di-symbols.js'; | ||||||
| import { bindThis } from '@/decorators.js'; | import { bindThis } from '@/decorators.js'; | ||||||
| import { MetaService } from '@/core/MetaService.js'; | import { MetaService } from '@/core/MetaService.js'; | ||||||
| @@ -211,8 +211,14 @@ export class RoleService implements OnApplicationShutdown { | |||||||
| 		const assignedRoleIds = assigns.map(x => x.roleId); | 		const assignedRoleIds = assigns.map(x => x.roleId); | ||||||
| 		const roles = await this.rolesCache.fetch(null, () => this.rolesRepository.findBy({})); | 		const roles = await this.rolesCache.fetch(null, () => this.rolesRepository.findBy({})); | ||||||
| 		const assignedBadgeRoles = roles.filter(r => r.asBadge && assignedRoleIds.includes(r.id)); | 		const assignedBadgeRoles = roles.filter(r => r.asBadge && assignedRoleIds.includes(r.id)); | ||||||
| 		// コンディショナルロールも含めるのは負荷高そうだから一旦無し | 		const badgeCondRoles = roles.filter(r => r.asBadge && (r.target === 'conditional')); | ||||||
| 		return assignedBadgeRoles; | 		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 | ||||||
|   | |||||||
| @@ -2,7 +2,7 @@ | |||||||
| import { Inject, Injectable, OnApplicationShutdown } from '@nestjs/common'; | import { Inject, Injectable, OnApplicationShutdown } from '@nestjs/common'; | ||||||
| import Redis from 'ioredis'; | 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 { User } from '@/models/entities/User.js'; | ||||||
| import type { Blocking } from '@/models/entities/Blocking.js'; | import type { Blocking } from '@/models/entities/Blocking.js'; | ||||||
| import { QueueService } from '@/core/QueueService.js'; | import { QueueService } from '@/core/QueueService.js'; | ||||||
| import { GlobalEventService } from '@/core/GlobalEventService.js'; | import { GlobalEventService } from '@/core/GlobalEventService.js'; | ||||||
| @@ -117,7 +117,7 @@ export class UserBlockingService implements OnApplicationShutdown { | |||||||
| 		}); | 		}); | ||||||
|  |  | ||||||
| 		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.addContext(this.apRendererService.renderBlock(blocking)); | ||||||
| 			this.queueService.deliver(blocker, content, blockee.inbox); | 			this.queueService.deliver(blocker, content, blockee.inbox); | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| @@ -162,13 +162,13 @@ export class UserBlockingService implements OnApplicationShutdown { | |||||||
|  |  | ||||||
| 		// リモートにフォローリクエストをしていたらUndoFollow送信 | 		// リモートにフォローリクエストをしていたらUndoFollow送信 | ||||||
| 		if (this.userEntityService.isLocalUser(follower) && this.userEntityService.isRemoteUser(followee)) { | 		if (this.userEntityService.isLocalUser(follower) && this.userEntityService.isRemoteUser(followee)) { | ||||||
| 			const content = this.apRendererService.renderActivity(this.apRendererService.renderUndo(this.apRendererService.renderFollow(follower, followee), follower)); | 			const content = this.apRendererService.addContext(this.apRendererService.renderUndo(this.apRendererService.renderFollow(follower, followee), follower)); | ||||||
| 			this.queueService.deliver(follower, content, followee.inbox); | 			this.queueService.deliver(follower, content, followee.inbox); | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		// リモートからフォローリクエストを受けていたらReject送信 | 		// リモートからフォローリクエストを受けていたらReject送信 | ||||||
| 		if (this.userEntityService.isRemoteUser(follower) && this.userEntityService.isLocalUser(followee)) { | 		if (this.userEntityService.isRemoteUser(follower) && this.userEntityService.isLocalUser(followee)) { | ||||||
| 			const content = this.apRendererService.renderActivity(this.apRendererService.renderReject(this.apRendererService.renderFollow(follower, followee, request.requestId!), followee)); | 			const content = this.apRendererService.addContext(this.apRendererService.renderReject(this.apRendererService.renderFollow(follower, followee, request.requestId!), followee)); | ||||||
| 			this.queueService.deliver(followee, content, follower.inbox); | 			this.queueService.deliver(followee, content, follower.inbox); | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| @@ -210,13 +210,13 @@ export class UserBlockingService implements OnApplicationShutdown { | |||||||
|  |  | ||||||
| 		// リモートにフォローをしていたらUndoFollow送信 | 		// リモートにフォローをしていたらUndoFollow送信 | ||||||
| 		if (this.userEntityService.isLocalUser(follower) && this.userEntityService.isRemoteUser(followee)) { | 		if (this.userEntityService.isLocalUser(follower) && this.userEntityService.isRemoteUser(followee)) { | ||||||
| 			const content = this.apRendererService.renderActivity(this.apRendererService.renderUndo(this.apRendererService.renderFollow(follower, followee), follower)); | 			const content = this.apRendererService.addContext(this.apRendererService.renderUndo(this.apRendererService.renderFollow(follower, followee), follower)); | ||||||
| 			this.queueService.deliver(follower, content, followee.inbox); | 			this.queueService.deliver(follower, content, followee.inbox); | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		// リモートからフォローをされていたらRejectFollow送信 | 		// リモートからフォローをされていたらRejectFollow送信 | ||||||
| 		if (this.userEntityService.isLocalUser(followee) && this.userEntityService.isRemoteUser(follower)) { | 		if (this.userEntityService.isLocalUser(followee) && this.userEntityService.isRemoteUser(follower)) { | ||||||
| 			const content = this.apRendererService.renderActivity(this.apRendererService.renderReject(this.apRendererService.renderFollow(follower, followee), followee)); | 			const content = this.apRendererService.addContext(this.apRendererService.renderReject(this.apRendererService.renderFollow(follower, followee), followee)); | ||||||
| 			this.queueService.deliver(followee, content, follower.inbox); | 			this.queueService.deliver(followee, content, follower.inbox); | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| @@ -236,7 +236,7 @@ export class UserBlockingService implements OnApplicationShutdown { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public async unblock(blocker: CacheableUser, blockee: CacheableUser) { | 	public async unblock(blocker: User, blockee: User) { | ||||||
| 		const blocking = await this.blockingsRepository.findOneBy({ | 		const blocking = await this.blockingsRepository.findOneBy({ | ||||||
| 			blockerId: blocker.id, | 			blockerId: blocker.id, | ||||||
| 			blockeeId: blockee.id, | 			blockeeId: blockee.id, | ||||||
| @@ -261,7 +261,7 @@ export class UserBlockingService implements OnApplicationShutdown { | |||||||
|  |  | ||||||
| 		// 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.addContext(this.apRendererService.renderUndo(this.apRendererService.renderBlock(blocking), blocker)); | ||||||
| 			this.queueService.deliver(blocker, content, blockee.inbox); | 			this.queueService.deliver(blocker, content, blockee.inbox); | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|   | |||||||
| @@ -2,7 +2,7 @@ import { Inject, Injectable } from '@nestjs/common'; | |||||||
| import Redis from 'ioredis'; | import Redis from 'ioredis'; | ||||||
| import type { UsersRepository } from '@/models/index.js'; | import type { UsersRepository } from '@/models/index.js'; | ||||||
| import { Cache } from '@/misc/cache.js'; | import { Cache } from '@/misc/cache.js'; | ||||||
| import type { CacheableLocalUser, CacheableUser, ILocalUser, User } from '@/models/entities/User.js'; | import type { LocalUser, User } from '@/models/entities/User.js'; | ||||||
| import { DI } from '@/di-symbols.js'; | import { DI } from '@/di-symbols.js'; | ||||||
| import { UserEntityService } from '@/core/entities/UserEntityService.js'; | import { UserEntityService } from '@/core/entities/UserEntityService.js'; | ||||||
| import { bindThis } from '@/decorators.js'; | import { bindThis } from '@/decorators.js'; | ||||||
| @@ -11,10 +11,10 @@ import type { OnApplicationShutdown } from '@nestjs/common'; | |||||||
|  |  | ||||||
| @Injectable() | @Injectable() | ||||||
| export class UserCacheService implements OnApplicationShutdown { | export class UserCacheService implements OnApplicationShutdown { | ||||||
| 	public userByIdCache: Cache<CacheableUser>; | 	public userByIdCache: Cache<User>; | ||||||
| 	public localUserByNativeTokenCache: Cache<CacheableLocalUser | null>; | 	public localUserByNativeTokenCache: Cache<LocalUser | null>; | ||||||
| 	public localUserByIdCache: Cache<CacheableLocalUser>; | 	public localUserByIdCache: Cache<LocalUser>; | ||||||
| 	public uriPersonCache: Cache<CacheableUser | null>; | 	public uriPersonCache: Cache<User | null>; | ||||||
|  |  | ||||||
| 	constructor( | 	constructor( | ||||||
| 		@Inject(DI.redisSubscriber) | 		@Inject(DI.redisSubscriber) | ||||||
| @@ -27,10 +27,10 @@ export class UserCacheService implements OnApplicationShutdown { | |||||||
| 	) { | 	) { | ||||||
| 		//this.onMessage = this.onMessage.bind(this); | 		//this.onMessage = this.onMessage.bind(this); | ||||||
|  |  | ||||||
| 		this.userByIdCache = new Cache<CacheableUser>(Infinity); | 		this.userByIdCache = new Cache<User>(Infinity); | ||||||
| 		this.localUserByNativeTokenCache = new Cache<CacheableLocalUser | null>(Infinity); | 		this.localUserByNativeTokenCache = new Cache<LocalUser | null>(Infinity); | ||||||
| 		this.localUserByIdCache = new Cache<CacheableLocalUser>(Infinity); | 		this.localUserByIdCache = new Cache<LocalUser>(Infinity); | ||||||
| 		this.uriPersonCache = new Cache<CacheableUser | null>(Infinity); | 		this.uriPersonCache = new Cache<User | null>(Infinity); | ||||||
|  |  | ||||||
| 		this.redisSubscriber.on('message', this.onMessage); | 		this.redisSubscriber.on('message', this.onMessage); | ||||||
| 	} | 	} | ||||||
| @@ -58,7 +58,7 @@ export class UserCacheService implements OnApplicationShutdown { | |||||||
| 					break; | 					break; | ||||||
| 				} | 				} | ||||||
| 				case 'userTokenRegenerated': { | 				case 'userTokenRegenerated': { | ||||||
| 					const user = await this.usersRepository.findOneByOrFail({ id: body.id }) as ILocalUser; | 					const user = await this.usersRepository.findOneByOrFail({ id: body.id }) as LocalUser; | ||||||
| 					this.localUserByNativeTokenCache.delete(body.oldToken); | 					this.localUserByNativeTokenCache.delete(body.oldToken); | ||||||
| 					this.localUserByNativeTokenCache.set(body.newToken, user); | 					this.localUserByNativeTokenCache.set(body.newToken, user); | ||||||
| 					break; | 					break; | ||||||
|   | |||||||
| @@ -1,5 +1,5 @@ | |||||||
| import { Inject, Injectable } from '@nestjs/common'; | import { Inject, Injectable } from '@nestjs/common'; | ||||||
| import type { CacheableUser, ILocalUser, IRemoteUser, User } from '@/models/entities/User.js'; | import type { LocalUser, RemoteUser, User } from '@/models/entities/User.js'; | ||||||
| import { IdentifiableError } from '@/misc/identifiable-error.js'; | import { IdentifiableError } from '@/misc/identifiable-error.js'; | ||||||
| import { QueueService } from '@/core/QueueService.js'; | import { QueueService } from '@/core/QueueService.js'; | ||||||
| import PerUserFollowingChart from '@/core/chart/charts/per-user-following.js'; | import PerUserFollowingChart from '@/core/chart/charts/per-user-following.js'; | ||||||
| @@ -21,16 +21,16 @@ import Logger from '../logger.js'; | |||||||
|  |  | ||||||
| const logger = new Logger('following/create'); | const logger = new Logger('following/create'); | ||||||
|  |  | ||||||
| type Local = ILocalUser | { | type Local = LocalUser | { | ||||||
| 	id: ILocalUser['id']; | 	id: LocalUser['id']; | ||||||
| 	host: ILocalUser['host']; | 	host: LocalUser['host']; | ||||||
| 	uri: ILocalUser['uri'] | 	uri: LocalUser['uri'] | ||||||
| }; | }; | ||||||
| type Remote = IRemoteUser | { | type Remote = RemoteUser | { | ||||||
| 	id: IRemoteUser['id']; | 	id: RemoteUser['id']; | ||||||
| 	host: IRemoteUser['host']; | 	host: RemoteUser['host']; | ||||||
| 	uri: IRemoteUser['uri']; | 	uri: RemoteUser['uri']; | ||||||
| 	inbox: IRemoteUser['inbox']; | 	inbox: RemoteUser['inbox']; | ||||||
| }; | }; | ||||||
| type Both = Local | Remote; | type Both = Local | Remote; | ||||||
|  |  | ||||||
| @@ -81,7 +81,7 @@ export class UserFollowingService { | |||||||
|  |  | ||||||
| 		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.addContext(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) { | ||||||
| @@ -130,7 +130,7 @@ export class UserFollowingService { | |||||||
| 		await this.insertFollowingDoc(followee, follower); | 		await this.insertFollowingDoc(followee, follower); | ||||||
|  |  | ||||||
| 		if (this.userEntityService.isRemoteUser(follower) && this.userEntityService.isLocalUser(followee)) { | 		if (this.userEntityService.isRemoteUser(follower) && this.userEntityService.isLocalUser(followee)) { | ||||||
| 			const content = this.apRendererService.renderActivity(this.apRendererService.renderAccept(this.apRendererService.renderFollow(follower, followee, requestId), followee)); | 			const content = this.apRendererService.addContext(this.apRendererService.renderAccept(this.apRendererService.renderFollow(follower, followee, requestId), followee)); | ||||||
| 			this.queueService.deliver(followee, content, follower.inbox); | 			this.queueService.deliver(followee, content, follower.inbox); | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| @@ -293,13 +293,13 @@ export class UserFollowingService { | |||||||
| 		} | 		} | ||||||
| 	 | 	 | ||||||
| 		if (this.userEntityService.isLocalUser(follower) && this.userEntityService.isRemoteUser(followee)) { | 		if (this.userEntityService.isLocalUser(follower) && this.userEntityService.isRemoteUser(followee)) { | ||||||
| 			const content = this.apRendererService.renderActivity(this.apRendererService.renderUndo(this.apRendererService.renderFollow(follower, followee), follower)); | 			const content = this.apRendererService.addContext(this.apRendererService.renderUndo(this.apRendererService.renderFollow(follower, followee), follower)); | ||||||
| 			this.queueService.deliver(follower, content, followee.inbox); | 			this.queueService.deliver(follower, content, followee.inbox); | ||||||
| 		} | 		} | ||||||
| 	 | 	 | ||||||
| 		if (this.userEntityService.isLocalUser(followee) && this.userEntityService.isRemoteUser(follower)) { | 		if (this.userEntityService.isLocalUser(followee) && this.userEntityService.isRemoteUser(follower)) { | ||||||
| 			// local user has null host | 			// local user has null host | ||||||
| 			const content = this.apRendererService.renderActivity(this.apRendererService.renderReject(this.apRendererService.renderFollow(follower, followee), followee)); | 			const content = this.apRendererService.addContext(this.apRendererService.renderReject(this.apRendererService.renderFollow(follower, followee), followee)); | ||||||
| 			this.queueService.deliver(followee, content, follower.inbox); | 			this.queueService.deliver(followee, content, follower.inbox); | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| @@ -388,7 +388,7 @@ export class UserFollowingService { | |||||||
| 		} | 		} | ||||||
| 	 | 	 | ||||||
| 		if (this.userEntityService.isLocalUser(follower) && this.userEntityService.isRemoteUser(followee)) { | 		if (this.userEntityService.isLocalUser(follower) && this.userEntityService.isRemoteUser(followee)) { | ||||||
| 			const content = this.apRendererService.renderActivity(this.apRendererService.renderFollow(follower, followee)); | 			const content = this.apRendererService.addContext(this.apRendererService.renderFollow(follower, followee)); | ||||||
| 			this.queueService.deliver(follower, content, followee.inbox); | 			this.queueService.deliver(follower, content, followee.inbox); | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| @@ -403,7 +403,7 @@ export class UserFollowingService { | |||||||
| 		}, | 		}, | ||||||
| 	): Promise<void> { | 	): Promise<void> { | ||||||
| 		if (this.userEntityService.isRemoteUser(followee)) { | 		if (this.userEntityService.isRemoteUser(followee)) { | ||||||
| 			const content = this.apRendererService.renderActivity(this.apRendererService.renderUndo(this.apRendererService.renderFollow(follower, followee), follower)); | 			const content = this.apRendererService.addContext(this.apRendererService.renderUndo(this.apRendererService.renderFollow(follower, followee), follower)); | ||||||
| 	 | 	 | ||||||
| 			if (this.userEntityService.isLocalUser(follower)) { // 本来このチェックは不要だけどTSに怒られるので | 			if (this.userEntityService.isLocalUser(follower)) { // 本来このチェックは不要だけどTSに怒られるので | ||||||
| 				this.queueService.deliver(follower, content, followee.inbox); | 				this.queueService.deliver(follower, content, followee.inbox); | ||||||
| @@ -434,7 +434,7 @@ export class UserFollowingService { | |||||||
| 		followee: { | 		followee: { | ||||||
| 			id: User['id']; host: User['host']; uri: User['host']; inbox: User['inbox']; sharedInbox: User['sharedInbox']; | 			id: User['id']; host: User['host']; uri: User['host']; inbox: User['inbox']; sharedInbox: User['sharedInbox']; | ||||||
| 		}, | 		}, | ||||||
| 		follower: CacheableUser, | 		follower: User, | ||||||
| 	): Promise<void> { | 	): Promise<void> { | ||||||
| 		const request = await this.followRequestsRepository.findOneBy({ | 		const request = await this.followRequestsRepository.findOneBy({ | ||||||
| 			followeeId: followee.id, | 			followeeId: followee.id, | ||||||
| @@ -448,7 +448,7 @@ export class UserFollowingService { | |||||||
| 		await this.insertFollowingDoc(followee, follower); | 		await this.insertFollowingDoc(followee, follower); | ||||||
| 	 | 	 | ||||||
| 		if (this.userEntityService.isRemoteUser(follower) && this.userEntityService.isLocalUser(followee)) { | 		if (this.userEntityService.isRemoteUser(follower) && this.userEntityService.isLocalUser(followee)) { | ||||||
| 			const content = this.apRendererService.renderActivity(this.apRendererService.renderAccept(this.apRendererService.renderFollow(follower, followee, request.requestId!), followee)); | 			const content = this.apRendererService.addContext(this.apRendererService.renderAccept(this.apRendererService.renderFollow(follower, followee, request.requestId!), followee)); | ||||||
| 			this.queueService.deliver(followee, content, follower.inbox); | 			this.queueService.deliver(followee, content, follower.inbox); | ||||||
| 		} | 		} | ||||||
| 	 | 	 | ||||||
| @@ -556,7 +556,7 @@ export class UserFollowingService { | |||||||
| 			followerId: follower.id, | 			followerId: follower.id, | ||||||
| 		}); | 		}); | ||||||
|  |  | ||||||
| 		const content = this.apRendererService.renderActivity(this.apRendererService.renderReject(this.apRendererService.renderFollow(follower, followee, request?.requestId ?? undefined), followee)); | 		const content = this.apRendererService.addContext(this.apRendererService.renderReject(this.apRendererService.renderFollow(follower, followee, request?.requestId ?? undefined), followee)); | ||||||
| 		this.queueService.deliver(followee, content, follower.inbox); | 		this.queueService.deliver(followee, content, follower.inbox); | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|   | |||||||
| @@ -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, | ||||||
| @@ -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({ | ||||||
|   | |||||||
| @@ -35,7 +35,7 @@ export class UserSuspendService { | |||||||
| 	 | 	 | ||||||
| 		if (this.userEntityService.isLocalUser(user)) { | 		if (this.userEntityService.isLocalUser(user)) { | ||||||
| 			// 知り得る全SharedInboxにDelete配信 | 			// 知り得る全SharedInboxにDelete配信 | ||||||
| 			const content = this.apRendererService.renderActivity(this.apRendererService.renderDelete(`${this.config.url}/users/${user.id}`, user)); | 			const content = this.apRendererService.addContext(this.apRendererService.renderDelete(`${this.config.url}/users/${user.id}`, user)); | ||||||
| 	 | 	 | ||||||
| 			const queue: string[] = []; | 			const queue: string[] = []; | ||||||
| 	 | 	 | ||||||
| @@ -65,7 +65,7 @@ export class UserSuspendService { | |||||||
| 	 | 	 | ||||||
| 		if (this.userEntityService.isLocalUser(user)) { | 		if (this.userEntityService.isLocalUser(user)) { | ||||||
| 			// 知り得る全SharedInboxにUndo Delete配信 | 			// 知り得る全SharedInboxにUndo Delete配信 | ||||||
| 			const content = this.apRendererService.renderActivity(this.apRendererService.renderUndo(this.apRendererService.renderDelete(`${this.config.url}/users/${user.id}`, user), user)); | 			const content = this.apRendererService.addContext(this.apRendererService.renderUndo(this.apRendererService.renderDelete(`${this.config.url}/users/${user.id}`, user), user)); | ||||||
| 	 | 	 | ||||||
| 			const queue: string[] = []; | 			const queue: string[] = []; | ||||||
| 	 | 	 | ||||||
|   | |||||||
| @@ -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, | ||||||
|  | 			}) | ||||||
|  | 		); | ||||||
|  | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,11 +1,9 @@ | |||||||
| import { Inject, Injectable } from '@nestjs/common'; | import { Injectable } from '@nestjs/common'; | ||||||
| import { In } from 'typeorm'; |  | ||||||
| import promiseLimit from 'promise-limit'; | import promiseLimit from 'promise-limit'; | ||||||
| import { DI } from '@/di-symbols.js'; | import type { RemoteUser, User } from '@/models/entities/User.js'; | ||||||
| import type { CacheableRemoteUser, CacheableUser } from '@/models/entities/User.js'; | import { concat, unique } from '@/misc/prelude/array.js'; | ||||||
| import { concat, toArray, toSingle, unique } from '@/misc/prelude/array.js'; |  | ||||||
| import { bindThis } from '@/decorators.js'; | import { bindThis } from '@/decorators.js'; | ||||||
| import { getApId, getApIds, getApType, isAccept, isActor, isAdd, isAnnounce, isBlock, isCollection, isCollectionOrOrderedCollection, isCreate, isDelete, isFlag, isFollow, isLike, isPost, isRead, isReject, isRemove, isTombstone, isUndo, isUpdate, validActor, validPost } from './type.js'; | import { getApIds } from './type.js'; | ||||||
| import { ApPersonService } from './models/ApPersonService.js'; | import { ApPersonService } from './models/ApPersonService.js'; | ||||||
| import type { ApObject } from './type.js'; | import type { ApObject } from './type.js'; | ||||||
| import type { Resolver } from './ApResolverService.js'; | import type { Resolver } from './ApResolverService.js'; | ||||||
| @@ -14,8 +12,8 @@ type Visibility = 'public' | 'home' | 'followers' | 'specified'; | |||||||
|  |  | ||||||
| type AudienceInfo = { | type AudienceInfo = { | ||||||
| 	visibility: Visibility, | 	visibility: Visibility, | ||||||
| 	mentionedUsers: CacheableUser[], | 	mentionedUsers: User[], | ||||||
| 	visibleUsers: CacheableUser[], | 	visibleUsers: User[], | ||||||
| }; | }; | ||||||
|  |  | ||||||
| @Injectable() | @Injectable() | ||||||
| @@ -26,16 +24,16 @@ export class ApAudienceService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public async parseAudience(actor: CacheableRemoteUser, to?: ApObject, cc?: ApObject, resolver?: Resolver): Promise<AudienceInfo> { | 	public async parseAudience(actor: RemoteUser, to?: ApObject, cc?: ApObject, resolver?: Resolver): Promise<AudienceInfo> { | ||||||
| 		const toGroups = this.groupingAudience(getApIds(to), actor); | 		const toGroups = this.groupingAudience(getApIds(to), actor); | ||||||
| 		const ccGroups = this.groupingAudience(getApIds(cc), actor); | 		const ccGroups = this.groupingAudience(getApIds(cc), actor); | ||||||
| 	 | 	 | ||||||
| 		const others = unique(concat([toGroups.other, ccGroups.other])); | 		const others = unique(concat([toGroups.other, ccGroups.other])); | ||||||
| 	 | 	 | ||||||
| 		const limit = promiseLimit<CacheableUser | null>(2); | 		const limit = promiseLimit<User | null>(2); | ||||||
| 		const mentionedUsers = (await Promise.all( | 		const mentionedUsers = (await Promise.all( | ||||||
| 			others.map(id => limit(() => this.apPersonService.resolvePerson(id, resolver).catch(() => null))), | 			others.map(id => limit(() => this.apPersonService.resolvePerson(id, resolver).catch(() => null))), | ||||||
| 		)).filter((x): x is CacheableUser => x != null); | 		)).filter((x): x is User => x != null); | ||||||
| 	 | 	 | ||||||
| 		if (toGroups.public.length > 0) { | 		if (toGroups.public.length > 0) { | ||||||
| 			return { | 			return { | ||||||
| @@ -69,7 +67,7 @@ export class ApAudienceService { | |||||||
| 	} | 	} | ||||||
| 	 | 	 | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	private groupingAudience(ids: string[], actor: CacheableRemoteUser) { | 	private groupingAudience(ids: string[], actor: RemoteUser) { | ||||||
| 		const groups = { | 		const groups = { | ||||||
| 			public: [] as string[], | 			public: [] as string[], | ||||||
| 			followers: [] as string[], | 			followers: [] as string[], | ||||||
| @@ -101,7 +99,7 @@ export class ApAudienceService { | |||||||
| 	} | 	} | ||||||
| 	 | 	 | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	private isFollowers(id: string, actor: CacheableRemoteUser) { | 	private isFollowers(id: string, actor: RemoteUser) { | ||||||
| 		return ( | 		return ( | ||||||
| 			id === (actor.followersUri ?? `${actor.uri}/followers`) | 			id === (actor.followersUri ?? `${actor.uri}/followers`) | ||||||
| 		); | 		); | ||||||
|   | |||||||
| @@ -1,15 +1,14 @@ | |||||||
| import { Inject, Injectable } from '@nestjs/common'; | import { Inject, Injectable } from '@nestjs/common'; | ||||||
| import escapeRegexp from 'escape-regexp'; | import escapeRegexp from 'escape-regexp'; | ||||||
| import { DI } from '@/di-symbols.js'; | import { DI } from '@/di-symbols.js'; | ||||||
| import type { MessagingMessagesRepository, NotesRepository, UserPublickeysRepository, UsersRepository } from '@/models/index.js'; | import type { NotesRepository, UserPublickeysRepository, UsersRepository } from '@/models/index.js'; | ||||||
| import type { Config } from '@/config.js'; | import type { Config } from '@/config.js'; | ||||||
| import type { CacheableRemoteUser, CacheableUser } from '@/models/entities/User.js'; |  | ||||||
| import { Cache } from '@/misc/cache.js'; | import { Cache } from '@/misc/cache.js'; | ||||||
| import type { UserPublickey } from '@/models/entities/UserPublickey.js'; | import type { UserPublickey } from '@/models/entities/UserPublickey.js'; | ||||||
| import { UserCacheService } from '@/core/UserCacheService.js'; | import { UserCacheService } from '@/core/UserCacheService.js'; | ||||||
| import type { Note } from '@/models/entities/Note.js'; | import type { Note } from '@/models/entities/Note.js'; | ||||||
| import type { MessagingMessage } from '@/models/entities/MessagingMessage.js'; |  | ||||||
| import { bindThis } from '@/decorators.js'; | import { bindThis } from '@/decorators.js'; | ||||||
|  | import { RemoteUser, User } from '@/models/entities/User.js'; | ||||||
| import { getApId } from './type.js'; | import { getApId } from './type.js'; | ||||||
| import { ApPersonService } from './models/ApPersonService.js'; | import { ApPersonService } from './models/ApPersonService.js'; | ||||||
| import type { IObject } from './type.js'; | import type { IObject } from './type.js'; | ||||||
| @@ -42,9 +41,6 @@ export class ApDbResolverService { | |||||||
| 		@Inject(DI.usersRepository) | 		@Inject(DI.usersRepository) | ||||||
| 		private usersRepository: UsersRepository, | 		private usersRepository: UsersRepository, | ||||||
|  |  | ||||||
| 		@Inject(DI.messagingMessagesRepository) |  | ||||||
| 		private messagingMessagesRepository: MessagingMessagesRepository, |  | ||||||
|  |  | ||||||
| 		@Inject(DI.notesRepository) | 		@Inject(DI.notesRepository) | ||||||
| 		private notesRepository: NotesRepository, | 		private notesRepository: NotesRepository, | ||||||
|  |  | ||||||
| @@ -101,28 +97,11 @@ export class ApDbResolverService { | |||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis |  | ||||||
| 	public async getMessageFromApId(value: string | IObject): Promise<MessagingMessage | null> { |  | ||||||
| 		const parsed = this.parseUri(value); |  | ||||||
|  |  | ||||||
| 		if (parsed.local) { |  | ||||||
| 			if (parsed.type !== 'notes') return null; |  | ||||||
|  |  | ||||||
| 			return await this.messagingMessagesRepository.findOneBy({ |  | ||||||
| 				id: parsed.id, |  | ||||||
| 			}); |  | ||||||
| 		} else { |  | ||||||
| 			return await this.messagingMessagesRepository.findOneBy({ |  | ||||||
| 				uri: parsed.uri, |  | ||||||
| 			}); |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	/** | 	/** | ||||||
| 	 * AP Person => Misskey User in DB | 	 * AP Person => Misskey User in DB | ||||||
| 	 */ | 	 */ | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public async getUserFromApId(value: string | IObject): Promise<CacheableUser | null> { | 	public async getUserFromApId(value: string | IObject): Promise<User | null> { | ||||||
| 		const parsed = this.parseUri(value); | 		const parsed = this.parseUri(value); | ||||||
|  |  | ||||||
| 		if (parsed.local) { | 		if (parsed.local) { | ||||||
| @@ -143,7 +122,7 @@ export class ApDbResolverService { | |||||||
| 	 */ | 	 */ | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public async getAuthUserFromKeyId(keyId: string): Promise<{ | 	public async getAuthUserFromKeyId(keyId: string): Promise<{ | ||||||
| 		user: CacheableRemoteUser; | 		user: RemoteUser; | ||||||
| 		key: UserPublickey; | 		key: UserPublickey; | ||||||
| 	} | null> { | 	} | null> { | ||||||
| 		const key = await this.publicKeyCache.fetch(keyId, async () => { | 		const key = await this.publicKeyCache.fetch(keyId, async () => { | ||||||
| @@ -159,7 +138,7 @@ export class ApDbResolverService { | |||||||
| 		if (key == null) return null; | 		if (key == null) return null; | ||||||
|  |  | ||||||
| 		return { | 		return { | ||||||
| 			user: await this.userCacheService.findById(key.userId) as CacheableRemoteUser, | 			user: await this.userCacheService.findById(key.userId) as RemoteUser, | ||||||
| 			key, | 			key, | ||||||
| 		}; | 		}; | ||||||
| 	} | 	} | ||||||
| @@ -169,10 +148,10 @@ export class ApDbResolverService { | |||||||
| 	 */ | 	 */ | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public async getAuthUserFromApId(uri: string): Promise<{ | 	public async getAuthUserFromApId(uri: string): Promise<{ | ||||||
| 		user: CacheableRemoteUser; | 		user: RemoteUser; | ||||||
| 		key: UserPublickey | null; | 		key: UserPublickey | null; | ||||||
| 	} | null> { | 	} | null> { | ||||||
| 		const user = await this.apPersonService.resolvePerson(uri) as CacheableRemoteUser; | 		const user = await this.apPersonService.resolvePerson(uri) as RemoteUser; | ||||||
|  |  | ||||||
| 		if (user == null) return null; | 		if (user == null) return null; | ||||||
|  |  | ||||||
|   | |||||||
| @@ -3,7 +3,7 @@ import { IsNull, Not } from 'typeorm'; | |||||||
| import { DI } from '@/di-symbols.js'; | import { DI } from '@/di-symbols.js'; | ||||||
| import type { FollowingsRepository, UsersRepository } from '@/models/index.js'; | import type { FollowingsRepository, UsersRepository } from '@/models/index.js'; | ||||||
| import type { Config } from '@/config.js'; | import type { Config } from '@/config.js'; | ||||||
| import type { ILocalUser, IRemoteUser, User } from '@/models/entities/User.js'; | import type { LocalUser, RemoteUser, User } from '@/models/entities/User.js'; | ||||||
| import { QueueService } from '@/core/QueueService.js'; | import { QueueService } from '@/core/QueueService.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'; | ||||||
| @@ -18,7 +18,7 @@ interface IFollowersRecipe extends IRecipe { | |||||||
|  |  | ||||||
| interface IDirectRecipe extends IRecipe { | interface IDirectRecipe extends IRecipe { | ||||||
| 	type: 'Direct'; | 	type: 'Direct'; | ||||||
| 	to: IRemoteUser; | 	to: RemoteUser; | ||||||
| } | } | ||||||
|  |  | ||||||
| const isFollowers = (recipe: any): recipe is IFollowersRecipe => | const isFollowers = (recipe: any): recipe is IFollowersRecipe => | ||||||
| @@ -50,7 +50,7 @@ export class ApDeliverManagerService { | |||||||
| 	 * @param from Followee | 	 * @param from Followee | ||||||
| 	 */ | 	 */ | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public async deliverToFollowers(actor: { id: ILocalUser['id']; host: null; }, activity: any) { | 	public async deliverToFollowers(actor: { id: LocalUser['id']; host: null; }, activity: any) { | ||||||
| 		const manager = new DeliverManager( | 		const manager = new DeliverManager( | ||||||
| 			this.userEntityService, | 			this.userEntityService, | ||||||
| 			this.followingsRepository, | 			this.followingsRepository, | ||||||
| @@ -68,7 +68,7 @@ export class ApDeliverManagerService { | |||||||
| 	 * @param to Target user | 	 * @param to Target user | ||||||
| 	 */ | 	 */ | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public async deliverToUser(actor: { id: ILocalUser['id']; host: null; }, activity: any, to: IRemoteUser) { | 	public async deliverToUser(actor: { id: LocalUser['id']; host: null; }, activity: any, to: RemoteUser) { | ||||||
| 		const manager = new DeliverManager( | 		const manager = new DeliverManager( | ||||||
| 			this.userEntityService, | 			this.userEntityService, | ||||||
| 			this.followingsRepository, | 			this.followingsRepository, | ||||||
| @@ -132,7 +132,7 @@ class DeliverManager { | |||||||
| 	 * @param to To | 	 * @param to To | ||||||
| 	 */ | 	 */ | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public addDirectRecipe(to: IRemoteUser) { | 	public addDirectRecipe(to: RemoteUser) { | ||||||
| 		const recipe = { | 		const recipe = { | ||||||
| 			type: 'Direct', | 			type: 'Direct', | ||||||
| 			to, | 			to, | ||||||
|   | |||||||
| @@ -2,7 +2,6 @@ import { Inject, Injectable } from '@nestjs/common'; | |||||||
| import { In } from 'typeorm'; | import { In } from 'typeorm'; | ||||||
| import { DI } from '@/di-symbols.js'; | import { DI } from '@/di-symbols.js'; | ||||||
| import type { Config } from '@/config.js'; | import type { Config } from '@/config.js'; | ||||||
| import type { CacheableRemoteUser } from '@/models/entities/User.js'; |  | ||||||
| import { UserFollowingService } from '@/core/UserFollowingService.js'; | import { UserFollowingService } from '@/core/UserFollowingService.js'; | ||||||
| import { ReactionService } from '@/core/ReactionService.js'; | import { ReactionService } from '@/core/ReactionService.js'; | ||||||
| import { RelayService } from '@/core/RelayService.js'; | import { RelayService } from '@/core/RelayService.js'; | ||||||
| @@ -20,9 +19,10 @@ import { UtilityService } from '@/core/UtilityService.js'; | |||||||
| import { NoteEntityService } from '@/core/entities/NoteEntityService.js'; | import { NoteEntityService } from '@/core/entities/NoteEntityService.js'; | ||||||
| import { UserEntityService } from '@/core/entities/UserEntityService.js'; | import { UserEntityService } from '@/core/entities/UserEntityService.js'; | ||||||
| import { QueueService } from '@/core/QueueService.js'; | import { QueueService } from '@/core/QueueService.js'; | ||||||
| import { MessagingService } from '@/core/MessagingService.js'; | import type { UsersRepository, NotesRepository, FollowingsRepository, AbuseUserReportsRepository, FollowRequestsRepository } from '@/models/index.js'; | ||||||
| import type { UsersRepository, NotesRepository, FollowingsRepository, MessagingMessagesRepository, AbuseUserReportsRepository, FollowRequestsRepository } from '@/models/index.js'; | import { bindThis } from '@/decorators.js'; | ||||||
| import { getApId, getApIds, getApType, isAccept, isActor, isAdd, isAnnounce, isBlock, isCollection, isCollectionOrOrderedCollection, isCreate, isDelete, isFlag, isFollow, isLike, isPost, isRead, isReject, isRemove, isTombstone, isUndo, isUpdate, validActor, validPost } from './type.js'; | import type { RemoteUser } from '@/models/entities/User.js'; | ||||||
|  | import { getApId, getApIds, getApType, isAccept, isActor, isAdd, isAnnounce, isBlock, isCollection, isCollectionOrOrderedCollection, isCreate, isDelete, isFlag, isFollow, isLike, isPost, isReject, isRemove, isTombstone, isUndo, isUpdate, validActor, validPost } from './type.js'; | ||||||
| import { ApNoteService } from './models/ApNoteService.js'; | import { ApNoteService } from './models/ApNoteService.js'; | ||||||
| import { ApLoggerService } from './ApLoggerService.js'; | import { ApLoggerService } from './ApLoggerService.js'; | ||||||
| import { ApDbResolverService } from './ApDbResolverService.js'; | import { ApDbResolverService } from './ApDbResolverService.js'; | ||||||
| @@ -31,8 +31,7 @@ import { ApAudienceService } from './ApAudienceService.js'; | |||||||
| import { ApPersonService } from './models/ApPersonService.js'; | import { ApPersonService } from './models/ApPersonService.js'; | ||||||
| import { ApQuestionService } from './models/ApQuestionService.js'; | import { ApQuestionService } from './models/ApQuestionService.js'; | ||||||
| import type { Resolver } from './ApResolverService.js'; | import type { Resolver } from './ApResolverService.js'; | ||||||
| import type { IAccept, IAdd, IAnnounce, IBlock, ICreate, IDelete, IFlag, IFollow, ILike, IObject, IRead, IReject, IRemove, IUndo, IUpdate } from './type.js'; | import type { IAccept, IAdd, IAnnounce, IBlock, ICreate, IDelete, IFlag, IFollow, ILike, IObject, IReject, IRemove, IUndo, IUpdate } from './type.js'; | ||||||
| import { bindThis } from '@/decorators.js'; |  | ||||||
|  |  | ||||||
| @Injectable() | @Injectable() | ||||||
| export class ApInboxService { | export class ApInboxService { | ||||||
| @@ -51,9 +50,6 @@ export class ApInboxService { | |||||||
| 		@Inject(DI.followingsRepository) | 		@Inject(DI.followingsRepository) | ||||||
| 		private followingsRepository: FollowingsRepository, | 		private followingsRepository: FollowingsRepository, | ||||||
|  |  | ||||||
| 		@Inject(DI.messagingMessagesRepository) |  | ||||||
| 		private messagingMessagesRepository: MessagingMessagesRepository, |  | ||||||
|  |  | ||||||
| 		@Inject(DI.abuseUserReportsRepository) | 		@Inject(DI.abuseUserReportsRepository) | ||||||
| 		private abuseUserReportsRepository: AbuseUserReportsRepository, | 		private abuseUserReportsRepository: AbuseUserReportsRepository, | ||||||
|  |  | ||||||
| @@ -81,13 +77,12 @@ export class ApInboxService { | |||||||
| 		private apPersonService: ApPersonService, | 		private apPersonService: ApPersonService, | ||||||
| 		private apQuestionService: ApQuestionService, | 		private apQuestionService: ApQuestionService, | ||||||
| 		private queueService: QueueService, | 		private queueService: QueueService, | ||||||
| 		private messagingService: MessagingService, |  | ||||||
| 	) { | 	) { | ||||||
| 		this.logger = this.apLoggerService.logger; | 		this.logger = this.apLoggerService.logger; | ||||||
| 	} | 	} | ||||||
| 	 | 	 | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public async performActivity(actor: CacheableRemoteUser, activity: IObject) { | 	public async performActivity(actor: RemoteUser, activity: IObject) { | ||||||
| 		if (isCollectionOrOrderedCollection(activity)) { | 		if (isCollectionOrOrderedCollection(activity)) { | ||||||
| 			const resolver = this.apResolverService.createResolver(); | 			const resolver = this.apResolverService.createResolver(); | ||||||
| 			for (const item of toArray(isCollection(activity) ? activity.items : activity.orderedItems)) { | 			for (const item of toArray(isCollection(activity) ? activity.items : activity.orderedItems)) { | ||||||
| @@ -115,7 +110,7 @@ export class ApInboxService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public async performOneActivity(actor: CacheableRemoteUser, activity: IObject): Promise<void> { | 	public async performOneActivity(actor: RemoteUser, activity: IObject): Promise<void> { | ||||||
| 		if (actor.isSuspended) return; | 		if (actor.isSuspended) return; | ||||||
|  |  | ||||||
| 		if (isCreate(activity)) { | 		if (isCreate(activity)) { | ||||||
| @@ -124,8 +119,6 @@ export class ApInboxService { | |||||||
| 			await this.delete(actor, activity); | 			await this.delete(actor, activity); | ||||||
| 		} else if (isUpdate(activity)) { | 		} else if (isUpdate(activity)) { | ||||||
| 			await this.update(actor, activity); | 			await this.update(actor, activity); | ||||||
| 		} else if (isRead(activity)) { |  | ||||||
| 			await this.read(actor, activity); |  | ||||||
| 		} else if (isFollow(activity)) { | 		} else if (isFollow(activity)) { | ||||||
| 			await this.follow(actor, activity); | 			await this.follow(actor, activity); | ||||||
| 		} else if (isAccept(activity)) { | 		} else if (isAccept(activity)) { | ||||||
| @@ -152,7 +145,7 @@ export class ApInboxService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	private async follow(actor: CacheableRemoteUser, activity: IFollow): Promise<string> { | 	private async follow(actor: RemoteUser, activity: IFollow): Promise<string> { | ||||||
| 		const followee = await this.apDbResolverService.getUserFromApId(activity.object); | 		const followee = await this.apDbResolverService.getUserFromApId(activity.object); | ||||||
| 	 | 	 | ||||||
| 		if (followee == null) { | 		if (followee == null) { | ||||||
| @@ -168,7 +161,7 @@ export class ApInboxService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	private async like(actor: CacheableRemoteUser, activity: ILike): Promise<string> { | 	private async like(actor: RemoteUser, activity: ILike): Promise<string> { | ||||||
| 		const targetUri = getApId(activity.object); | 		const targetUri = getApId(activity.object); | ||||||
|  |  | ||||||
| 		const note = await this.apNoteService.fetchNote(targetUri); | 		const note = await this.apNoteService.fetchNote(targetUri); | ||||||
| @@ -186,30 +179,7 @@ export class ApInboxService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	private async read(actor: CacheableRemoteUser, activity: IRead): Promise<string> { | 	private async accept(actor: RemoteUser, activity: IAccept): Promise<string> { | ||||||
| 		const id = await getApId(activity.object); |  | ||||||
|  |  | ||||||
| 		if (!this.utilityService.isSelfHost(this.utilityService.extractDbHost(id))) { |  | ||||||
| 			return `skip: Read to foreign host (${id})`; |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		const messageId = id.split('/').pop(); |  | ||||||
|  |  | ||||||
| 		const message = await this.messagingMessagesRepository.findOneBy({ id: messageId }); |  | ||||||
| 		if (message == null) { |  | ||||||
| 			return 'skip: message not found'; |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		if (actor.id !== message.recipientId) { |  | ||||||
| 			return 'skip: actor is not a message recipient'; |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		await this.messagingService.readUserMessagingMessage(message.recipientId!, message.userId, [message.id]); |  | ||||||
| 		return `ok: mark as read (${message.userId} => ${message.recipientId} ${message.id})`; |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	@bindThis |  | ||||||
| 	private async accept(actor: CacheableRemoteUser, activity: IAccept): Promise<string> { |  | ||||||
| 		const uri = activity.id ?? activity; | 		const uri = activity.id ?? activity; | ||||||
|  |  | ||||||
| 		this.logger.info(`Accept: ${uri}`); | 		this.logger.info(`Accept: ${uri}`); | ||||||
| @@ -227,7 +197,7 @@ export class ApInboxService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	private async acceptFollow(actor: CacheableRemoteUser, activity: IFollow): Promise<string> { | 	private async acceptFollow(actor: RemoteUser, activity: IFollow): Promise<string> { | ||||||
| 		// ※ activityはこっちから投げたフォローリクエストなので、activity.actorは存在するローカルユーザーである必要がある | 		// ※ activityはこっちから投げたフォローリクエストなので、activity.actorは存在するローカルユーザーである必要がある | ||||||
|  |  | ||||||
| 		const follower = await this.apDbResolverService.getUserFromApId(activity.actor); | 		const follower = await this.apDbResolverService.getUserFromApId(activity.actor); | ||||||
| @@ -251,7 +221,7 @@ export class ApInboxService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	private async add(actor: CacheableRemoteUser, activity: IAdd): Promise<void> { | 	private async add(actor: RemoteUser, activity: IAdd): Promise<void> { | ||||||
| 		if ('actor' in activity && actor.uri !== activity.actor) { | 		if ('actor' in activity && actor.uri !== activity.actor) { | ||||||
| 			throw new Error('invalid actor'); | 			throw new Error('invalid actor'); | ||||||
| 		} | 		} | ||||||
| @@ -271,7 +241,7 @@ export class ApInboxService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	private async announce(actor: CacheableRemoteUser, activity: IAnnounce): Promise<void> { | 	private async announce(actor: RemoteUser, activity: IAnnounce): Promise<void> { | ||||||
| 		const uri = getApId(activity); | 		const uri = getApId(activity); | ||||||
|  |  | ||||||
| 		this.logger.info(`Announce: ${uri}`); | 		this.logger.info(`Announce: ${uri}`); | ||||||
| @@ -282,7 +252,7 @@ export class ApInboxService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	private async announceNote(actor: CacheableRemoteUser, activity: IAnnounce, targetUri: string): Promise<void> { | 	private async announceNote(actor: RemoteUser, activity: IAnnounce, targetUri: string): Promise<void> { | ||||||
| 		const uri = getApId(activity); | 		const uri = getApId(activity); | ||||||
|  |  | ||||||
| 		if (actor.isSuspended) { | 		if (actor.isSuspended) { | ||||||
| @@ -342,7 +312,7 @@ export class ApInboxService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	private async block(actor: CacheableRemoteUser, activity: IBlock): Promise<string> { | 	private async block(actor: RemoteUser, activity: IBlock): Promise<string> { | ||||||
| 		// ※ activity.objectにブロック対象があり、それは存在するローカルユーザーのはず | 		// ※ activity.objectにブロック対象があり、それは存在するローカルユーザーのはず | ||||||
|  |  | ||||||
| 		const blockee = await this.apDbResolverService.getUserFromApId(activity.object); | 		const blockee = await this.apDbResolverService.getUserFromApId(activity.object); | ||||||
| @@ -360,7 +330,7 @@ export class ApInboxService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	private async create(actor: CacheableRemoteUser, activity: ICreate): Promise<void> { | 	private async create(actor: RemoteUser, activity: ICreate): Promise<void> { | ||||||
| 		const uri = getApId(activity); | 		const uri = getApId(activity); | ||||||
|  |  | ||||||
| 		this.logger.info(`Create: ${uri}`); | 		this.logger.info(`Create: ${uri}`); | ||||||
| @@ -396,7 +366,7 @@ export class ApInboxService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	private async createNote(resolver: Resolver, actor: CacheableRemoteUser, note: IObject, silent = false, activity?: ICreate): Promise<string> { | 	private async createNote(resolver: Resolver, actor: RemoteUser, note: IObject, silent = false, activity?: ICreate): Promise<string> { | ||||||
| 		const uri = getApId(note); | 		const uri = getApId(note); | ||||||
|  |  | ||||||
| 		if (typeof note === 'object') { | 		if (typeof note === 'object') { | ||||||
| @@ -431,7 +401,7 @@ export class ApInboxService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	private async delete(actor: CacheableRemoteUser, activity: IDelete): Promise<string> { | 	private async delete(actor: RemoteUser, activity: IDelete): Promise<string> { | ||||||
| 		if ('actor' in activity && actor.uri !== activity.actor) { | 		if ('actor' in activity && actor.uri !== activity.actor) { | ||||||
| 			throw new Error('invalid actor'); | 			throw new Error('invalid actor'); | ||||||
| 		} | 		} | ||||||
| @@ -473,7 +443,7 @@ export class ApInboxService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	private async deleteActor(actor: CacheableRemoteUser, uri: string): Promise<string> { | 	private async deleteActor(actor: RemoteUser, uri: string): Promise<string> { | ||||||
| 		this.logger.info(`Deleting the Actor: ${uri}`); | 		this.logger.info(`Deleting the Actor: ${uri}`); | ||||||
| 	 | 	 | ||||||
| 		if (actor.uri !== uri) { | 		if (actor.uri !== uri) { | ||||||
| @@ -482,7 +452,7 @@ export class ApInboxService { | |||||||
| 	 | 	 | ||||||
| 		const user = await this.usersRepository.findOneByOrFail({ id: actor.id }); | 		const user = await this.usersRepository.findOneByOrFail({ id: actor.id }); | ||||||
| 		if (user.isDeleted) { | 		if (user.isDeleted) { | ||||||
| 			this.logger.info('skip: already deleted'); | 			return 'skip: already deleted'; | ||||||
| 		} | 		} | ||||||
| 	 | 	 | ||||||
| 		const job = await this.queueService.createDeleteAccountJob(actor); | 		const job = await this.queueService.createDeleteAccountJob(actor); | ||||||
| @@ -495,7 +465,7 @@ export class ApInboxService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	private async deleteNote(actor: CacheableRemoteUser, uri: string): Promise<string> { | 	private async deleteNote(actor: RemoteUser, uri: string): Promise<string> { | ||||||
| 		this.logger.info(`Deleting the Note: ${uri}`); | 		this.logger.info(`Deleting the Note: ${uri}`); | ||||||
| 	 | 	 | ||||||
| 		const unlock = await this.appLockService.getApLock(uri); | 		const unlock = await this.appLockService.getApLock(uri); | ||||||
| @@ -504,16 +474,7 @@ export class ApInboxService { | |||||||
| 			const note = await this.apDbResolverService.getNoteFromApId(uri); | 			const note = await this.apDbResolverService.getNoteFromApId(uri); | ||||||
| 	 | 	 | ||||||
| 			if (note == null) { | 			if (note == null) { | ||||||
| 				const message = await this.apDbResolverService.getMessageFromApId(uri); | 				return 'message not found'; | ||||||
| 				if (message == null) return 'message not found'; |  | ||||||
| 	 |  | ||||||
| 				if (message.userId !== actor.id) { |  | ||||||
| 					return '投稿を削除しようとしているユーザーは投稿の作成者ではありません'; |  | ||||||
| 				} |  | ||||||
| 	 |  | ||||||
| 				await this.messagingService.deleteMessage(message); |  | ||||||
| 	 |  | ||||||
| 				return 'ok: message deleted'; |  | ||||||
| 			} | 			} | ||||||
| 	 | 	 | ||||||
| 			if (note.userId !== actor.id) { | 			if (note.userId !== actor.id) { | ||||||
| @@ -528,7 +489,7 @@ export class ApInboxService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	private async flag(actor: CacheableRemoteUser, activity: IFlag): Promise<string> { | 	private async flag(actor: RemoteUser, activity: IFlag): Promise<string> { | ||||||
| 		// objectは `(User|Note) | (User|Note)[]` だけど、全パターンDBスキーマと対応させられないので | 		// objectは `(User|Note) | (User|Note)[]` だけど、全パターンDBスキーマと対応させられないので | ||||||
| 		// 対象ユーザーは一番最初のユーザー として あとはコメントとして格納する | 		// 対象ユーザーは一番最初のユーザー として あとはコメントとして格納する | ||||||
| 		const uris = getApIds(activity.object); | 		const uris = getApIds(activity.object); | ||||||
| @@ -553,7 +514,7 @@ export class ApInboxService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	private async reject(actor: CacheableRemoteUser, activity: IReject): Promise<string> { | 	private async reject(actor: RemoteUser, activity: IReject): Promise<string> { | ||||||
| 		const uri = activity.id ?? activity; | 		const uri = activity.id ?? activity; | ||||||
|  |  | ||||||
| 		this.logger.info(`Reject: ${uri}`); | 		this.logger.info(`Reject: ${uri}`); | ||||||
| @@ -571,7 +532,7 @@ export class ApInboxService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	private async rejectFollow(actor: CacheableRemoteUser, activity: IFollow): Promise<string> { | 	private async rejectFollow(actor: RemoteUser, activity: IFollow): Promise<string> { | ||||||
| 		// ※ activityはこっちから投げたフォローリクエストなので、activity.actorは存在するローカルユーザーである必要がある | 		// ※ activityはこっちから投げたフォローリクエストなので、activity.actorは存在するローカルユーザーである必要がある | ||||||
| 	 | 	 | ||||||
| 		const follower = await this.apDbResolverService.getUserFromApId(activity.actor); | 		const follower = await this.apDbResolverService.getUserFromApId(activity.actor); | ||||||
| @@ -595,7 +556,7 @@ export class ApInboxService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	private async remove(actor: CacheableRemoteUser, activity: IRemove): Promise<void> { | 	private async remove(actor: RemoteUser, activity: IRemove): Promise<void> { | ||||||
| 		if ('actor' in activity && actor.uri !== activity.actor) { | 		if ('actor' in activity && actor.uri !== activity.actor) { | ||||||
| 			throw new Error('invalid actor'); | 			throw new Error('invalid actor'); | ||||||
| 		} | 		} | ||||||
| @@ -615,7 +576,7 @@ export class ApInboxService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	private async undo(actor: CacheableRemoteUser, activity: IUndo): Promise<string> { | 	private async undo(actor: RemoteUser, activity: IUndo): Promise<string> { | ||||||
| 		if ('actor' in activity && actor.uri !== activity.actor) { | 		if ('actor' in activity && actor.uri !== activity.actor) { | ||||||
| 			throw new Error('invalid actor'); | 			throw new Error('invalid actor'); | ||||||
| 		} | 		} | ||||||
| @@ -641,7 +602,7 @@ export class ApInboxService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	private async undoAccept(actor: CacheableRemoteUser, activity: IAccept): Promise<string> { | 	private async undoAccept(actor: RemoteUser, activity: IAccept): Promise<string> { | ||||||
| 		const follower = await this.apDbResolverService.getUserFromApId(activity.object); | 		const follower = await this.apDbResolverService.getUserFromApId(activity.object); | ||||||
| 		if (follower == null) { | 		if (follower == null) { | ||||||
| 			return 'skip: follower not found'; | 			return 'skip: follower not found'; | ||||||
| @@ -661,7 +622,7 @@ export class ApInboxService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	private async undoAnnounce(actor: CacheableRemoteUser, activity: IAnnounce): Promise<string> { | 	private async undoAnnounce(actor: RemoteUser, activity: IAnnounce): Promise<string> { | ||||||
| 		const uri = getApId(activity); | 		const uri = getApId(activity); | ||||||
|  |  | ||||||
| 		const note = await this.notesRepository.findOneBy({ | 		const note = await this.notesRepository.findOneBy({ | ||||||
| @@ -676,7 +637,7 @@ export class ApInboxService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	private async undoBlock(actor: CacheableRemoteUser, activity: IBlock): Promise<string> { | 	private async undoBlock(actor: RemoteUser, activity: IBlock): Promise<string> { | ||||||
| 		const blockee = await this.apDbResolverService.getUserFromApId(activity.object); | 		const blockee = await this.apDbResolverService.getUserFromApId(activity.object); | ||||||
|  |  | ||||||
| 		if (blockee == null) { | 		if (blockee == null) { | ||||||
| @@ -692,7 +653,7 @@ export class ApInboxService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	private async undoFollow(actor: CacheableRemoteUser, activity: IFollow): Promise<string> { | 	private async undoFollow(actor: RemoteUser, activity: IFollow): Promise<string> { | ||||||
| 		const followee = await this.apDbResolverService.getUserFromApId(activity.object); | 		const followee = await this.apDbResolverService.getUserFromApId(activity.object); | ||||||
| 		if (followee == null) { | 		if (followee == null) { | ||||||
| 			return 'skip: followee not found'; | 			return 'skip: followee not found'; | ||||||
| @@ -726,7 +687,7 @@ export class ApInboxService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	private async undoLike(actor: CacheableRemoteUser, activity: ILike): Promise<string> { | 	private async undoLike(actor: RemoteUser, activity: ILike): Promise<string> { | ||||||
| 		const targetUri = getApId(activity.object); | 		const targetUri = getApId(activity.object); | ||||||
|  |  | ||||||
| 		const note = await this.apNoteService.fetchNote(targetUri); | 		const note = await this.apNoteService.fetchNote(targetUri); | ||||||
| @@ -741,7 +702,7 @@ export class ApInboxService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	private async update(actor: CacheableRemoteUser, activity: IUpdate): Promise<string> { | 	private async update(actor: RemoteUser, activity: IUpdate): Promise<string> { | ||||||
| 		if ('actor' in activity && actor.uri !== activity.actor) { | 		if ('actor' in activity && actor.uri !== activity.actor) { | ||||||
| 			return 'skip: invalid actor'; | 			return 'skip: invalid actor'; | ||||||
| 		} | 		} | ||||||
|   | |||||||
| @@ -1,7 +1,6 @@ | |||||||
| import { Inject, Injectable } from '@nestjs/common'; | import { Injectable } from '@nestjs/common'; | ||||||
| import type Logger from '@/logger.js'; | import type Logger from '@/logger.js'; | ||||||
| import { RemoteLoggerService } from '@/core/RemoteLoggerService.js'; | import { RemoteLoggerService } from '@/core/RemoteLoggerService.js'; | ||||||
| import { bindThis } from '@/decorators.js'; |  | ||||||
|  |  | ||||||
| @Injectable() | @Injectable() | ||||||
| export class ApLoggerService { | export class ApLoggerService { | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ import { v4 as uuid } from 'uuid'; | |||||||
| import * as mfm from 'mfm-js'; | import * as mfm from 'mfm-js'; | ||||||
| import { DI } from '@/di-symbols.js'; | import { DI } from '@/di-symbols.js'; | ||||||
| import type { Config } from '@/config.js'; | import type { Config } from '@/config.js'; | ||||||
| import type { ILocalUser, IRemoteUser, User } from '@/models/entities/User.js'; | import type { LocalUser, RemoteUser, User } from '@/models/entities/User.js'; | ||||||
| import type { IMentionedRemoteUsers, Note } from '@/models/entities/Note.js'; | import type { IMentionedRemoteUsers, Note } from '@/models/entities/Note.js'; | ||||||
| import type { Blocking } from '@/models/entities/Blocking.js'; | import type { Blocking } from '@/models/entities/Blocking.js'; | ||||||
| import type { Relay } from '@/models/entities/Relay.js'; | import type { Relay } from '@/models/entities/Relay.js'; | ||||||
| @@ -13,7 +13,6 @@ import type { DriveFile } from '@/models/entities/DriveFile.js'; | |||||||
| import type { NoteReaction } from '@/models/entities/NoteReaction.js'; | import type { NoteReaction } from '@/models/entities/NoteReaction.js'; | ||||||
| import type { Emoji } from '@/models/entities/Emoji.js'; | import type { Emoji } from '@/models/entities/Emoji.js'; | ||||||
| import type { Poll } from '@/models/entities/Poll.js'; | import type { Poll } from '@/models/entities/Poll.js'; | ||||||
| import type { MessagingMessage } from '@/models/entities/MessagingMessage.js'; |  | ||||||
| import type { PollVote } from '@/models/entities/PollVote.js'; | import type { PollVote } from '@/models/entities/PollVote.js'; | ||||||
| import { UserKeypairStoreService } from '@/core/UserKeypairStoreService.js'; | import { UserKeypairStoreService } from '@/core/UserKeypairStoreService.js'; | ||||||
| import { MfmService } from '@/core/MfmService.js'; | import { MfmService } from '@/core/MfmService.js'; | ||||||
| @@ -24,7 +23,7 @@ import type { UsersRepository, UserProfilesRepository, NotesRepository, DriveFil | |||||||
| import { bindThis } from '@/decorators.js'; | import { bindThis } from '@/decorators.js'; | ||||||
| import { LdSignatureService } from './LdSignatureService.js'; | import { LdSignatureService } from './LdSignatureService.js'; | ||||||
| import { ApMfmService } from './ApMfmService.js'; | import { ApMfmService } from './ApMfmService.js'; | ||||||
| import type { IActivity, IObject } from './type.js'; | import type { IAccept, IActivity, IAdd, IAnnounce, IApDocument, IApEmoji, IApHashtag, IApImage, IApMention, IBlock, ICreate, IDelete, IFlag, IFollow, IKey, ILike, IObject, IPost, IQuestion, IReject, IRemove, ITombstone, IUndo, IUpdate } from './type.js'; | ||||||
| import type { IIdentifier } from './models/identifier.js'; | import type { IIdentifier } from './models/identifier.js'; | ||||||
|  |  | ||||||
| @Injectable() | @Injectable() | ||||||
| @@ -61,7 +60,7 @@ export class ApRendererService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public renderAccept(object: any, user: { id: User['id']; host: null }) { | 	public renderAccept(object: any, user: { id: User['id']; host: null }): IAccept { | ||||||
| 		return { | 		return { | ||||||
| 			type: 'Accept', | 			type: 'Accept', | ||||||
| 			actor: `${this.config.url}/users/${user.id}`, | 			actor: `${this.config.url}/users/${user.id}`, | ||||||
| @@ -70,7 +69,7 @@ export class ApRendererService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public renderAdd(user: ILocalUser, target: any, object: any) { | 	public renderAdd(user: LocalUser, target: any, object: any): IAdd { | ||||||
| 		return { | 		return { | ||||||
| 			type: 'Add', | 			type: 'Add', | ||||||
| 			actor: `${this.config.url}/users/${user.id}`, | 			actor: `${this.config.url}/users/${user.id}`, | ||||||
| @@ -80,7 +79,7 @@ export class ApRendererService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public renderAnnounce(object: any, note: Note) { | 	public renderAnnounce(object: any, note: Note): IAnnounce { | ||||||
| 		const attributedTo = `${this.config.url}/users/${note.userId}`; | 		const attributedTo = `${this.config.url}/users/${note.userId}`; | ||||||
|  |  | ||||||
| 		let to: string[] = []; | 		let to: string[] = []; | ||||||
| @@ -93,7 +92,7 @@ export class ApRendererService { | |||||||
| 			to = [`${attributedTo}/followers`]; | 			to = [`${attributedTo}/followers`]; | ||||||
| 			cc = ['https://www.w3.org/ns/activitystreams#Public']; | 			cc = ['https://www.w3.org/ns/activitystreams#Public']; | ||||||
| 		} else { | 		} else { | ||||||
| 			return null; | 			throw new Error('renderAnnounce: cannot render non-public note'); | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		return { | 		return { | ||||||
| @@ -113,7 +112,7 @@ export class ApRendererService { | |||||||
| 	 * @param block The block to be rendered. The blockee relation must be loaded. | 	 * @param block The block to be rendered. The blockee relation must be loaded. | ||||||
| 	 */ | 	 */ | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public renderBlock(block: Blocking) { | 	public renderBlock(block: Blocking): IBlock { | ||||||
| 		if (block.blockee?.uri == null) { | 		if (block.blockee?.uri == null) { | ||||||
| 			throw new Error('renderBlock: missing blockee uri'); | 			throw new Error('renderBlock: missing blockee uri'); | ||||||
| 		} | 		} | ||||||
| @@ -127,14 +126,14 @@ export class ApRendererService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public renderCreate(object: any, note: Note) { | 	public renderCreate(object: IObject, note: Note): ICreate { | ||||||
| 		const activity = { | 		const activity = { | ||||||
| 			id: `${this.config.url}/notes/${note.id}/activity`, | 			id: `${this.config.url}/notes/${note.id}/activity`, | ||||||
| 			actor: `${this.config.url}/users/${note.userId}`, | 			actor: `${this.config.url}/users/${note.userId}`, | ||||||
| 			type: 'Create', | 			type: 'Create', | ||||||
| 			published: note.createdAt.toISOString(), | 			published: note.createdAt.toISOString(), | ||||||
| 			object, | 			object, | ||||||
| 		} as any; | 		} as ICreate; | ||||||
| 	 | 	 | ||||||
| 		if (object.to) activity.to = object.to; | 		if (object.to) activity.to = object.to; | ||||||
| 		if (object.cc) activity.cc = object.cc; | 		if (object.cc) activity.cc = object.cc; | ||||||
| @@ -143,7 +142,7 @@ export class ApRendererService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public renderDelete(object: any, user: { id: User['id']; host: null }) { | 	public renderDelete(object: IObject | string, user: { id: User['id']; host: null }): IDelete { | ||||||
| 		return { | 		return { | ||||||
| 			type: 'Delete', | 			type: 'Delete', | ||||||
| 			actor: `${this.config.url}/users/${user.id}`, | 			actor: `${this.config.url}/users/${user.id}`, | ||||||
| @@ -153,7 +152,7 @@ export class ApRendererService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public renderDocument(file: DriveFile) { | 	public renderDocument(file: DriveFile): IApDocument { | ||||||
| 		return { | 		return { | ||||||
| 			type: 'Document', | 			type: 'Document', | ||||||
| 			mediaType: file.type, | 			mediaType: file.type, | ||||||
| @@ -163,12 +162,12 @@ export class ApRendererService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public renderEmoji(emoji: Emoji) { | 	public renderEmoji(emoji: Emoji): IApEmoji { | ||||||
| 		return { | 		return { | ||||||
| 			id: `${this.config.url}/emojis/${emoji.name}`, | 			id: `${this.config.url}/emojis/${emoji.name}`, | ||||||
| 			type: 'Emoji', | 			type: 'Emoji', | ||||||
| 			name: `:${emoji.name}:`, | 			name: `:${emoji.name}:`, | ||||||
| 			updated: emoji.updatedAt != null ? emoji.updatedAt.toISOString() : new Date().toISOString, | 			updated: emoji.updatedAt != null ? emoji.updatedAt.toISOString() : new Date().toISOString(), | ||||||
| 			icon: { | 			icon: { | ||||||
| 				type: 'Image', | 				type: 'Image', | ||||||
| 				mediaType: emoji.type ?? 'image/png', | 				mediaType: emoji.type ?? 'image/png', | ||||||
| @@ -179,9 +178,8 @@ export class ApRendererService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	// to anonymise reporters, the reporting actor must be a system user | 	// to anonymise reporters, the reporting actor must be a system user | ||||||
| 	// object has to be a uri or array of uris |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public renderFlag(user: ILocalUser, object: [string], content: string) { | 	public renderFlag(user: LocalUser, object: IObject | string, content: string): IFlag { | ||||||
| 		return { | 		return { | ||||||
| 			type: 'Flag', | 			type: 'Flag', | ||||||
| 			actor: `${this.config.url}/users/${user.id}`, | 			actor: `${this.config.url}/users/${user.id}`, | ||||||
| @@ -191,15 +189,13 @@ export class ApRendererService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public renderFollowRelay(relay: Relay, relayActor: ILocalUser) { | 	public renderFollowRelay(relay: Relay, relayActor: LocalUser): IFollow { | ||||||
| 		const follow = { | 		return { | ||||||
| 			id: `${this.config.url}/activities/follow-relay/${relay.id}`, | 			id: `${this.config.url}/activities/follow-relay/${relay.id}`, | ||||||
| 			type: 'Follow', | 			type: 'Follow', | ||||||
| 			actor: `${this.config.url}/users/${relayActor.id}`, | 			actor: `${this.config.url}/users/${relayActor.id}`, | ||||||
| 			object: 'https://www.w3.org/ns/activitystreams#Public', | 			object: 'https://www.w3.org/ns/activitystreams#Public', | ||||||
| 		}; | 		}; | ||||||
| 	 |  | ||||||
| 		return follow; |  | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	/** | 	/** | ||||||
| @@ -217,19 +213,17 @@ export class ApRendererService { | |||||||
| 		follower: { id: User['id']; host: User['host']; uri: User['host'] }, | 		follower: { id: User['id']; host: User['host']; uri: User['host'] }, | ||||||
| 		followee: { id: User['id']; host: User['host']; uri: User['host'] }, | 		followee: { id: User['id']; host: User['host']; uri: User['host'] }, | ||||||
| 		requestId?: string, | 		requestId?: string, | ||||||
| 	) { | 	): IFollow { | ||||||
| 		const follow = { | 		return { | ||||||
| 			id: requestId ?? `${this.config.url}/follows/${follower.id}/${followee.id}`, | 			id: requestId ?? `${this.config.url}/follows/${follower.id}/${followee.id}`, | ||||||
| 			type: 'Follow', | 			type: 'Follow', | ||||||
| 			actor: this.userEntityService.isLocalUser(follower) ? `${this.config.url}/users/${follower.id}` : follower.uri, | 			actor: this.userEntityService.isLocalUser(follower) ? `${this.config.url}/users/${follower.id}` : follower.uri!, | ||||||
| 			object: this.userEntityService.isLocalUser(followee) ? `${this.config.url}/users/${followee.id}` : followee.uri, | 			object: this.userEntityService.isLocalUser(followee) ? `${this.config.url}/users/${followee.id}` : followee.uri!, | ||||||
| 		} as any; | 		}; | ||||||
| 	 |  | ||||||
| 		return follow; |  | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public renderHashtag(tag: string) { | 	public renderHashtag(tag: string): IApHashtag { | ||||||
| 		return { | 		return { | ||||||
| 			type: 'Hashtag', | 			type: 'Hashtag', | ||||||
| 			href: `${this.config.url}/tags/${encodeURIComponent(tag)}`, | 			href: `${this.config.url}/tags/${encodeURIComponent(tag)}`, | ||||||
| @@ -238,7 +232,7 @@ export class ApRendererService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public renderImage(file: DriveFile) { | 	public renderImage(file: DriveFile): IApImage { | ||||||
| 		return { | 		return { | ||||||
| 			type: 'Image', | 			type: 'Image', | ||||||
| 			url: this.driveFileEntityService.getPublicUrl(file), | 			url: this.driveFileEntityService.getPublicUrl(file), | ||||||
| @@ -248,7 +242,7 @@ export class ApRendererService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public renderKey(user: ILocalUser, key: UserKeypair, postfix?: string) { | 	public renderKey(user: LocalUser, key: UserKeypair, postfix?: string): IKey { | ||||||
| 		return { | 		return { | ||||||
| 			id: `${this.config.url}/users/${user.id}${postfix ?? '/publickey'}`, | 			id: `${this.config.url}/users/${user.id}${postfix ?? '/publickey'}`, | ||||||
| 			type: 'Key', | 			type: 'Key', | ||||||
| @@ -261,7 +255,7 @@ export class ApRendererService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public async renderLike(noteReaction: NoteReaction, note: { uri: string | null }) { | 	public async renderLike(noteReaction: NoteReaction, note: { uri: string | null }): Promise<ILike> { | ||||||
| 		const reaction = noteReaction.reaction; | 		const reaction = noteReaction.reaction; | ||||||
|  |  | ||||||
| 		const object = { | 		const object = { | ||||||
| @@ -271,10 +265,11 @@ export class ApRendererService { | |||||||
| 			object: note.uri ? note.uri : `${this.config.url}/notes/${noteReaction.noteId}`, | 			object: note.uri ? note.uri : `${this.config.url}/notes/${noteReaction.noteId}`, | ||||||
| 			content: reaction, | 			content: reaction, | ||||||
| 			_misskey_reaction: reaction, | 			_misskey_reaction: reaction, | ||||||
| 		} as any; | 		} as ILike; | ||||||
|  |  | ||||||
| 		if (reaction.startsWith(':')) { | 		if (reaction.startsWith(':')) { | ||||||
| 			const name = reaction.replaceAll(':', ''); | 			const name = reaction.replaceAll(':', ''); | ||||||
|  | 			// TODO: cache | ||||||
| 			const emoji = await this.emojisRepository.findOneBy({ | 			const emoji = await this.emojisRepository.findOneBy({ | ||||||
| 				name, | 				name, | ||||||
| 				host: IsNull(), | 				host: IsNull(), | ||||||
| @@ -287,16 +282,16 @@ export class ApRendererService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public renderMention(mention: User) { | 	public renderMention(mention: User): IApMention { | ||||||
| 		return { | 		return { | ||||||
| 			type: 'Mention', | 			type: 'Mention', | ||||||
| 			href: this.userEntityService.isRemoteUser(mention) ? mention.uri : `${this.config.url}/users/${(mention as ILocalUser).id}`, | 			href: this.userEntityService.isRemoteUser(mention) ? mention.uri! : `${this.config.url}/users/${(mention as LocalUser).id}`, | ||||||
| 			name: this.userEntityService.isRemoteUser(mention) ? `@${mention.username}@${mention.host}` : `@${(mention as ILocalUser).username}`, | 			name: this.userEntityService.isRemoteUser(mention) ? `@${mention.username}@${mention.host}` : `@${(mention as LocalUser).username}`, | ||||||
| 		}; | 		}; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public async renderNote(note: Note, dive = true, isTalk = false): Promise<IObject> { | 	public async renderNote(note: Note, dive = true): Promise<IPost> { | ||||||
| 		const getPromisedFiles = async (ids: string[]) => { | 		const getPromisedFiles = async (ids: string[]) => { | ||||||
| 			if (!ids || ids.length === 0) return []; | 			if (!ids || ids.length === 0) return []; | ||||||
| 			const items = await this.driveFilesRepository.findBy({ id: In(ids) }); | 			const items = await this.driveFilesRepository.findBy({ id: In(ids) }); | ||||||
| @@ -409,12 +404,8 @@ export class ApRendererService { | |||||||
| 					totalItems: poll!.votes[i], | 					totalItems: poll!.votes[i], | ||||||
| 				}, | 				}, | ||||||
| 			})), | 			})), | ||||||
| 		} : {}; | 		} as const : {}; | ||||||
| 	 |  | ||||||
| 		const asTalk = isTalk ? { |  | ||||||
| 			_misskey_talk: true, |  | ||||||
| 		} : {}; |  | ||||||
| 	 |  | ||||||
| 		return { | 		return { | ||||||
| 			id: `${this.config.url}/notes/${note.id}`, | 			id: `${this.config.url}/notes/${note.id}`, | ||||||
| 			type: 'Note', | 			type: 'Note', | ||||||
| @@ -436,12 +427,11 @@ export class ApRendererService { | |||||||
| 			sensitive: note.cw != null || files.some(file => file.isSensitive), | 			sensitive: note.cw != null || files.some(file => file.isSensitive), | ||||||
| 			tag, | 			tag, | ||||||
| 			...asPoll, | 			...asPoll, | ||||||
| 			...asTalk, |  | ||||||
| 		}; | 		}; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public async renderPerson(user: ILocalUser) { | 	public async renderPerson(user: LocalUser) { | ||||||
| 		const id = `${this.config.url}/users/${user.id}`; | 		const id = `${this.config.url}/users/${user.id}`; | ||||||
| 		const isSystem = !!user.username.match(/\./); | 		const isSystem = !!user.username.match(/\./); | ||||||
|  |  | ||||||
| @@ -518,8 +508,8 @@ export class ApRendererService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public async renderQuestion(user: { id: User['id'] }, note: Note, poll: Poll) { | 	public renderQuestion(user: { id: User['id'] }, note: Note, poll: Poll): IQuestion { | ||||||
| 		const question = { | 		return { | ||||||
| 			type: 'Question', | 			type: 'Question', | ||||||
| 			id: `${this.config.url}/questions/${note.id}`, | 			id: `${this.config.url}/questions/${note.id}`, | ||||||
| 			actor: `${this.config.url}/users/${user.id}`, | 			actor: `${this.config.url}/users/${user.id}`, | ||||||
| @@ -533,21 +523,10 @@ export class ApRendererService { | |||||||
| 				}, | 				}, | ||||||
| 			})), | 			})), | ||||||
| 		}; | 		}; | ||||||
| 	 |  | ||||||
| 		return question; |  | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public renderRead(user: { id: User['id'] }, message: MessagingMessage) { | 	public renderReject(object: any, user: { id: User['id'] }): IReject { | ||||||
| 		return { |  | ||||||
| 			type: 'Read', |  | ||||||
| 			actor: `${this.config.url}/users/${user.id}`, |  | ||||||
| 			object: message.uri, |  | ||||||
| 		}; |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	@bindThis |  | ||||||
| 	public renderReject(object: any, user: { id: User['id'] }) { |  | ||||||
| 		return { | 		return { | ||||||
| 			type: 'Reject', | 			type: 'Reject', | ||||||
| 			actor: `${this.config.url}/users/${user.id}`, | 			actor: `${this.config.url}/users/${user.id}`, | ||||||
| @@ -556,7 +535,7 @@ export class ApRendererService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public renderRemove(user: { id: User['id'] }, target: any, object: any) { | 	public renderRemove(user: { id: User['id'] }, target: any, object: any): IRemove { | ||||||
| 		return { | 		return { | ||||||
| 			type: 'Remove', | 			type: 'Remove', | ||||||
| 			actor: `${this.config.url}/users/${user.id}`, | 			actor: `${this.config.url}/users/${user.id}`, | ||||||
| @@ -566,7 +545,7 @@ export class ApRendererService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public renderTombstone(id: string) { | 	public renderTombstone(id: string): ITombstone { | ||||||
| 		return { | 		return { | ||||||
| 			id, | 			id, | ||||||
| 			type: 'Tombstone', | 			type: 'Tombstone', | ||||||
| @@ -574,8 +553,7 @@ export class ApRendererService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public renderUndo(object: any, user: { id: User['id'] }) { | 	public renderUndo(object: any, user: { id: User['id'] }): IUndo { | ||||||
| 		if (object == null) return null; |  | ||||||
| 		const id = typeof object.id === 'string' && object.id.startsWith(this.config.url) ? `${object.id}/undo` : undefined; | 		const id = typeof object.id === 'string' && object.id.startsWith(this.config.url) ? `${object.id}/undo` : undefined; | ||||||
|  |  | ||||||
| 		return { | 		return { | ||||||
| @@ -588,21 +566,19 @@ export class ApRendererService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public renderUpdate(object: any, user: { id: User['id'] }) { | 	public renderUpdate(object: any, user: { id: User['id'] }): IUpdate { | ||||||
| 		const activity = { | 		return { | ||||||
| 			id: `${this.config.url}/users/${user.id}#updates/${new Date().getTime()}`, | 			id: `${this.config.url}/users/${user.id}#updates/${new Date().getTime()}`, | ||||||
| 			actor: `${this.config.url}/users/${user.id}`, | 			actor: `${this.config.url}/users/${user.id}`, | ||||||
| 			type: 'Update', | 			type: 'Update', | ||||||
| 			to: ['https://www.w3.org/ns/activitystreams#Public'], | 			to: ['https://www.w3.org/ns/activitystreams#Public'], | ||||||
| 			object, | 			object, | ||||||
| 			published: new Date().toISOString(), | 			published: new Date().toISOString(), | ||||||
| 		} as any; | 		}; | ||||||
| 	 |  | ||||||
| 		return activity; |  | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public renderVote(user: { id: User['id'] }, vote: PollVote, note: Note, poll: Poll, pollOwner: IRemoteUser) { | 	public renderVote(user: { id: User['id'] }, vote: PollVote, note: Note, poll: Poll, pollOwner: RemoteUser): ICreate { | ||||||
| 		return { | 		return { | ||||||
| 			id: `${this.config.url}/users/${user.id}#votes/${vote.id}/activity`, | 			id: `${this.config.url}/users/${user.id}#votes/${vote.id}/activity`, | ||||||
| 			actor: `${this.config.url}/users/${user.id}`, | 			actor: `${this.config.url}/users/${user.id}`, | ||||||
| @@ -621,9 +597,7 @@ export class ApRendererService { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	@bindThis | 	@bindThis | ||||||
| 	public renderActivity(x: any): IActivity | null { | 	public addContext<T extends IObject>(x: T): T & { '@context': any; id: string; } { | ||||||
| 		if (x == null) return null; |  | ||||||
| 	 |  | ||||||
| 		if (typeof x === 'object' && x.id == null) { | 		if (typeof x === 'object' && x.id == null) { | ||||||
| 			x.id = `${this.config.url}/${uuid()}`; | 			x.id = `${this.config.url}/${uuid()}`; | ||||||
| 		} | 		} | ||||||
| @@ -653,13 +627,12 @@ export class ApRendererService { | |||||||
| 					'_misskey_quote': 'misskey:_misskey_quote', | 					'_misskey_quote': 'misskey:_misskey_quote', | ||||||
| 					'_misskey_reaction': 'misskey:_misskey_reaction', | 					'_misskey_reaction': 'misskey:_misskey_reaction', | ||||||
| 					'_misskey_votes': 'misskey:_misskey_votes', | 					'_misskey_votes': 'misskey:_misskey_votes', | ||||||
| 					'_misskey_talk': 'misskey:_misskey_talk', |  | ||||||
| 					'isCat': 'misskey:isCat', | 					'isCat': 'misskey:isCat', | ||||||
| 					// vcard | 					// vcard | ||||||
| 					vcard: 'http://www.w3.org/2006/vcard/ns#', | 					vcard: 'http://www.w3.org/2006/vcard/ns#', | ||||||
| 				}, | 				}, | ||||||
| 			], | 			], | ||||||
| 		}, x); | 		}, x as T & { id: string; }); | ||||||
| 	} | 	} | ||||||
| 	 | 	 | ||||||
| 	@bindThis | 	@bindThis | ||||||
|   | |||||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user