Compare commits
	
		
			270 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | bcbe83cb38 | ||
|   | 37f983aee3 | ||
|   | 77de3f2b9d | ||
|   | f655b54937 | ||
|   | cac99ebdd4 | ||
|   | f0d0a1546a | ||
|   | 8e8459fa55 | ||
|   | d53c55ecb5 | ||
|   | ea33d61a90 | ||
|   | 2fcc3388dd | ||
|   | ef7f033c32 | ||
|   | 7aa54dc92e | ||
|   | d10ad1b413 | ||
|   | 34063a0b84 | ||
|   | 8c9d975d69 | ||
|   | 03b072b894 | ||
|   | d1bed49808 | ||
|   | 6c3417d9b5 | ||
|   | ba65226460 | ||
|   | 9c9cd168ee | ||
|   | abb3d2a8d9 | ||
|   | 637fe8a04b | ||
|   | be321e95e5 | ||
|   | ed46c1486c | ||
|   | c9fcfc6862 | ||
|   | 8495e37566 | ||
|   | 247bd43ae2 | ||
|   | a6685b1559 | ||
|   | 66c4e8064b | ||
|   | 9d1fa3f202 | ||
|   | a6985d7dc7 | ||
|   | 027c021ac9 | ||
|   | 604205ec09 | ||
|   | 77db016866 | ||
|   | c6a009dbae | ||
|   | 4299e3f90c | ||
|   | 19f4812c03 | ||
|   | d01c465a8d | ||
|   | 4f1409601e | ||
|   | 52cffe0864 | ||
|   | 0866d5c055 | ||
|   | 78c08f6503 | ||
|   | 27d0ac3d75 | ||
|   | a8776002f3 | ||
|   | 31aa008566 | ||
|   | 9d405b4581 | ||
|   | 80c490a18b | ||
|   | 30c9c3739f | ||
|   | ee0e7a09e0 | ||
|   | bfd9577f0d | ||
|   | 0cada4ca76 | ||
|   | a718ccc0b6 | ||
|   | 1fcfd8e645 | ||
|   | c6dd932a0b | ||
|   | b79eed01e0 | ||
|   | 3a7dbe9764 | ||
|   | bef2534fa8 | ||
|   | 888dcd2559 | ||
|   | 2b69fca6bd | ||
|   | 7d088d42b4 | ||
|   | f8ad303b13 | ||
|   | 3c59c6fc9b | ||
|   | 7353d729d7 | ||
|   | 62591e0e7a | ||
|   | 012f15d84b | ||
|   | e57c6f94d2 | ||
|   | 40b27e8ad8 | ||
|   | 055e9f21b7 | ||
|   | d7085b17fe | ||
|   | 0d4d7c9c0c | ||
|   | 99209d36e1 | ||
|   | e2a9a0ff3d | ||
|   | ab166959a4 | ||
|   | ab692cfa3d | ||
|   | c3ae6f3a4a | ||
|   | 5ef4a52bbd | ||
|   | 582768a5e4 | ||
|   | 1852d1cc6f | ||
|   | 7a5a541a4e | ||
|   | 72b03e009c | ||
|   | 1b113c1045 | ||
|   | 54959557ea | ||
|   | d44cb7f256 | ||
|   | 3d063c95d1 | ||
|   | 09cab605fc | ||
|   | 666c8c0498 | ||
|   | d3e764d7f9 | ||
|   | 7060625adf | ||
|   | 21b6e23e98 | ||
|   | a0f794e372 | ||
|   | 9195504329 | ||
|   | 8c5d9dd549 | ||
|   | 580f6a5b6c | ||
|   | 74e76b460b | ||
|   | c4570b37b7 | ||
|   | cd0b0012d9 | ||
|   | c055b4d32d | ||
|   | 75a9ff832a | ||
|   | b64d3af1f3 | ||
|   | fb6605bb40 | ||
|   | 3bfae80fa7 | ||
|   | cb16cb0610 | ||
|   | 0baed1a275 | ||
|   | 42162c8015 | ||
|   | 0fab0c416d | ||
|   | e2e262c8ce | ||
|   | cf6596203b | ||
|   | 471911a54f | ||
|   | 9394f4f540 | ||
|   | 4e968216ad | ||
|   | 84a7a9555f | ||
|   | 8d12fd152b | ||
|   | 629b765abc | ||
|   | 63a89fa84a | ||
|   | a3f89236a0 | ||
|   | 01560abafb | ||
|   | b5698026ba | ||
|   | 6258ce75b7 | ||
|   | 6f34c74027 | ||
|   | 8add4f359b | ||
|   | d8933c135f | ||
|   | eb350e8d6c | ||
|   | 615fedd64d | ||
|   | 25bd82ecaa | ||
|   | e0938e5e3a | ||
|   | ec5e6c8443 | ||
|   | 25d8077474 | ||
|   | 06083f40d9 | ||
|   | ec203f7f79 | ||
|   | 1b30d7d47a | ||
|   | d9be9c958f | ||
|   | ed09796e0d | ||
|   | 4bfa29c0ab | ||
|   | 4804bbb211 | ||
|   | 749102f9c2 | ||
|   | 0bcb1434b0 | ||
|   | 2e537e618c | ||
|   | fe3b7a2ad3 | ||
|   | 90db793fd0 | ||
|   | 7bd2a6ad61 | ||
|   | 745f4d2439 | ||
|   | 254cfaea28 | ||
|   | d4da5a1eea | ||
|   | c0f8297414 | ||
|   | 834cb2ea1a | ||
|   | d82769abd4 | ||
|   | adf01ed4a4 | ||
|   | 09c007b3aa | ||
|   | 526ff177aa | ||
|   | 0e40d4e796 | ||
|   | 172ebab7bd | ||
|   | aa4493fe5c | ||
|   | a68a88f79e | ||
|   | 1de7dc94e1 | ||
|   | 59cb7992e2 | ||
|   | 87b15df47b | ||
|   | 6932d86240 | ||
|   | 87f61e714a | ||
|   | 5762e2d9ba | ||
|   | f0691c8a4f | ||
|   | 6d7e4fe2a1 | ||
|   | 7dc789f470 | ||
|   | 190d1bbf3c | ||
|   | a755dd5f9e | ||
|   | 0846a7b94e | ||
|   | d8be0511f1 | ||
|   | fb07116a4c | ||
|   | fe453c15e3 | ||
|   | 059aeef6a0 | ||
|   | 30e25451d6 | ||
|   | 7f0fd55c9a | ||
|   | 0e9b496deb | ||
|   | 8294c18e70 | ||
|   | 39575b4696 | ||
|   | 29e9801d5c | ||
|   | eaf83bffb0 | ||
|   | bb25ece745 | ||
|   | 754b5629e4 | ||
|   | ee63403548 | ||
|   | 87edeb41da | ||
|   | 41fe804587 | ||
|   | 02466acc4b | ||
|   | 8470a64e6b | ||
|   | 9939e0f9a9 | ||
|   | ce5f552d0c | ||
|   | 7d4c535233 | ||
|   | 57cd0fb93f | ||
|   | a15299ae53 | ||
|   | 1df7abfbb9 | ||
|   | 85a0f696bc | ||
|   | ba3c62bf9c | ||
|   | c17e97b6a6 | ||
|   | 2d80cd0e7b | ||
|   | eedc572f0c | ||
|   | 2de1df3514 | ||
|   | 2d96af1255 | ||
|   | 163325ef89 | ||
|   | 23979bf09a | ||
|   | 7199e6f4e0 | ||
|   | a40f38b2b5 | ||
|   | 00a17ed5d4 | ||
|   | b594366f06 | ||
|   | e9284930df | ||
|   | ea7504f564 | ||
|   | c1f6d996f6 | ||
|   | df71dbb024 | ||
|   | f104e9b6cc | ||
|   | e29b5c2326 | ||
|   | 925868dcdb | ||
|   | d7df26d92b | ||
|   | 42d1c67d56 | ||
|   | c2d7929391 | ||
|   | 5864b52a81 | ||
|   | 493d32b3dc | ||
|   | ed141338fb | ||
|   | 95db488c48 | ||
|   | d5e1e523b6 | ||
|   | cd0f8a4ef9 | ||
|   | 6dac505af9 | ||
|   | 9d398040cb | ||
|   | aa55acedc9 | ||
|   | eb70d6f226 | ||
|   | 36f0963d78 | ||
|   | c3c111529e | ||
|   | 2dbab66cfe | ||
|   | 01238d6b1a | ||
|   | c34f302b1c | ||
|   | 6870262f8d | ||
|   | c54d5e7040 | ||
|   | 0ace009a54 | ||
|   | 48e8ee440b | ||
|   | 9855405b89 | ||
|   | 122076e8ea | ||
|   | 7c5ac2cbb4 | ||
|   | ccda2181c1 | ||
|   | b5fe4ba9be | ||
|   | fd9c7d525a | ||
|   | 080574e13d | ||
|   | ee0a445590 | ||
|   | bb342c7601 | ||
|   | ed17636fb9 | ||
|   | c59d7d941a | ||
|   | 377377595a | ||
|   | d63aef9963 | ||
|   | e9b28fa3c0 | ||
|   | be255dc583 | ||
|   | 18eb7c6087 | ||
|   | cf29e69813 | ||
|   | 132da7e3c0 | ||
|   | 26df23bb64 | ||
|   | 76389ad619 | ||
|   | 7cde8cfbf2 | ||
|   | 4eb2ddac4e | ||
|   | dc51eef27c | ||
|   | bff8a23cbc | ||
|   | 9c5efb9da0 | ||
|   | 48b8320e5e | ||
|   | 9b2ed96c1c | ||
|   | 69d9aa71f2 | ||
|   | 13683780cd | ||
|   | d780e5b251 | ||
|   | 917d3d0bd3 | ||
|   | 4b19c53697 | ||
|   | 2d40a15d2b | ||
|   | 2bdcd22ad4 | ||
|   | f73a4e1304 | ||
|   | b265cdbd84 | ||
|   | a04d8b95c2 | ||
|   | 0e9a8c0cd4 | ||
|   | 5ae8a3c7e8 | 
| @@ -154,3 +154,6 @@ id: 'aid' | ||||
|  | ||||
| # Media Proxy | ||||
| #mediaProxy: https://example.com/proxy | ||||
|  | ||||
| # Sign to ActivityPub GET request (default: false) | ||||
| #signToActivityPubGet: true | ||||
|   | ||||
| @@ -5,6 +5,7 @@ | ||||
| .vscode | ||||
| Dockerfile | ||||
| build/ | ||||
| built/ | ||||
| db/ | ||||
| docker-compose.yml | ||||
| elasticsearch/ | ||||
|   | ||||
							
								
								
									
										2
									
								
								.github/workflows/nodejs.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/nodejs.yml
									
									
									
									
										vendored
									
									
								
							| @@ -12,7 +12,7 @@ jobs: | ||||
|  | ||||
|     strategy: | ||||
|       matrix: | ||||
|         node-version: [12.x, 14.x] | ||||
|         node-version: [12.x, 14.x, 15.x] | ||||
|  | ||||
|     services: | ||||
|       postgres: | ||||
|   | ||||
| @@ -1 +1 @@ | ||||
| v14.4.0 | ||||
| v14.15.0 | ||||
|   | ||||
							
								
								
									
										2556
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						
									
										2556
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -1,9 +1,7 @@ | ||||
| FROM node:14.4.0-alpine AS base | ||||
| FROM node:14.15.0-alpine AS base | ||||
|  | ||||
| ENV NODE_ENV=production | ||||
|  | ||||
| RUN npm i -g npm@latest | ||||
|  | ||||
| WORKDIR /misskey | ||||
|  | ||||
| FROM base AS builder | ||||
| @@ -12,6 +10,7 @@ RUN apk add --no-cache \ | ||||
|     autoconf \ | ||||
|     automake \ | ||||
|     file \ | ||||
| 		git \ | ||||
|     g++ \ | ||||
|     gcc \ | ||||
|     libc-dev \ | ||||
|   | ||||
							
								
								
									
										17
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										17
									
								
								README.md
									
									
									
									
									
								
							| @@ -6,8 +6,9 @@ | ||||
| [](https://circleci.com/gh/syuilo/misskey) | ||||
| [](https://david-dm.org/syuilo/misskey) | ||||
| [](http://makeapullrequest.com) | ||||
| [](https://github.com/humanetech-community/awesome-humane-tech) | ||||
|  | ||||
| **A forever evolving, sophisticated microblogging platform.** | ||||
| **A forever evolving, professional microblogging platform.** | ||||
|  | ||||
| <p align="justify"> | ||||
| <a href="https://join.misskey.page/">Misskey</a> is a decentralized microblogging platform born on Earth. | ||||
| @@ -130,7 +131,6 @@ Please see the [Contribution Guide](./CONTRIBUTING.md). | ||||
| <td><img src="https://c8.patreon.com/2/200/557245" alt="mkatze " width="100"></td> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/23915207/25428766ecd745478e600b3d7f871eb2/1.png?token-time=2145916800&token-hash=urCLLA4KjJZX92Y1CxcBP4d8bVTHGkiaPnQZp-Tqz68%3D" alt="kabo2468y " width="100"></td> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/8249688/4aacf36b6b244ab1bc6653591b6640df/2.png?token-time=2145916800&token-hash=1ZEf2w6L34253cZXS_HlVevLEENWS9QqrnxGUAYblPo%3D" alt="AureoleArk " width="100"></td> | ||||
| <td><img src="https://c8.patreon.com/2/200/21285325" alt="Nie(sha) " width="100"></td> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/5670915/ee175f0bfb6347ffa4ea101a8c097bff/1.jpg?token-time=2145916800&token-hash=mPLM9CA-riFHx-myr3bLZJuH2xBRHA9se5VbHhLIOuA%3D" alt="osapon " width="100"></td> | ||||
| <td><img src="https://c8.patreon.com/2/200/16869916" alt="見当かなみ " width="100"></td> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/36813045/29876ea679d443bcbba3c3f16edab8c2/2.jpeg?token-time=2145916800&token-hash=YCKWnIhrV9rjUCV9KqtJnEqjy_uGYF3WMXftjUdpi7o%3D" alt="Wataru Manji (manji0)" width="100"></td> | ||||
| @@ -141,7 +141,6 @@ Please see the [Contribution Guide](./CONTRIBUTING.md). | ||||
| <td><a href="https://www.patreon.com/user?u=557245">mkatze </a></td> | ||||
| <td><a href="https://www.patreon.com/user?u=23915207">kabo2468y </a></td> | ||||
| <td><a href="https://www.patreon.com/AureoleArk">AureoleArk </a></td> | ||||
| <td><a href="https://www.patreon.com/user?u=21285325">Nie(sha) </a></td> | ||||
| <td><a href="https://www.patreon.com/osapon">osapon </a></td> | ||||
| <td><a href="https://www.patreon.com/user?u=16869916">見当かなみ </a></td> | ||||
| <td><a href="https://www.patreon.com/user?u=36813045">Wataru Manji (manji0)</a></td> | ||||
| @@ -153,8 +152,8 @@ Please see the [Contribution Guide](./CONTRIBUTING.md). | ||||
| <td><img src="https://c8.patreon.com/2/200/16542964" alt="Takumi Sugita" width="100"></td> | ||||
| <td><img src="https://c8.patreon.com/2/200/17866454" alt="sikyosyounin " width="100"></td> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/5881381/6235ca5d3fb04c8e95ef5b4ff2abcc18/3.png?token-time=2145916800&token-hash=KjfQL8nf3AIf6WqzLshBYAyX44piAqOAZiYXgZS_H6A%3D" alt="YUKIMOCHI" width="100"></td> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/38837364/9421361c54c645ac8f5fc442a40c32e9/1.png?token-time=2145916800&token-hash=TUZB48Nem3BeUPLBH6s3P6WyKBnQOy0xKaDSTBBUNzA%3D" alt="xianon" width="100"></td> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/26340354/08834cf767b3449e93098ef73a434e2f/2.png?token-time=2145916800&token-hash=nyM8DnKRL8hR47HQ619mUzsqVRpkWZjgtgBU9RY15Uc%3D" alt="totokoro " width="100"></td> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/19356899/496b4681d33b4520bd7688e0fd19c04d/2.jpeg?token-time=2145916800&token-hash=_sTj3dUBOhn9qwiJ7F19Qd-yWWfUqJC_0jG1h0agEqQ%3D" alt="sheeta.s " width="100"></td> | ||||
| </tr><tr> | ||||
| <td><a href="https://www.patreon.com/Yuzulia">YuzuRyo61 </a></td> | ||||
| <td><a href="https://www.patreon.com/hs_sh_net">mewl hayabusa</a></td> | ||||
| @@ -162,10 +161,11 @@ Please see the [Contribution Guide](./CONTRIBUTING.md). | ||||
| <td><a href="https://www.patreon.com/user?u=16542964">Takumi Sugita</a></td> | ||||
| <td><a href="https://www.patreon.com/user?u=17866454">sikyosyounin </a></td> | ||||
| <td><a href="https://www.patreon.com/yukimochi">YUKIMOCHI</a></td> | ||||
| <td><a href="https://www.patreon.com/user?u=38837364">xianon</a></td> | ||||
| <td><a href="https://www.patreon.com/user?u=26340354">totokoro </a></td> | ||||
| <td><a href="https://www.patreon.com/user?u=19356899">sheeta.s </a></td> | ||||
| </tr></table> | ||||
| <table><tr> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/19356899/496b4681d33b4520bd7688e0fd19c04d/2.jpeg?token-time=2145916800&token-hash=_sTj3dUBOhn9qwiJ7F19Qd-yWWfUqJC_0jG1h0agEqQ%3D" alt="sheeta.s " width="100"></td> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/5827393/59893c191dda408f9cabd0f20a3a5627/1.jpeg?token-time=2145916800&token-hash=i9N05vOph-eP1LTLb9_npATjYOpntL0ZsHNaZFSsPmE%3D" alt="motcha " width="100"></td> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/20494440/540beaf2445f408ea6597bc61e077bb3/1.png?token-time=2145916800&token-hash=UJ0JQge64Bx9XmN_qYA1inMQhrWf4U91fqz7VAKJeSg%3D" alt="axtuki1 " width="100"></td> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/13737140/1adf7835017d479280d90fe8d30aade2/1.png?token-time=2145916800&token-hash=0pdle8h5pDZrww0BDOjdz6zO-HudeGTh36a3qi1biVU%3D" alt="Satsuki Yanagi" width="100"></td> | ||||
| @@ -176,8 +176,8 @@ Please see the [Contribution Guide](./CONTRIBUTING.md). | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/4389829/9f709180ac714651a70f74a82f3ffdb9/3.png?token-time=2145916800&token-hash=FTm3WVom4dJ9NwWMU4OpCL_8Yc13WiwEbKrDPyTZTPs%3D" alt="natalie" width="100"></td> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/26144593/9514b10a5c1b42a3af58621aee213d1d/1.png?token-time=2145916800&token-hash=v1PYRsjzu4c_mndN4Hvi_dlispZJsuGRCQeNS82pUSM%3D" alt="EBISUME" width="100"></td> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/5923936/2a743cbfbff946c2af3f09026047c0da/2.png?token-time=2145916800&token-hash=h6yphW1qnM0n_NOWaf8qtszMRLXEwIxfk5beu4RxdT0%3D" alt="noellabo " width="100"></td> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/2384390/5681180e1efb46a8b28e0e8d4c8b9037/1.jpg?token-time=2145916800&token-hash=SJcMy-Q1BcS940-LFUVOMfR7-5SgrzsEQGhYb3yowFk%3D" alt="CG " width="100"></td> | ||||
| </tr><tr> | ||||
| <td><a href="https://www.patreon.com/user?u=19356899">sheeta.s </a></td> | ||||
| <td><a href="https://www.patreon.com/user?u=5827393">motcha </a></td> | ||||
| <td><a href="https://www.patreon.com/user?u=20494440">axtuki1 </a></td> | ||||
| <td><a href="https://www.patreon.com/user?u=13737140">Satsuki Yanagi</a></td> | ||||
| @@ -188,9 +188,9 @@ Please see the [Contribution Guide](./CONTRIBUTING.md). | ||||
| <td><a href="https://www.patreon.com/user?u=4389829">natalie</a></td> | ||||
| <td><a href="https://www.patreon.com/user?u=26144593">EBISUME</a></td> | ||||
| <td><a href="https://www.patreon.com/noellabo">noellabo </a></td> | ||||
| <td><a href="https://www.patreon.com/Corset">CG </a></td> | ||||
| </tr></table> | ||||
| <table><tr> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/2384390/5681180e1efb46a8b28e0e8d4c8b9037/1.jpg?token-time=2145916800&token-hash=SJcMy-Q1BcS940-LFUVOMfR7-5SgrzsEQGhYb3yowFk%3D" alt="CG " width="100"></td> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/18072312/98e894d960314fa7bc236a72a39488fe/1.jpg?token-time=2145916800&token-hash=7bkMqTwHPRsJPGAq42PYdDXDZBVGLqdgr1ZmBxX8GFQ%3D" alt="Hekovic " width="100"></td> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/24641572/b4fd175424814f15b0ca9178d2d2d2e4/1.png?token-time=2145916800&token-hash=e2fyqdbuJbpCckHcwux7rbuW6OPkKdERcus0u2wIEWU%3D" alt="uroco @99" width="100"></td> | ||||
| <td><img src="https://c8.patreon.com/2/200/14661394" alt="Chandler " width="100"></td> | ||||
| @@ -199,6 +199,7 @@ Please see the [Contribution Guide](./CONTRIBUTING.md). | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/9481273/7fa89168e72943859c3d3c96e424ed31/4.jpeg?token-time=2145916800&token-hash=5w1QV1qXe-NdWbdFmp1H7O_-QBsSiV0haumk3XTHIEg%3D" alt="Efertone " width="100"></td> | ||||
| <td><img src="https://c10.patreonusercontent.com/3/eyJ3IjoyMDB9/patreon-media/p/user/12531784/93a45137841849329ba692da92ac7c60/1.jpeg?token-time=2145916800&token-hash=vGe7wXGqmA8Q7m-kDNb6fyGdwk-Dxk4F-ut8ZZu51RM%3D" alt="Takashi Shibuya" width="100"></td> | ||||
| </tr><tr> | ||||
| <td><a href="https://www.patreon.com/Corset">CG </a></td> | ||||
| <td><a href="https://www.patreon.com/hekovic">Hekovic </a></td> | ||||
| <td><a href="https://www.patreon.com/user?u=24641572">uroco @99</a></td> | ||||
| <td><a href="https://www.patreon.com/user?u=14661394">Chandler </a></td> | ||||
| @@ -208,7 +209,7 @@ Please see the [Contribution Guide](./CONTRIBUTING.md). | ||||
| <td><a href="https://www.patreon.com/user?u=12531784">Takashi Shibuya</a></td> | ||||
| </tr></table> | ||||
|  | ||||
| **Last updated:** Tue, 14 Jul 2020 09:00:09 UTC | ||||
| **Last updated:** Sun, 26 Jul 2020 07:00:10 UTC | ||||
| <!-- PATREON_END --> | ||||
|  | ||||
| [backer-url]: #backers | ||||
|   | ||||
							
								
								
									
										11
									
								
								gulpfile.ts
									
									
									
									
									
								
							
							
						
						
									
										11
									
								
								gulpfile.ts
									
									
									
									
									
								
							| @@ -7,9 +7,6 @@ import * as gulp from 'gulp'; | ||||
| import * as ts from 'gulp-typescript'; | ||||
| import * as rimraf from 'rimraf'; | ||||
| import * as rename from 'gulp-rename'; | ||||
| const cleanCSS = require('gulp-clean-css'); | ||||
| const sass = require('gulp-dart-sass'); | ||||
| const fiber = require('fibers'); | ||||
|  | ||||
| const locales: { [x: string]: any } = require('./locales'); | ||||
| const meta = require('./package.json'); | ||||
| @@ -61,13 +58,6 @@ gulp.task('cleanall', gulp.parallel('clean', cb => | ||||
| 	rimraf('./node_modules', cb) | ||||
| )); | ||||
|  | ||||
| gulp.task('build:client:styles', () => | ||||
| 	gulp.src('./src/client/style.scss') | ||||
| 		.pipe(sass({ fiber })) | ||||
| 		.pipe(cleanCSS()) | ||||
| 		.pipe(gulp.dest('./built/client/assets/')) | ||||
| ); | ||||
|  | ||||
| gulp.task('copy:client', () => | ||||
| 		gulp.src([ | ||||
| 			'./assets/**/*', | ||||
| @@ -87,7 +77,6 @@ gulp.task('copy:docs', () => | ||||
| ); | ||||
|  | ||||
| gulp.task('build:client', gulp.parallel( | ||||
| 	'build:client:styles', | ||||
| 	'copy:client', | ||||
| 	'copy:docs' | ||||
| )); | ||||
|   | ||||
| @@ -1,5 +1,6 @@ | ||||
| --- | ||||
| _lang_: "العربية" | ||||
| introMisskey: "اهلا بك! ميسكي هو منصة تدوين مصغر لا مركزية ومفتوحة المصدر.\nيمكنك مشاركة \"ملاحظات\" عن ما يجري حولك، وإخبار الجميع عن نفسك 📡\nتسمح لك \"الانفعالات\" بتعبير عن شعورك حول ملاحظات الآخرين 👍\nاكتشف عالمًا جديدًا 🚀" | ||||
| monthAndDay: "{day}/{month}" | ||||
| search: "البحث" | ||||
| notifications: "الإشعارات" | ||||
| @@ -14,8 +15,12 @@ noNotes: "لم يتم العثور على أية ملاحظات" | ||||
| noNotifications: "ليس هناك أية اشعارات" | ||||
| instance: "مثيل الخادم" | ||||
| settings: "الاعدادات" | ||||
| basicSettings: "الاعدادات الأساسية" | ||||
| otherSettings: "إعدادات أخرى" | ||||
| openInWindow: "افتح في نافذة جديدة" | ||||
| profile: "الملف التعريفي" | ||||
| timeline: "الخيط الزمني" | ||||
| noAccountDescription: "لم يكتب هذا المستخدم سيرته بعد." | ||||
| login: "لِج" | ||||
| loggingIn: "جارٍ تسجيل الدخول" | ||||
| logout: "الخروج" | ||||
| @@ -28,22 +33,31 @@ favorite: "إضافة إلى المفضلة" | ||||
| favorites: "المفضلات" | ||||
| unfavorite: "إزالة من المفضلة" | ||||
| pin: "دبّسها على الصفحة الشخصية" | ||||
| unpin: "ألغ تثبيتها من ملفك الشخصي" | ||||
| copyContent: "انسخ المحتوى" | ||||
| copyLink: "انسخ الرابط" | ||||
| delete: "حذف" | ||||
| deleteAndEdit: "إزالة وإعادة الصياغة" | ||||
| deleteAndEditConfirm: "أمتأكد من حذف الملاحظة؟ ستفقد كل مشاركاتها، والتفاعلات، والردود عليها." | ||||
| addToList: "أضفه إلى قائمة" | ||||
| sendMessage: "أرسل رسالة" | ||||
| copyUsername: "انسخ اسم المستخدم" | ||||
| searchUser: "ابحث عن مستخدمين" | ||||
| reply: "رد" | ||||
| loadMore: "عرض المزيد" | ||||
| youGotNewFollower: "يتابعك" | ||||
| receiveFollowRequest: "تلقيت طلب متابعة" | ||||
| followRequestAccepted: "قُبل طلب المتابعة" | ||||
| mention: "أشر الى" | ||||
| mentions: "الإشارات" | ||||
| directNotes: "الملاحظات المباشرة" | ||||
| importAndExport: "إستورد / صدر" | ||||
| import: "استيراد" | ||||
| export: "تصدير" | ||||
| files: "الملفات" | ||||
| download: "تنزيل" | ||||
| driveFileDeleteConfirm: "أمتأكد من حذف ملف {name}؟ كل الملاحظات المُرفق بها هذا الملف ستحذف." | ||||
| unfollowConfirm: "أمتأكد من إلغاء متابعة {name}؟" | ||||
| lists: "القوائم" | ||||
| noLists: "ليس لديك أية قائمة" | ||||
| note: "ملاحظة" | ||||
| @@ -53,8 +67,10 @@ followers: "المتابِعين" | ||||
| followsYou: "يتابعك" | ||||
| createList: "إنشاء قائمة" | ||||
| manageLists: "إدارة القوائم" | ||||
| error: "حدث خطأ ما" | ||||
| error: "خطأ" | ||||
| somethingHappened: "حدث خطأ" | ||||
| retry: "حاول مجددًا" | ||||
| pageLoadError: "فشل تحميل الصفحة" | ||||
| enterListName: "اسم القائمة" | ||||
| privacy: "الخصوصية" | ||||
| makeFollowManuallyApprove: "القبول يدويا طلبات الإشتراك" | ||||
| @@ -64,6 +80,7 @@ followRequest: "طلب اشتراك" | ||||
| followRequests: "طلبات الإشتراك" | ||||
| unfollow: "إلغاء الاشتراك" | ||||
| followRequestPending: "طلبات الإشتراك المعلّقة" | ||||
| enterEmoji: "أدخل إيموجي" | ||||
| unrenote: "إلغاء مشاركة الملاحظة" | ||||
| quote: "اقتبس" | ||||
| pinnedNote: "ملاحظة مدبسة" | ||||
| @@ -71,16 +88,26 @@ you: "أنت" | ||||
| clickToShow: "اضغط للعرض" | ||||
| sensitive: "محتوى حساس" | ||||
| add: "إضافة" | ||||
| reaction: "تفاعل" | ||||
| rememberNoteVisibility: "تذكر إعدادت مدى رؤية الملاحظات" | ||||
| attachCancel: "أزل المرفق" | ||||
| enterFileName: "ادخل اسم الملف" | ||||
| mute: "اكتم" | ||||
| unmute: "إلغاء الكتم" | ||||
| block: "احجب" | ||||
| unblock: "إلغاء الحجب" | ||||
| suspend: "علِق" | ||||
| unsuspend: "ألغ التعليق" | ||||
| blockConfirm: "أمتأكد من حجب هذا الحساب؟" | ||||
| unblockConfirm: "أمتأكد من إلغاء حجب هذا الحساب؟" | ||||
| selectList: "اختر قائمة" | ||||
| editWidgetsExit: "تم" | ||||
| customEmojis: "إيموجي مخصص" | ||||
| addEmoji: "إضافة إيموجي" | ||||
| cacheRemoteFiles: "خزن مؤقتا الملفات البعيدة" | ||||
| autoAcceptFollowed: "اقبل طلبات المتابعة تلقائيا من الحسابات المتابَعة" | ||||
| addAcount: "إضافة حساب" | ||||
| loginFailed: "فشل الولوج" | ||||
| showOnRemote: "رؤيته على مثيل الخادم البُعدي" | ||||
| general: "الرئيسية" | ||||
| wallpaper: "خلفية الشاشة" | ||||
| @@ -88,6 +115,7 @@ setWallpaper: "استخدم خلفية الشاشة" | ||||
| removeWallpaper: "إزالة خلفية الشاشة" | ||||
| searchWith: "البحث: {q}" | ||||
| youHaveNoLists: "لا تمتلك أية قائمة" | ||||
| followConfirm: "أتريد متابعة {name}؟" | ||||
| proxyAccount: "حساب وكيل البروكسي" | ||||
| host: "المضيف" | ||||
| selectUser: "حدّد مستخدمًا" | ||||
| @@ -96,6 +124,8 @@ annotation: "التعليقات" | ||||
| federation: "الفديرالية" | ||||
| instances: "مثيل الخادم" | ||||
| latestRequestSentAt: "آخر طلب أرسِل في" | ||||
| latestRequestReceivedAt: "آخر طلب تُلقي في" | ||||
| storageUsage: "مساحة التخزين المستخدمة" | ||||
| charts: "المنحنيات البيانية" | ||||
| perHour: "في الساعة" | ||||
| perDay: "في اليوم" | ||||
| @@ -127,7 +157,6 @@ processing: "المعالجة جارية" | ||||
| preview: "معاينة" | ||||
| default: "افتراضي" | ||||
| noCustomEmojis: "ليس هناك إيموجيات" | ||||
| customEmojisOfRemote: "الإيموجيات القادمة مِن مثيلات الخوادم الأخرى" | ||||
| federating: "الفديرالية جارية" | ||||
| blocked: "محجوب" | ||||
| suspended: "مُعلّق" | ||||
| @@ -145,6 +174,7 @@ imageUrl: "عنوان URL للصورة" | ||||
| remove: "حذف" | ||||
| removed: "تم حذفه بنجاح" | ||||
| removeAreYouSure: "متأكد من أنك تريد حذف {x}؟" | ||||
| deleteAreYouSure: "متأكد من أنك تريد حذف {x}؟" | ||||
| saved: "تم حفظه" | ||||
| messaging: "الدردشة" | ||||
| upload: "تحميل" | ||||
| @@ -256,8 +286,6 @@ unregister: "إلغاء التسجيل" | ||||
| passwordLessLogin: "لِج مِن دون كلمة سرية" | ||||
| resetPassword: "أعد تعيين كلمتك السرية" | ||||
| newPasswordIs: "كلمتك السرية الجديدة هي {password}" | ||||
| autoReloadWhenDisconnected: "إنعاش تلقائي عندما يُقطَع الإتصال بالخادم" | ||||
| autoNoteWatch: "راقب الملاحظات تلقائيا" | ||||
| share: "شارِك" | ||||
| notFound: "غير موجود" | ||||
| help: "المساعدة" | ||||
| @@ -281,6 +309,7 @@ noteOf: "ملاحظات {user}" | ||||
| inviteToGroup: "دعوة إلى فريق" | ||||
| noMessagesYet: "ليس هناك رسائل بعد" | ||||
| newMessageExists: "لقد تلقيت رسالة جديدة" | ||||
| invitations: "دعوة" | ||||
| invitationCode: "رمز الدعوة" | ||||
| checking: "التحقق جارٍ" | ||||
| available: "متوفر" | ||||
| @@ -314,7 +343,6 @@ total: "المجموع" | ||||
| weekOverWeekChanges: "أسبوعيا" | ||||
| dayOverDayChanges: "يوميا" | ||||
| appearance: "المظهر" | ||||
| clinetSettings: "إعدادات التطبيق" | ||||
| accountSettings: "إعدادات الحساب" | ||||
| promotion: "ترقية" | ||||
| promote: "روِّج" | ||||
| @@ -351,7 +379,18 @@ pluginInstallWarn: "يرجى تنصيب إضافات ذات مصدر موثوق | ||||
| smtpHost: "المضيف" | ||||
| smtpUser: "اسم المستخدم" | ||||
| smtpPass: "الكلمة السرية" | ||||
| display: "المظهر" | ||||
| _mfm: | ||||
|   mention: "أشر الى" | ||||
|   quote: "اقتبس" | ||||
|   emoji: "إيموجي مخصص" | ||||
|   search: "البحث" | ||||
| _reversi: | ||||
|   total: "المجموع" | ||||
| _channel: | ||||
|   featured: "المتداوَلة" | ||||
| _sidebar: | ||||
|   full: "كامل" | ||||
|   icon: "الصورة الرمزية" | ||||
|   hide: "إخفاء" | ||||
| _theme: | ||||
| @@ -363,6 +402,7 @@ _theme: | ||||
|   make: "إنشاء قالب" | ||||
|   alpha: "الشفافية" | ||||
|   keys: | ||||
|     mention: "أشر الى" | ||||
|     messageBg: "خلفية الدردشة" | ||||
| _sfx: | ||||
|   note: "الملاحظات" | ||||
| @@ -505,7 +545,9 @@ _notification: | ||||
|   youWereFollowed: "يتابعك" | ||||
|   _types: | ||||
|     follow: "المتابَعون" | ||||
|     mention: "أشر الى" | ||||
|     quote: "اقتبس" | ||||
|     reaction: "تفاعل" | ||||
| _deck: | ||||
|   _columns: | ||||
|     notifications: "الإشعارات" | ||||
|   | ||||
| @@ -16,6 +16,9 @@ noNotes: "Keine Notizen" | ||||
| noNotifications: "Keine Benachrichtigungen" | ||||
| instance: "Instanz" | ||||
| settings: "Einstellungen" | ||||
| basicSettings: "Allgemeine Einstellungen" | ||||
| otherSettings: "Andere Einstellungen" | ||||
| openInWindow: "In Fenster öffnen" | ||||
| profile: "Profil" | ||||
| timeline: "Chronik" | ||||
| noAccountDescription: "Dieser Nutzer hat seine Profilbeschreibung noch nicht ausgefüllt." | ||||
| @@ -40,6 +43,7 @@ deleteAndEditConfirm: "Möchtest du diese Notiz wirklich löschen und bearbeiten | ||||
| addToList: "Zu Liste hinzufügen" | ||||
| sendMessage: "Nachricht senden" | ||||
| copyUsername: "Benutzernamen kopieren" | ||||
| searchUser: "Benutzersuche" | ||||
| reply: "Antworten" | ||||
| loadMore: "Mehr anzeigen" | ||||
| youGotNewFollower: "Du hast einen neuen Follower" | ||||
| @@ -66,8 +70,11 @@ followers: "Gefolgt von" | ||||
| followsYou: "Folgt dir" | ||||
| createList: "Liste erstellen" | ||||
| manageLists: "Listen verwalten" | ||||
| error: "Ein Problem ist aufgetreten" | ||||
| error: "Fehler" | ||||
| somethingHappened: "Ein Fehler ist aufgetreten" | ||||
| retry: "Wiederholen" | ||||
| pageLoadError: "Laden der Seite fehlgeschlagen." | ||||
| pageLoadErrorDescription: "Dieser Fehler wird meist durch Netzwerkfehler oder den Browser-Cache verursacht. Versuche den Browser-Cache zu leeren und es nach kurzer Zeit noch einmal zu probieren." | ||||
| enterListName: "Listennamen eingeben" | ||||
| privacy: "Privatsphäre" | ||||
| makeFollowManuallyApprove: "Follow-Anfragen benötigen Bestätigung" | ||||
| @@ -84,14 +91,14 @@ quote: "Zitieren" | ||||
| pinnedNote: "Angepinnte Notiz" | ||||
| you: "Du" | ||||
| clickToShow: "Klicke, um diesen Inhalt anzusehen" | ||||
| sensitive: "Dieser Inhalt ist NSFW" | ||||
| sensitive: "NSFW" | ||||
| add: "Hinzufügen" | ||||
| reaction: "Reaktionen" | ||||
| reactionSettingDescription: "Gib deine Lieblingsreaktionen ein, um sie der Reaktionsauswahl hinzuzufügen." | ||||
| rememberNoteVisibility: "Notizsichtbarkeit merken" | ||||
| attachCancel: "Anhang entfernen" | ||||
| markAsSensitive: "Als sensitiv markieren" | ||||
| unmarkAsSensitive: "Markierung als sensitiv zurücknehmen" | ||||
| markAsSensitive: "Als NSFW markieren" | ||||
| unmarkAsSensitive: "Markierung als NSFW zurücknehmen" | ||||
| enterFileName: "Dateinamen eingeben" | ||||
| mute: "Stummschalten" | ||||
| unmute: "Stummschaltung aufheben" | ||||
| @@ -106,6 +113,8 @@ unsuspendConfirm: "Möchtest du die Sperrung dieses Benutzers wirklich aufheben? | ||||
| selectList: "Wähle eine Liste aus" | ||||
| selectAntenna: "Antenne auswählen" | ||||
| selectWidget: "Widget auswählen" | ||||
| editWidgets: "Widgets bearbeiten" | ||||
| editWidgetsExit: "Fertig" | ||||
| customEmojis: "Benutzerdefinierte Emojis" | ||||
| emoji: "Emoji" | ||||
| emojiName: "Emojiname" | ||||
| @@ -165,7 +174,7 @@ clearCachedFilesConfirm: "Sollen alle im Cache gespeicherten Dateien von anderen | ||||
| blockedInstances: "Blockierte Instanzen" | ||||
| blockedInstancesDescription: "Gib den Hostnamen der Instanz an, die blockiert werden soll. Blockierte Instanzen können nicht mehr mit dieser kommunizieren." | ||||
| muteAndBlock: "Stummgeschaltet / Blockiert" | ||||
| mutedUsers: "Stummgestellte Benutzer" | ||||
| mutedUsers: "Stummgeschaltete Benutzer" | ||||
| blockedUsers: "Blockierte Benutzer" | ||||
| noUsers: "Keine Benutzer" | ||||
| editProfile: "Profil bearbeiten" | ||||
| @@ -177,7 +186,6 @@ processing: "In Bearbeitung" | ||||
| preview: "Vorschau" | ||||
| default: "Standard" | ||||
| noCustomEmojis: "Es existieren keine Emojis" | ||||
| customEmojisOfRemote: "Emojis von anderen Instanzen" | ||||
| noJobs: "Es gibt keine Jobs" | ||||
| federating: "Föderiert" | ||||
| blocked: "Blockiert" | ||||
| @@ -206,6 +214,8 @@ imageUrl: "Bild-URL" | ||||
| remove: "Löschen" | ||||
| removed: "Erfolgreich gelöscht" | ||||
| removeAreYouSure: "Möchtest du \"{x}\" wirklich löschen?" | ||||
| deleteAreYouSure: "Möchtest du \"{x}\" wirklich löschen?" | ||||
| resetAreYouSure: "Wirklich zurücksetzen?" | ||||
| saved: "Gespeichert" | ||||
| messaging: "Chat" | ||||
| upload: "Hochladen" | ||||
| @@ -263,7 +273,8 @@ copyUrl: "URL kopieren" | ||||
| rename: "Umbenennen" | ||||
| avatar: "Profilbild" | ||||
| banner: "Banner" | ||||
| nsfw: "Dieser Inhalt ist NSFW" | ||||
| nsfw: "NSFW" | ||||
| whenServerDisconnected: "Bei Verbindungsverlust zum Server" | ||||
| disconnectedFromServer: "Verbindung zum Server wurde getrennt" | ||||
| reload: "Aktualisieren" | ||||
| doNothing: "Ignorieren" | ||||
| @@ -364,9 +375,6 @@ unregister: "Deaktivieren" | ||||
| passwordLessLogin: "Passwortloses Anmelden einrichten" | ||||
| resetPassword: "Passwort zurücksetzen" | ||||
| newPasswordIs: "Das neue Passwort ist \"{password}\"" | ||||
| autoReloadWhenDisconnected: "Automatisch aktualisieren wenn die Serververbindung getrennt wird" | ||||
| autoNoteWatch: "Notizen automatisch beobachten" | ||||
| autoNoteWatchDescription: "Werde über Notizen, auf die du reagiert oder geantwortet hast, informiert" | ||||
| reduceUiAnimation: "Animationen der Benutzeroberfläche reduzieren" | ||||
| share: "Teilen" | ||||
| notFound: "Nicht gefunden" | ||||
| @@ -404,6 +412,7 @@ noMessagesYet: "Noch keine Nachrichten" | ||||
| newMessageExists: "Du hast eine neue Nachricht" | ||||
| onlyOneFileCanBeAttached: "Es kann pro Nachricht nur eine Datei angehängt werden" | ||||
| signinRequired: "Anmeldung erforderlich" | ||||
| invitations: "Einladungen" | ||||
| invitationCode: "Einladungscode" | ||||
| checking: "Wird überprüft..." | ||||
| available: "Verfügbar" | ||||
| @@ -445,7 +454,7 @@ total: "Gesamt" | ||||
| weekOverWeekChanges: "Wöchentlich" | ||||
| dayOverDayChanges: "Täglich" | ||||
| appearance: "Aussehen" | ||||
| clinetSettings: "Client-Einstellungen" | ||||
| clientSettings: "Client-Einstellungen" | ||||
| accountSettings: "Benutzerkonto-Einstellungen" | ||||
| promotion: "Hervorgehoben" | ||||
| promote: "Hervorheben" | ||||
| @@ -468,6 +477,7 @@ objectStorageUseSSL: "SSL verwenden" | ||||
| objectStorageUseSSLDesc: "Deaktiviere dies falls du für die API-Verbindungen kein HTTPS verwenden wirst" | ||||
| objectStorageUseProxy: "Über Proxy verbinden" | ||||
| objectStorageUseProxyDesc: "Deaktiviere dies falls du keinen Proxy für den Objektspeicher verwenden wirst" | ||||
| objectStorageSetPublicRead: "Bei Upload auf \"public-read\" stellen" | ||||
| serverLogs: "Serverprotokolle" | ||||
| deleteAll: "Alle löschen" | ||||
| showFixedPostForm: "Bereich zum Schreiben neuer Notizen am Anfang der Chronik anzeigen" | ||||
| @@ -475,6 +485,8 @@ newNoteRecived: "Es gibt neue Notizen" | ||||
| sounds: "Töne" | ||||
| listen: "Anhören" | ||||
| none: "Keine" | ||||
| showInPage: "In Seite anzeigen" | ||||
| popout: "Pop-Up" | ||||
| volume: "Lautstärke" | ||||
| details: "Details" | ||||
| chooseEmoji: "Wähle ein Emoji" | ||||
| @@ -517,7 +529,6 @@ enableInfiniteScroll: "Automatisch mehr Notizen laden" | ||||
| visibility: "Sichtbarkeit" | ||||
| poll: "Umfrage" | ||||
| useCw: "Inhalt verstecken" | ||||
| fixedWidgetsPosition: "Widgetposition fixieren" | ||||
| enablePlayer: "Video-Player öffnen" | ||||
| disablePlayer: "Video-Player schließen" | ||||
| expandTweet: "Tweet ausklappen" | ||||
| @@ -531,6 +542,7 @@ pluginInstallWarn: "Installiere nur vertrauenswürdige Plugins." | ||||
| deck: "Deck" | ||||
| undeck: "Deck verlassen" | ||||
| useBlurEffectForModal: "Weichzeichnungseffekt für Modals verwenden" | ||||
| useFullReactionPicker: "Vollständige Reaktionsauswahl nutzen" | ||||
| generateAccessToken: "Zugriffstoken generieren" | ||||
| permission: "Berechtigungen" | ||||
| enableAll: "Alle aktivieren" | ||||
| @@ -557,9 +569,144 @@ wordMute: "Wort-Stummschaltung" | ||||
| userSaysSomething: "{name} hat etwas gesagt." | ||||
| makeActive: "Aktivieren" | ||||
| display: "Anzeige" | ||||
| copy: "Kopieren" | ||||
| metrics: "Metriken" | ||||
| overview: "Übersicht" | ||||
| logs: "Logs" | ||||
| delayed: "Verzögert" | ||||
| database: "Datenbank" | ||||
| channel: "Kanäle" | ||||
| create: "Erstellen" | ||||
| notificationSetting: "Benachrichtigungseinstellungen" | ||||
| notificationSettingDesc: "Wähle die Art der anzuzeigenden Benachrichtigung" | ||||
| useGlobalSetting: "Globale Einstellung verwenden" | ||||
| useGlobalSettingDesc: "Wenn dies eingeschaltet ist, werden die Benachrichtigungseinstellungen deines Benutzerkontos verwendet. Wenn dies ausgeschaltet ist, können individuelle Einstellungen vorgenommen werden." | ||||
| other: "Andere" | ||||
| regenerateLoginToken: "Login-Token regenerieren" | ||||
| regenerateLoginTokenDescription: "Den bei Logins intern verwendeten Token regenerieren. Normalerweise wird dies nicht benötigt. Bei Regeneration werden alle Geräte ausgeloggt." | ||||
| setMultipleBySeparatingWithSpace: "Trenne Elemente durch ein Leerzeichen um mehrere Einstellungen zu kofigurieren." | ||||
| fileIdOrUrl: "Datei-ID oder URL" | ||||
| chatOpenBehavior: "Verhalten des Chatfensters bei Öffnung" | ||||
| sample: "Beispiel" | ||||
| abuseReports: "Melden" | ||||
| reportAbuse: "Melden" | ||||
| reportAbuseOf: "{name} melden" | ||||
| fillAbuseReportDescription: "Bitte gib Details für diese Meldung an. Falls es sich um eine spezielle Notiz handelt, bitte gib dessen URL an." | ||||
| abuseReported: "Die Meldung wurde versendet. Vielen Dank." | ||||
| send: "Senden" | ||||
| abuseMarkAsResolved: "Meldung als gelöst markieren" | ||||
| openInNewTab: "In neuem Tab öffnen" | ||||
| openInSideView: "In Seitenansicht öffnen" | ||||
| defaultNavigationBehaviour: "Standardnavigationsverhalten" | ||||
| editTheseSettingsMayBreakAccount: "Bei Bearbeitung dieser Einstellungen besteht die Gefahr, dein Benutzerkonto zu beschädigen." | ||||
| instanceTicker: "Instanz-Informationen von Notizen" | ||||
| waitingFor: "Warte auf {x}" | ||||
| random: "Zufällig" | ||||
| system: "System" | ||||
| switchUi: "UI wechseln" | ||||
| desktop: "Desktop" | ||||
| _mfm: | ||||
|   cheatSheet: "MFM Spickzettel" | ||||
|   intro: "MFM ist eine an vielen Stellen verwendbare und Misskey-exklusive Markup-Sprache. Hier kannst du eine Liste von verfügbarer MFM-Syntax anschauen." | ||||
|   dummy: "Misskey erweitert die Welt des Fediverse" | ||||
|   mention: "Erwähnung" | ||||
|   mentionDescription: "Mit At-Zeichen und Nutzername kann ein individueller Nutzer angegeben werden." | ||||
|   hashtag: "Hashtag" | ||||
|   hashtagDescription: "Mit einer Raute und Text kann ein Hashtag angegeben werden." | ||||
|   url: "URL" | ||||
|   urlDescription: "URLs können angezeigt werden." | ||||
|   link: "Link" | ||||
|   linkDescription: "Ein spezifizierter Textabschnitt kann als URL angezeigt werden." | ||||
|   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 lassen." | ||||
|   inlineCode: "Code (Eingebettet)" | ||||
|   inlineCodeDescription: "Syntax-Hervorhebung für (Programm-)Code eingebettet anzeigen lassen." | ||||
|   blockCode: "Code (Block)" | ||||
|   blockCodeDescription: "Syntax-Hervorhebung für mehrzeiligen (Programm-)Code als Block anzeigen lassen." | ||||
|   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 lassen." | ||||
|   emoji: "Benutzerdefinierte Emojis" | ||||
|   emojiDescription: "Emoji-Namen mit Doppelpunkten umschließen, um benutzerdefinierte Emojis anzeigen zu lassen." | ||||
|   search: "Suche" | ||||
|   searchDescription: "Eine vorgefertige Suchanfragebox anzeigen lassen." | ||||
|   flip: "Spiegelung" | ||||
|   flipDescription: "Inhalt horizontal oder vertikal gespiegelt anzeigen lassen." | ||||
|   jelly: "Animation (Dehnen)" | ||||
|   jellyDescription: "Verleiht eine sich dehnende Animation." | ||||
|   tada: "Animation (Tada)" | ||||
|   tadaDescription: "Verleiht eine Animation mit \"Tada!\"-Gefühl" | ||||
|   jump: "Animation (Sprung)" | ||||
|   jumpDescription: "Verleiht eine springende Animation." | ||||
|   bounce: "Animation (Federn)" | ||||
|   bounceDescription: "Erzeugt eine federnde Animation." | ||||
|   shake: "Animation (Zittern)" | ||||
|   shakeDescription: "Verleiht eine zitternde Animation." | ||||
|   twitch: "Animation (Zucken)" | ||||
|   twitchDescription: "Verleiht eine sehr stark zuckende Animation." | ||||
|   spin: "Animation (Rotieren)" | ||||
|   spinDescription: "Verleiht eine rotierende Animation." | ||||
| _reversi: | ||||
|   reversi: "Reversi" | ||||
|   gameSettings: "Spieleinstellungen" | ||||
|   chooseBoard: "Spielbrett auswählen" | ||||
|   blackOrWhite: "Schwarz/Weiß" | ||||
|   blackIs: "{name} spielt Schwarz" | ||||
|   rules: "Regeln" | ||||
|   botSettings: "Optionen des Computergegners" | ||||
|   thisGameIsStartedSoon: "Dieses Spiel beginnt in wenigen Sekunden" | ||||
|   waitingForOther: "Warte auf den Zug des Gegenspielers" | ||||
|   waitingForMe: "Warte auf deinen Zug" | ||||
|   waitingBoth: "Mach dich bereit" | ||||
|   ready: "Bereit" | ||||
|   cancelReady: "Nicht bereit" | ||||
|   opponentTurn: "Zug deines Gegners" | ||||
|   myTurn: "Dein Zug" | ||||
|   turnOf: "Zug von {name}" | ||||
|   pastTurnOf: "Zug von {name}" | ||||
|   surrender: "Aufgeben" | ||||
|   surrendered: "durch Aufgabe" | ||||
|   drawn: "Unentschieden" | ||||
|   won: "{name} hat gesiegt" | ||||
|   black: "Schwarz" | ||||
|   white: "Weiß" | ||||
|   total: "Gesamt" | ||||
|   turnCount: " Zug {count}" | ||||
|   myGames: "Meine Runden" | ||||
|   allGames: "Alle Runden" | ||||
|   ended: "Beendet" | ||||
|   playing: "Laufend" | ||||
|   isLlotheo: "Der mit weniger Steinen gewinnt (Llotheo)" | ||||
|   loopedMap: "Wiederholendes Spielbrett" | ||||
|   canPutEverywhere: "Steine können überall platziert werden" | ||||
| _instanceTicker: | ||||
|   none: "Nie anzeigen" | ||||
|   remote: "Für Benutzer fremder Instanzen anzeigen" | ||||
|   always: "Immer anzeigen" | ||||
| _serverDisconnectedBehavior: | ||||
|   reload: "Automatisch aktualisieren" | ||||
|   dialog: "Warnungsfenster zeigen" | ||||
|   quiet: "Unaufdringlich warnen" | ||||
| _channel: | ||||
|   create: "Kanal erstellen" | ||||
|   edit: "Kanal bearbeiten" | ||||
|   setBanner: "Kanalbanner festlegen" | ||||
|   removeBanner: "Kanalbanner entfernen" | ||||
|   featured: "Trends" | ||||
|   owned: "Besitzt" | ||||
|   following: "Gefolgt" | ||||
|   usersCount: "{n} Teilnehmer" | ||||
|   notesCount: "{n} Notizen" | ||||
| _sidebar: | ||||
|   full: "Voll" | ||||
|   icon: "Profilbild" | ||||
|   icon: "Symbol" | ||||
|   hide: "Ausblenden" | ||||
| _wordMute: | ||||
|   muteWords: "Wort stummschalten" | ||||
| @@ -569,6 +716,7 @@ _wordMute: | ||||
|   hardDescription: "Verhindern, dass Notizen, die die eingestellten Konditionen erfüllen, der Chronik hinzugefügt werden. Zudem werden diese Notizen auch nicht der Chronik hinzugefügt, falls die Konditionen geändert werden." | ||||
|   soft: "Leicht" | ||||
|   hard: "Schwer" | ||||
|   mutedNotes: "Stummgeschaltete Notizen" | ||||
| _theme: | ||||
|   explore: "Themen erforschen" | ||||
|   install: "Thema installieren" | ||||
| @@ -647,6 +795,7 @@ _sfx: | ||||
|   chat: "Chat" | ||||
|   chatBg: "Nachrichten (Hintergrund)" | ||||
|   antenna: "Antennen" | ||||
|   channel: "Kanalbenachrichtigung" | ||||
| _ago: | ||||
|   unknown: "Unbekannt" | ||||
|   future: "Zukunft" | ||||
| @@ -712,9 +861,9 @@ _permissions: | ||||
|   "write:mutes": "Stummschaltungen bearbeiten" | ||||
|   "write:notes": "Notizen schreiben oder löschen" | ||||
|   "read:notifications": "Benachrichtigungen lesen" | ||||
|   "write:notifications": "Mit Benachrichtigungen arbeiten" | ||||
|   "write:notifications": "Benachrichtigungen bearbeiten" | ||||
|   "read:reactions": "Reaktionen lesen" | ||||
|   "write:reactions": "Reaktionen hinzufügen und bearbeiten" | ||||
|   "write:reactions": "Reaktionen hinzufügen und ändern" | ||||
|   "write:votes": "In Umfragen abstimmen" | ||||
|   "read:pages": "Deine Seiten lesen" | ||||
|   "write:pages": "Deine Seiten bearbeiten oder löschen" | ||||
| @@ -722,6 +871,8 @@ _permissions: | ||||
|   "write:page-likes": "Liste der Seiten, die mir gefallen, bearbeiten" | ||||
|   "read:user-groups": "Benutzergruppen lesen" | ||||
|   "write:user-groups": "Benutzergruppen bearbeiten oder löschen" | ||||
|   "read:channels": "Kanäle lesen" | ||||
|   "write:channels": "Kanäle bearbeiten" | ||||
| _auth: | ||||
|   shareAccess: "Möchtest du \"{name}\" authorisieren, auf dieses Benuzerkonto zugreifen zu können?" | ||||
|   shareAccessAsk: "Bist du dir sicher, dass du diese Anwendung authorisieren möchtest, auf dein Benutzerkonto zugreifen zu können?" | ||||
| @@ -755,6 +906,7 @@ _widgets: | ||||
|   photos: "Fotos" | ||||
|   digitalClock: "Digitaluhr" | ||||
|   federation: "Föderation" | ||||
|   postForm: "Neue Notiz anfertigen" | ||||
| _cw: | ||||
|   hide: "Ausblenden" | ||||
|   show: "Mehr anzeigen" | ||||
| @@ -796,6 +948,7 @@ _visibility: | ||||
| _postForm: | ||||
|   replyPlaceholder: "Dieser Notiz antworten..." | ||||
|   quotePlaceholder: "Diese Notiz zitieren..." | ||||
|   channelPlaceholder: "In einen Kanal senden" | ||||
|   _placeholders: | ||||
|     a: "Was machst du momentan?" | ||||
|     b: "Was ist um dich herum los?" | ||||
| @@ -928,6 +1081,7 @@ _pages: | ||||
|   my: "Meine Seiten" | ||||
|   liked: "Seiten, die mir gefallen" | ||||
|   inspector: "Inspektor" | ||||
|   contents: "Inhalt" | ||||
|   content: "Inhalt" | ||||
|   variables: "Variablen" | ||||
|   title: "Titel" | ||||
| @@ -1210,14 +1364,17 @@ _notification: | ||||
|   youWereInvitedToGroup: "Du wurdest in eine Gruppe eingeladen" | ||||
|   _types: | ||||
|     all: "Alle" | ||||
|     follow: "Folgt" | ||||
|     mention: "Erwähnung" | ||||
|     follow: "Neue Follower" | ||||
|     mention: "Erwähnungen" | ||||
|     reply: "Antworten" | ||||
|     renote: "Renote" | ||||
|     quote: "Zitieren" | ||||
|     renote: "Renotes" | ||||
|     quote: "Zitationen" | ||||
|     reaction: "Reaktionen" | ||||
|     pollVote: "Umfragen" | ||||
|     receiveFollowRequest: "Follow-Anfragen" | ||||
|     pollVote: "Antworten auf Umfragen" | ||||
|     receiveFollowRequest: "Follow-Anfrage erhalten" | ||||
|     followRequestAccepted: "Follow-Anfrage akzeptiert" | ||||
|     groupInvited: "Gruppeneinladung erhalten" | ||||
|     app: "Benachrichtigungen von Apps" | ||||
| _deck: | ||||
|   alwaysShowMainColumn: "Hauptspalte immer zeigen" | ||||
|   columnAlign: "Spalten ausrichten" | ||||
|   | ||||
| @@ -16,6 +16,9 @@ noNotes: "No notes" | ||||
| noNotifications: "No notifications" | ||||
| instance: "Instance" | ||||
| settings: "Settings" | ||||
| basicSettings: "Basic Settings" | ||||
| otherSettings: "Other Settings" | ||||
| openInWindow: "Open in window" | ||||
| profile: "Profile" | ||||
| timeline: "Timeline" | ||||
| noAccountDescription: "This user has not written their bio yet." | ||||
| @@ -29,7 +32,7 @@ users: "Users" | ||||
| addUser: "Add a user" | ||||
| favorite: "Favorite" | ||||
| favorites: "Favorites" | ||||
| unfavorite: "Undo favorite" | ||||
| unfavorite: "Unfavorite" | ||||
| pin: "Pin to profile" | ||||
| unpin: "Unpin from profile" | ||||
| copyContent: "Copy contents" | ||||
| @@ -40,6 +43,7 @@ deleteAndEditConfirm: "Are you sure you want to delete this note and edit it? Yo | ||||
| addToList: "Add to list" | ||||
| sendMessage: "Send a message" | ||||
| copyUsername: "Copy username" | ||||
| searchUser: "User search" | ||||
| reply: "Reply" | ||||
| loadMore: "Load more" | ||||
| youGotNewFollower: "Followed you" | ||||
| @@ -66,8 +70,11 @@ followers: "Followers" | ||||
| followsYou: "Follows you" | ||||
| createList: "Create list" | ||||
| manageLists: "Manage lists" | ||||
| error: "Something happened :(" | ||||
| error: "Error" | ||||
| somethingHappened: "An error occurred" | ||||
| retry: "Retry" | ||||
| pageLoadError: "Failed to load page" | ||||
| pageLoadErrorDescription: "This is normally caused by network errors or the browser's cache. Try clearung the cache and then try again after waiting a little while." | ||||
| enterListName: "List name" | ||||
| privacy: "Privacy" | ||||
| makeFollowManuallyApprove: "Follow requests require approval" | ||||
| @@ -106,6 +113,8 @@ unsuspendConfirm: "Are you sure you that want to unsuspend this account?" | ||||
| selectList: "Select a list" | ||||
| selectAntenna: "Select an Antenna" | ||||
| selectWidget: "Select a widget" | ||||
| editWidgets: "Edit widgets" | ||||
| editWidgetsExit: "Done" | ||||
| customEmojis: "Custom Emoji" | ||||
| emoji: "Emoji" | ||||
| emojiName: "Emoji name" | ||||
| @@ -177,7 +186,6 @@ processing: "Processing" | ||||
| preview: "Preview" | ||||
| default: "Default" | ||||
| noCustomEmojis: "There are no emojis" | ||||
| customEmojisOfRemote: "Emojis from other instances" | ||||
| noJobs: "There are no jobs" | ||||
| federating: "Federating" | ||||
| blocked: "Blocked" | ||||
| @@ -206,6 +214,8 @@ imageUrl: "Image URL" | ||||
| remove: "Delete" | ||||
| removed: "Successfully deleted" | ||||
| removeAreYouSure: "Are you sure that you want to delete \"{x}\"?" | ||||
| deleteAreYouSure: "Are you sure that you want to delete \"{x}\"?" | ||||
| resetAreYouSure: "Really reset?" | ||||
| saved: "Saved" | ||||
| messaging: "Messaging" | ||||
| upload: "Upload" | ||||
| @@ -264,6 +274,7 @@ rename: "Rename" | ||||
| avatar: "Avatar" | ||||
| banner: "Banner" | ||||
| nsfw: "NSFW" | ||||
| whenServerDisconnected: "When losing connection to the server" | ||||
| disconnectedFromServer: "Connection to the server was interrupted." | ||||
| reload: "Refresh" | ||||
| doNothing: "Ignore" | ||||
| @@ -364,9 +375,6 @@ unregister: "Unregister" | ||||
| passwordLessLogin: "Set up password-less login" | ||||
| resetPassword: "Reset password" | ||||
| newPasswordIs: "The new password is \"{password}\"" | ||||
| autoReloadWhenDisconnected: "Auto refresh when disconnected from server" | ||||
| autoNoteWatch: "Watch note automatically" | ||||
| autoNoteWatchDescription: "Get notified about the notes which you reactioned or replied." | ||||
| reduceUiAnimation: "Reduce UI animation" | ||||
| share: "Share" | ||||
| notFound: "Not found" | ||||
| @@ -397,13 +405,14 @@ next: "Next" | ||||
| retype: "Enter again" | ||||
| noteOf: "{user}'s notes" | ||||
| inviteToGroup: "Invite to group" | ||||
| maxNoteTextLength: "Character limit of the note" | ||||
| maxNoteTextLength: "Character limit of notes" | ||||
| quoteAttached: "Quoted" | ||||
| quoteQuestion: "Do you want to append a quote?" | ||||
| noMessagesYet: "No messages yet" | ||||
| newMessageExists: "You've got a new message" | ||||
| onlyOneFileCanBeAttached: "You can only attach one file to a message" | ||||
| signinRequired: "Please sign in" | ||||
| invitations: "Invitations" | ||||
| invitationCode: "Invitation code" | ||||
| checking: "Checking" | ||||
| available: "Available" | ||||
| @@ -445,7 +454,7 @@ total: "Total" | ||||
| weekOverWeekChanges: "Weekly" | ||||
| dayOverDayChanges: "Daily" | ||||
| appearance: "Appearance" | ||||
| clinetSettings: "Client Settings" | ||||
| clientSettings: "Client settings" | ||||
| accountSettings: "Account Settings" | ||||
| promotion: "Promoted" | ||||
| promote: "Promote" | ||||
| @@ -468,6 +477,7 @@ objectStorageUseSSL: "Use SSL" | ||||
| objectStorageUseSSLDesc: "Turn off this if you are not going to use HTTPS for API connection" | ||||
| objectStorageUseProxy: "Connect over Proxy" | ||||
| objectStorageUseProxyDesc: "Turn off this if you are not going to use Proxy for ObjectStorage connection" | ||||
| objectStorageSetPublicRead: "Set \"public-read\" on upload" | ||||
| serverLogs: "Server logs" | ||||
| deleteAll: "Delete all" | ||||
| showFixedPostForm: "Display the posting form at the top of the timeline" | ||||
| @@ -475,6 +485,8 @@ newNoteRecived: "You've got a new note" | ||||
| sounds: "Sounds" | ||||
| listen: "Listen" | ||||
| none: "None" | ||||
| showInPage: "Show in page" | ||||
| popout: "Pop-out" | ||||
| volume: "Volume" | ||||
| details: "Details" | ||||
| chooseEmoji: "Choose an emoji" | ||||
| @@ -517,7 +529,6 @@ enableInfiniteScroll: "Enable infinite scrolling" | ||||
| visibility: "Visiblility" | ||||
| poll: "Poll" | ||||
| useCw: "Hide content" | ||||
| fixedWidgetsPosition: "Make widget position fixed" | ||||
| enablePlayer: "Open video player" | ||||
| disablePlayer: "Close video player" | ||||
| expandTweet: "Expand tweet" | ||||
| @@ -531,6 +542,7 @@ pluginInstallWarn: "Please do not install untrustworthy plugins." | ||||
| deck: "Deck" | ||||
| undeck: "Leave Deck" | ||||
| useBlurEffectForModal: "Use blur effect for modals" | ||||
| useFullReactionPicker: "Use full-size reaction picker" | ||||
| generateAccessToken: "Generate access token" | ||||
| permission: "Permissions" | ||||
| enableAll: "Enable all" | ||||
| @@ -557,6 +569,141 @@ wordMute: "Word mute" | ||||
| userSaysSomething: "{name} said something" | ||||
| makeActive: "Activate" | ||||
| display: "Display" | ||||
| copy: "Copy" | ||||
| metrics: "Metrics" | ||||
| overview: "Overview" | ||||
| logs: "Logs" | ||||
| delayed: "Delayed" | ||||
| database: "Database" | ||||
| channel: "Channels" | ||||
| create: "Create" | ||||
| notificationSetting: "Notification settings" | ||||
| notificationSettingDesc: "Select the type of notification to display" | ||||
| useGlobalSetting: "Use global setting" | ||||
| useGlobalSettingDesc: "If turned on, your account's notification settings will be used. If turned off, individual configurations can be made." | ||||
| other: "Other" | ||||
| regenerateLoginToken: "Regenerate login token" | ||||
| regenerateLoginTokenDescription: "Regenerate the token used internally during login. Normally this action is not necessary. If regenerated, all devices will be logged out." | ||||
| setMultipleBySeparatingWithSpace: "You can set multiple by separating them with spaces." | ||||
| fileIdOrUrl: "File-ID or URL" | ||||
| chatOpenBehavior: "Behavior of the chat window when opened" | ||||
| sample: "Sample" | ||||
| abuseReports: "Reports" | ||||
| reportAbuse: "Report" | ||||
| reportAbuseOf: "Report {name}" | ||||
| fillAbuseReportDescription: "Please fill in the report details. If it is about a specific note, please include its URL." | ||||
| abuseReported: "Your report has been sent. Thank you very much." | ||||
| send: "Send" | ||||
| abuseMarkAsResolved: "Mark report as resolved" | ||||
| openInNewTab: "Open in new tab" | ||||
| openInSideView: "Open in side view" | ||||
| defaultNavigationBehaviour: "Default navigation behavior" | ||||
| editTheseSettingsMayBreakAccount: "Editing these settings may damage your account." | ||||
| instanceTicker: "Instance information of notes" | ||||
| waitingFor: "Waiting for {x}" | ||||
| random: "Random" | ||||
| system: "System" | ||||
| switchUi: "Switch UI" | ||||
| desktop: "Desktop" | ||||
| _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: "Using an At-Symbol and a username, you can specify a specific user." | ||||
|   hashtag: "Hashtag" | ||||
|   hashtagDescription: "Using a number sign and text, you can specify a hashtag." | ||||
|   url: "URL" | ||||
|   urlDescription: "URLs can be displayed." | ||||
|   link: "Link" | ||||
|   linkDescription: "Specific parts of text can be displayed as URL." | ||||
|   bold: "Bold" | ||||
|   boldDescription: "Highlights letters by making them thicker." | ||||
|   small: "Small" | ||||
|   smallDescription: "Displays contents small and thinn." | ||||
|   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 (In-line)" | ||||
|   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 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: "Infuses a jelly-like animation." | ||||
|   tada: "Animation (Tada)" | ||||
|   tadaDescription: "Infuses a \"Tada!\"-like animation." | ||||
|   jump: "Animation (Jump)" | ||||
|   jumpDescription: "Infuses a jumping animation." | ||||
|   bounce: "Animation (Bounce)" | ||||
|   bounceDescription: "Causes a bouncy animation." | ||||
|   shake: "Animation (Shake)" | ||||
|   shakeDescription: "Infuses a shaking animation." | ||||
|   twitch: "Animation (Twitch)" | ||||
|   twitchDescription: "Infuses a strongly twitching animation." | ||||
|   spin: "Animation (Spin)" | ||||
|   spinDescription: "Infuses a spinning animation." | ||||
| _reversi: | ||||
|   reversi: "Reversi" | ||||
|   gameSettings: "Game settings" | ||||
|   chooseBoard: "Choose a board" | ||||
|   blackOrWhite: "Black/White" | ||||
|   blackIs: "{name} is playing Black" | ||||
|   rules: "Rules" | ||||
|   botSettings: "Bot options" | ||||
|   thisGameIsStartedSoon: "The game will start in a few seconds" | ||||
|   waitingForOther: "Waiting for the opponent's turn" | ||||
|   waitingForMe: "Waiting for your turn" | ||||
|   waitingBoth: "Get ready" | ||||
|   ready: "Ready" | ||||
|   cancelReady: "Cancel ready" | ||||
|   opponentTurn: "Opponent's turn" | ||||
|   myTurn: "Your turn" | ||||
|   turnOf: "{name}'s turn" | ||||
|   pastTurnOf: "{name}'s turn" | ||||
|   surrender: "Surrender" | ||||
|   surrendered: "By surrender" | ||||
|   drawn: "Draw" | ||||
|   won: "{name}'s win" | ||||
|   black: "Black" | ||||
|   white: "White" | ||||
|   total: "Total" | ||||
|   turnCount: "Turn {count}" | ||||
|   myGames: "My rounds" | ||||
|   allGames: "All rounds" | ||||
|   ended: "Ended" | ||||
|   playing: "Currently playing" | ||||
|   isLlotheo: "The one with fewer stones wins (Llotheo)" | ||||
|   loopedMap: "Looped map" | ||||
|   canPutEverywhere: "Tiles are placeable everywhere" | ||||
| _instanceTicker: | ||||
|   none: "Never show" | ||||
|   remote: "Show for remote users" | ||||
|   always: "Always show" | ||||
| _serverDisconnectedBehavior: | ||||
|   reload: "Automatically reload" | ||||
|   dialog: "Show warning dialog" | ||||
|   quiet: "Show unobtrusive warning" | ||||
| _channel: | ||||
|   create: "Create channel" | ||||
|   edit: "Edit channel" | ||||
|   setBanner: "Set banner" | ||||
|   removeBanner: "Remove banner" | ||||
|   featured: "Trending" | ||||
|   owned: "Owned" | ||||
|   following: "Followed" | ||||
|   usersCount: "{n} Participants" | ||||
|   notesCount: "{n} Notes" | ||||
| _sidebar: | ||||
|   full: "Full" | ||||
|   icon: "Avatar" | ||||
| @@ -569,6 +716,7 @@ _wordMute: | ||||
|   hardDescription: "Prevent notes fulfilling the set conditions from being added to the timeline. In addition, these notes will not be added to the timeline even if the conditions are changed." | ||||
|   soft: "Soft" | ||||
|   hard: "Hard" | ||||
|   mutedNotes: "Muted notes" | ||||
| _theme: | ||||
|   explore: "Explore Themes" | ||||
|   install: "Install theme" | ||||
| @@ -647,6 +795,7 @@ _sfx: | ||||
|   chat: "Messaging" | ||||
|   chatBg: "Messaging (Background)" | ||||
|   antenna: "Antenna Reception" | ||||
|   channel: "Channel notifications" | ||||
| _ago: | ||||
|   unknown: "Unknown" | ||||
|   future: "Future" | ||||
| @@ -722,6 +871,8 @@ _permissions: | ||||
|   "write:page-likes": "Edit likes on pages" | ||||
|   "read:user-groups": "View user groups" | ||||
|   "write:user-groups": "Edit or delete user groups" | ||||
|   "read:channels": "Read channels" | ||||
|   "write:channels": "Modify channels" | ||||
| _auth: | ||||
|   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?" | ||||
| @@ -755,6 +906,7 @@ _widgets: | ||||
|   photos: "Photos" | ||||
|   digitalClock: "Digital clock" | ||||
|   federation: "Federation" | ||||
|   postForm: "Compose a note" | ||||
| _cw: | ||||
|   hide: "Hide" | ||||
|   show: "Load more" | ||||
| @@ -796,6 +948,7 @@ _visibility: | ||||
| _postForm: | ||||
|   replyPlaceholder: "Reply to this note..." | ||||
|   quotePlaceholder: "Quote this note..." | ||||
|   channelPlaceholder: "Post to channel" | ||||
|   _placeholders: | ||||
|     a: "What are you up to?" | ||||
|     b: "What's happening around you?" | ||||
| @@ -928,6 +1081,7 @@ _pages: | ||||
|   my: "My pages" | ||||
|   liked: "Liked pages" | ||||
|   inspector: "Inspector" | ||||
|   contents: "Content" | ||||
|   content: "Page block" | ||||
|   variables: "Variables" | ||||
|   title: "Title" | ||||
| @@ -1210,14 +1364,17 @@ _notification: | ||||
|   youWereInvitedToGroup: "Invited to group" | ||||
|   _types: | ||||
|     all: "All" | ||||
|     follow: "Following" | ||||
|     mention: "Mention" | ||||
|     follow: "Follows" | ||||
|     mention: "Mentions" | ||||
|     reply: "Replies" | ||||
|     renote: "Renote" | ||||
|     quote: "Quote" | ||||
|     reaction: "Reaction" | ||||
|     pollVote: "Polls" | ||||
|     receiveFollowRequest: "Follow requests" | ||||
|     renote: "Renotes" | ||||
|     quote: "Quotes" | ||||
|     reaction: "Reactions" | ||||
|     pollVote: "Votes on polls" | ||||
|     receiveFollowRequest: "Follow request received" | ||||
|     followRequestAccepted: "Follow request accepted" | ||||
|     groupInvited: "Invited to groups" | ||||
|     app: "Notifications from apps" | ||||
| _deck: | ||||
|   alwaysShowMainColumn: "Always show main column" | ||||
|   columnAlign: "Align columns" | ||||
|   | ||||
| @@ -16,6 +16,9 @@ noNotes: "No hay notas" | ||||
| noNotifications: "No hay notificaciones" | ||||
| instance: "Instancia" | ||||
| settings: "Configuración" | ||||
| basicSettings: "Configuración Básica" | ||||
| otherSettings: "Configuración avanzada" | ||||
| openInWindow: "Abrir en una ventana" | ||||
| profile: "Perfil" | ||||
| timeline: "Linea de tiempo" | ||||
| noAccountDescription: "Este usuario no tiene una descripción" | ||||
| @@ -40,6 +43,7 @@ deleteAndEditConfirm: "¿Quieres borrar y editar este nota? Las reacciones, reno | ||||
| addToList: "Agregar a lista" | ||||
| sendMessage: "Énviar mensaje" | ||||
| copyUsername: "Copiar nombre de usuario" | ||||
| searchUser: "Búsqueda de usuarios" | ||||
| reply: "Responder" | ||||
| loadMore: "Ver más" | ||||
| youGotNewFollower: "te ha seguido" | ||||
| @@ -66,8 +70,11 @@ followers: "Seguidores" | ||||
| followsYou: "Te sigue" | ||||
| createList: "Crear lista" | ||||
| manageLists: "Administrar listas" | ||||
| error: "Ocurrió un problema" | ||||
| error: "Error" | ||||
| somethingHappened: "Ocurrió un error" | ||||
| retry: "Reintentar" | ||||
| pageLoadError: "Error al leer la página" | ||||
| pageLoadErrorDescription: "Normalmente es debido a la red o al caché del navegador. Por favor limpie el caché o intente más tarde." | ||||
| enterListName: "Ingrese nombre de lista" | ||||
| privacy: "Privacidad" | ||||
| makeFollowManuallyApprove: "Aprobar manualmente las solicitudes de seguimiento" | ||||
| @@ -106,6 +113,8 @@ unsuspendConfirm: "¿Quiere dejar de suspender esta cuenta?" | ||||
| selectList: "Seleccione una lista" | ||||
| selectAntenna: "Seleccionar antena" | ||||
| selectWidget: "Seleccionar widget" | ||||
| editWidgets: "Editar widgets" | ||||
| editWidgetsExit: "Terminar edición" | ||||
| customEmojis: "Emojis personalizados" | ||||
| emoji: "Emoji" | ||||
| emojiName: "Nombre del emoji" | ||||
| @@ -177,7 +186,6 @@ processing: "Procesando" | ||||
| preview: "Vista previa" | ||||
| default: "Predeterminado" | ||||
| noCustomEmojis: "No hay emojis personalizados" | ||||
| customEmojisOfRemote: "Emojis remotos" | ||||
| noJobs: "No hay trabajos" | ||||
| federating: "Federando" | ||||
| blocked: "Bloqueando" | ||||
| @@ -206,6 +214,7 @@ imageUrl: "URL de la imágen" | ||||
| remove: "Borrar" | ||||
| removed: "Borrado" | ||||
| removeAreYouSure: "¿Desea borrar \"{x}\"?" | ||||
| deleteAreYouSure: "¿Desea borrar \"{x}\"?" | ||||
| saved: "Guardado" | ||||
| messaging: "Chat" | ||||
| upload: "Subir" | ||||
| @@ -264,6 +273,7 @@ rename: "Renombrar" | ||||
| avatar: "Avatar" | ||||
| banner: "Banner" | ||||
| nsfw: "Marcado como sensible" | ||||
| whenServerDisconnected: "Cuando se pierda la conexión con el servidor" | ||||
| disconnectedFromServer: "Desconectado del servidor" | ||||
| reload: "Recargar" | ||||
| doNothing: "No hacer nada" | ||||
| @@ -364,9 +374,6 @@ unregister: "Cancelar registro" | ||||
| passwordLessLogin: "Iniciar sesión sin contraseña" | ||||
| resetPassword: "Resetear contraseña" | ||||
| newPasswordIs: "La nueva contraseña es \"{password}\"" | ||||
| autoReloadWhenDisconnected: "Recargar automáticamente cuando el servidor está desconectado" | ||||
| autoNoteWatch: "Ver nota automáticamente" | ||||
| autoNoteWatchDescription: "Recibe notificaciones sobre las notas de otros usuarios que a los que respondiste y reaccionaste" | ||||
| reduceUiAnimation: "Reducir la animación de la UI" | ||||
| share: "Compartir" | ||||
| notFound: "No se encuentra" | ||||
| @@ -404,6 +411,7 @@ noMessagesYet: "Aún no hay chat" | ||||
| newMessageExists: "Tienes un mensaje nuevo" | ||||
| onlyOneFileCanBeAttached: "Solo se puede añadir un archivo al mensaje" | ||||
| signinRequired: "Iniciar sesión" | ||||
| invitations: "Invitar" | ||||
| invitationCode: "Código de invitación" | ||||
| checking: "Comprobando" | ||||
| available: "Disponible" | ||||
| @@ -445,7 +453,7 @@ total: "Total" | ||||
| weekOverWeekChanges: "Dif semanal" | ||||
| dayOverDayChanges: "Dif diaria" | ||||
| appearance: "Apariencia" | ||||
| clinetSettings: "Ajustes del cliente" | ||||
| clientSettings: "Configuración del cliente" | ||||
| accountSettings: "Ajustes de cuenta" | ||||
| promotion: "Promovido" | ||||
| promote: "Promover" | ||||
| @@ -468,6 +476,7 @@ objectStorageUseSSL: "Usar SSL" | ||||
| objectStorageUseSSLDesc: "Desactive esto si no va a usar HTTPS para la conexión API" | ||||
| objectStorageUseProxy: "Conectarse a través de Proxy" | ||||
| objectStorageUseProxyDesc: "Desactive esto si no va a usar Proxy para la conexión de Almacenamiento de objetos" | ||||
| objectStorageSetPublicRead: "Seleccionar \"public-read\" al subir " | ||||
| serverLogs: "Registros del servidor" | ||||
| deleteAll: "Eliminar todos" | ||||
| showFixedPostForm: "Mostrar el formulario de las entradas encima de la línea de tiempo" | ||||
| @@ -475,6 +484,8 @@ newNoteRecived: "Tienes una nota nuevo" | ||||
| sounds: "Sonidos" | ||||
| listen: "Escuchar" | ||||
| none: "Ninguna" | ||||
| showInPage: "Mostrar en la página" | ||||
| popout: "Popout" | ||||
| volume: "Volumen" | ||||
| details: "Detalles" | ||||
| chooseEmoji: "Elije un emoji" | ||||
| @@ -517,7 +528,6 @@ enableInfiniteScroll: "Activar scroll infinito" | ||||
| visibility: "Visibilidad" | ||||
| poll: "Encuesta" | ||||
| useCw: "Esconder contenidos" | ||||
| fixedWidgetsPosition: "Fijar la posición de los widgets" | ||||
| enablePlayer: "Abrir reproductor" | ||||
| disablePlayer: "Cerrar reproductor" | ||||
| expandTweet: "Expandir tweet" | ||||
| @@ -531,6 +541,7 @@ pluginInstallWarn: "Por favor no instale plugins que no son de confianza" | ||||
| deck: "Deck" | ||||
| undeck: "Quitar deck" | ||||
| useBlurEffectForModal: "Usar efecto borroso en modales" | ||||
| useFullReactionPicker: "Reacción" | ||||
| generateAccessToken: "Generar token de acceso" | ||||
| permission: "Permisos" | ||||
| enableAll: "Activar todo" | ||||
| @@ -556,7 +567,113 @@ testEmail: "Prueba de envío" | ||||
| wordMute: "Silenciar palabras" | ||||
| userSaysSomething: "{name} dijo algo" | ||||
| makeActive: "Activar" | ||||
| display: "Apariencia" | ||||
| copy: "Copiar" | ||||
| metrics: "Métricas" | ||||
| overview: "Resumen" | ||||
| logs: "Registros" | ||||
| delayed: "atrasado" | ||||
| database: "Base de datos" | ||||
| channel: "Canal" | ||||
| create: "Crear" | ||||
| notificationSetting: "Ajustes de Notificaciones" | ||||
| notificationSettingDesc: "Por favor elija el tipo de notificación a mostrar" | ||||
| useGlobalSetting: "Usar ajustes globales" | ||||
| useGlobalSettingDesc: "Al activarse, se usará la configuración de notificaciones de la cuenta, al desactivarse se pueden hacer configuraciones particulares." | ||||
| other: "Otro" | ||||
| regenerateLoginToken: "Regenerar token de login" | ||||
| regenerateLoginTokenDescription: "Regenerar el token usado internamente durante el login. No siempre es necesario hacerlo. Al hacerlo de nuevo, se deslogueará en todos los dispositivos." | ||||
| setMultipleBySeparatingWithSpace: "Puedes añadir mas de uno, separado por espacios." | ||||
| fileIdOrUrl: "Id del archivo o URL" | ||||
| chatOpenBehavior: "Comportamiento al abrir el chat" | ||||
| sample: "Muestra" | ||||
| abuseReports: "Reportes" | ||||
| reportAbuse: "Reportar" | ||||
| reportAbuseOf: "Reportar a {name}" | ||||
| fillAbuseReportDescription: "Ingrese los detalles del reporte. Si hay una nota en particular, ingrese la URL de esta." | ||||
| abuseReported: "Se ha enviado el reporte. Muchas gracias." | ||||
| send: "Enviar" | ||||
| abuseMarkAsResolved: "Marcar reporte como resuelto" | ||||
| openInNewTab: "Abrir en una Nueva Pestaña" | ||||
| openInSideView: "Abrir en una vista al costado" | ||||
| defaultNavigationBehaviour: "Navegación por defecto" | ||||
| editTheseSettingsMayBreakAccount: "Editar estas configuraciones puede dañar su cuenta." | ||||
| instanceTicker: "Información de notas de la instancia" | ||||
| waitingFor: "Esperando a {x}" | ||||
| random: "Aleatorio" | ||||
| system: "Sistema" | ||||
| switchUi: "Cambiar interfaz de usuario" | ||||
| desktop: "Escritorio" | ||||
| _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." | ||||
|   mention: "Menciones" | ||||
|   mentionDescription: "El signo @ seguido de un nombre de usuario se puede utilizar para notificar a un usuario en particular." | ||||
|   hashtag: "Hashtag" | ||||
|   url: "URL" | ||||
|   link: "Vínculo" | ||||
|   bold: "Negrita" | ||||
|   center: "Centrar" | ||||
|   blockCode: "Código (bloque)" | ||||
|   blockCodeDescription: "Código de resaltado de sintaxis, como programas de varias líneas con bloques." | ||||
|   quote: "Citar" | ||||
|   emoji: "Emojis personalizados" | ||||
|   search: "Buscar" | ||||
|   flip: "Echar de un capirotazo" | ||||
|   flipDescription: "Voltea el contenido hacia arriba / abajo o hacia la izquierda / derecha." | ||||
| _reversi: | ||||
|   reversi: "Reversi" | ||||
|   gameSettings: "Configuración del juego" | ||||
|   chooseBoard: "Elegir tablero" | ||||
|   blackOrWhite: "Blancas/Negras" | ||||
|   blackIs: "{name} juega con fichas negras" | ||||
|   rules: "Reglas" | ||||
|   botSettings: "Opciones del bot" | ||||
|   thisGameIsStartedSoon: "El juego empezará en segundos" | ||||
|   waitingForOther: "Esperando el turno del adversario" | ||||
|   waitingForMe: "Esperando mi turno" | ||||
|   waitingBoth: "Prepárate" | ||||
|   ready: "Listo" | ||||
|   cancelReady: "No estoy listo" | ||||
|   opponentTurn: "Turno del adversario" | ||||
|   myTurn: "Mi turno" | ||||
|   turnOf: "Turno de {name}" | ||||
|   pastTurnOf: "Turno de {name}" | ||||
|   surrender: "Rendirse" | ||||
|   surrendered: "Por rendirse" | ||||
|   drawn: "Empate" | ||||
|   won: "{name} ha ganado" | ||||
|   black: "Negro" | ||||
|   white: "Blanco" | ||||
|   total: "Total" | ||||
|   turnCount: "Turno {count}" | ||||
|   myGames: "Mis juegos" | ||||
|   allGames: "Todos los juegos" | ||||
|   ended: "Finalizado" | ||||
|   playing: "Jugando" | ||||
|   isLlotheo: "El que tenga menos fichas gana (LLoTheO)" | ||||
|   loopedMap: "Mapa en bucle" | ||||
|   canPutEverywhere: "Puedes colocar donde quieras" | ||||
| _instanceTicker: | ||||
|   none: "No mostrar" | ||||
|   remote: "Mostrar a usuarios remotos" | ||||
|   always: "Mostrar siempre" | ||||
| _serverDisconnectedBehavior: | ||||
|   reload: "Recargar automáticamente" | ||||
|   dialog: "Mostrar diálogo de advertencia" | ||||
|   quiet: "Advertencia discreta" | ||||
| _channel: | ||||
|   create: "Crear canal" | ||||
|   edit: "Editar canal" | ||||
|   setBanner: "Elegir banner" | ||||
|   removeBanner: "Borrar banner" | ||||
|   featured: "Tendencias" | ||||
|   owned: "Dueño" | ||||
|   following: "Siguiendo" | ||||
|   usersCount: "{n} participantes" | ||||
|   notesCount: "{n} notas" | ||||
| _sidebar: | ||||
|   full: "Completo" | ||||
|   icon: "Avatar" | ||||
|   hide: "Ocultar" | ||||
| _wordMute: | ||||
| @@ -567,6 +684,7 @@ _wordMute: | ||||
|   hardDescription: "Evitar que se agreguen a la linea de tiempo las notas que cumplen las condiciones. Las notas no agregadas seguirán quitadas aunque cambien las condiciones." | ||||
|   soft: "Suave" | ||||
|   hard: "Duro" | ||||
|   mutedNotes: "Notas silenciadas" | ||||
| _theme: | ||||
|   explore: "Explorar temas" | ||||
|   install: "Instalar tema" | ||||
| @@ -645,6 +763,7 @@ _sfx: | ||||
|   chat: "Chat" | ||||
|   chatBg: "Chat (Fondo)" | ||||
|   antenna: "Antena receptora" | ||||
|   channel: "Notificaciones del canal" | ||||
| _ago: | ||||
|   unknown: "Desconocido" | ||||
|   future: "Futuro" | ||||
| @@ -720,6 +839,8 @@ _permissions: | ||||
|   "write:page-likes": "Administrar páginas que te gustan" | ||||
|   "read:user-groups": "Ver grupos de usuarios" | ||||
|   "write:user-groups": "Administrar grupos de usuarios" | ||||
|   "read:channels": "Ver canal" | ||||
|   "write:channels": "Modificar canal" | ||||
| _auth: | ||||
|   shareAccess: "¿Desea permitir el acceso a la cuenta \"{name}\"?" | ||||
|   shareAccessAsk: "¿Está seguro de que desea autorizar esta aplicación para acceder a su cuenta?" | ||||
| @@ -753,6 +874,7 @@ _widgets: | ||||
|   photos: "Fotos" | ||||
|   digitalClock: "Reloj digital" | ||||
|   federation: "Federación" | ||||
|   postForm: "Formulario" | ||||
| _cw: | ||||
|   hide: "Ocultar" | ||||
|   show: "Ver más" | ||||
| @@ -794,6 +916,7 @@ _visibility: | ||||
| _postForm: | ||||
|   replyPlaceholder: "Responder a esta nota" | ||||
|   quotePlaceholder: "Citar esta nota" | ||||
|   channelPlaceholder: "Postear en el canal" | ||||
|   _placeholders: | ||||
|     a: "¿Qué haces?" | ||||
|     b: "¿Te pasó algo?" | ||||
| @@ -926,6 +1049,7 @@ _pages: | ||||
|   my: "Mis páginas" | ||||
|   liked: "Páginas que me gustan" | ||||
|   inspector: "Inspector" | ||||
|   contents: "Contenido" | ||||
|   content: "Bloque de página" | ||||
|   variables: "Variables" | ||||
|   title: "Título" | ||||
| @@ -1214,8 +1338,11 @@ _notification: | ||||
|     renote: "Renotar" | ||||
|     quote: "Citar" | ||||
|     reaction: "Reacción" | ||||
|     pollVote: "Encuestas" | ||||
|     receiveFollowRequest: "Solicitudes de seguimiento" | ||||
|     pollVote: "Votado en la encuesta" | ||||
|     receiveFollowRequest: "Recibió una solicitud de seguimiento" | ||||
|     followRequestAccepted: "El seguimiento fue aceptado" | ||||
|     groupInvited: "Invitado al grupo" | ||||
|     app: "Notificaciones desde aplicaciones" | ||||
| _deck: | ||||
|   alwaysShowMainColumn: "Siempre mostrar la columna principal" | ||||
|   columnAlign: "Alinear columnas" | ||||
|   | ||||
| @@ -16,6 +16,9 @@ noNotes: "Aucune note" | ||||
| noNotifications: "Aucune notification" | ||||
| instance: "Instance" | ||||
| settings: "Paramètres" | ||||
| basicSettings: "Paramètres basiques" | ||||
| otherSettings: "Autres paramètres" | ||||
| openInWindow: "Ouvrir dans une nouvelle fenêtre" | ||||
| profile: "Profil" | ||||
| timeline: "Fil" | ||||
| noAccountDescription: "L’utilisateur·rice n’a pas encore renseigné de biographie de présentation sur son profil." | ||||
| @@ -40,6 +43,7 @@ deleteAndEditConfirm: "Êtes-vous sûr·e de vouloir supprimer cette note et la | ||||
| addToList: "Ajouter à une liste" | ||||
| sendMessage: "Envoyer un message" | ||||
| copyUsername: "Copier le nom d’utilisateur·rice" | ||||
| searchUser: "Chercher un·e utilisateur·rice" | ||||
| reply: "Répondre" | ||||
| loadMore: "Afficher plus …" | ||||
| youGotNewFollower: "Vous suit" | ||||
| @@ -66,8 +70,10 @@ followers: "Abonné·e·s" | ||||
| followsYou: "Vous suit" | ||||
| createList: "Créer une liste" | ||||
| manageLists: "Gérer les listes" | ||||
| error: "Une erreur est survenue" | ||||
| error: "Erreur" | ||||
| somethingHappened: "Une erreur est survenue" | ||||
| retry: "Réessayer" | ||||
| pageLoadError: "Le chargement de la page a échoué" | ||||
| enterListName: "Nom de la liste" | ||||
| privacy: "Confidentialité" | ||||
| makeFollowManuallyApprove: "Accepter manuellement les demandes d’abonnement" | ||||
| @@ -106,6 +112,8 @@ unsuspendConfirm: "Êtes-vous sûr·e de vouloir annuler la suspension de ce com | ||||
| selectList: "Sélectionner une liste" | ||||
| selectAntenna: "Sélectionner une antenne" | ||||
| selectWidget: "Sélectionner un widget" | ||||
| editWidgets: "Modifier les widgets" | ||||
| editWidgetsExit: "Fait" | ||||
| customEmojis: "Émojis personnalisés" | ||||
| emoji: "Émoji" | ||||
| emojiName: "Nom de l’émoji" | ||||
| @@ -177,7 +185,6 @@ processing: "Traitement en cours" | ||||
| preview: "Prévisualisation" | ||||
| default: "Par défaut" | ||||
| noCustomEmojis: "Il n'y a pas d’émoji" | ||||
| customEmojisOfRemote: "Émojis provenant des autres instances" | ||||
| noJobs: "Il n’y a aucune tâche planifiée" | ||||
| federating: "En cours de fédération" | ||||
| blocked: "Bloqué·e" | ||||
| @@ -206,6 +213,7 @@ imageUrl: "URL de l’image" | ||||
| remove: "Supprimer" | ||||
| removed: "Supprimé" | ||||
| removeAreYouSure: "Supprimer «{x}» ?" | ||||
| deleteAreYouSure: "Supprimer «{x}» ?" | ||||
| saved: "Enregistré" | ||||
| messaging: "Discuter" | ||||
| upload: "Téléverser" | ||||
| @@ -264,6 +272,7 @@ rename: "Renommer" | ||||
| avatar: "Avatar" | ||||
| banner: "Bannière" | ||||
| nsfw: "Contenu sensible" | ||||
| whenServerDisconnected: "Lorsque la connexion au serveur est perdue" | ||||
| disconnectedFromServer: "Déconnecté·e du serveur" | ||||
| reload: "Rafraîchir" | ||||
| doNothing: "Ignorer" | ||||
| @@ -364,9 +373,6 @@ unregister: "Se désinscrire" | ||||
| passwordLessLogin: "Connectez-vous sans mot de passe" | ||||
| resetPassword: "Réinitialiser mot de passe" | ||||
| newPasswordIs: "Votre nouveau mot de passe est \"{password}\"" | ||||
| autoReloadWhenDisconnected: "Rechargement automatique lorsque le serveur se déconnecte" | ||||
| autoNoteWatch: "Surveiller les notes automatiquement" | ||||
| autoNoteWatchDescription: "Soyez informé des notes auxquelles vous avez réagi ou répondu." | ||||
| reduceUiAnimation: "Réduire les animations dans l’interface" | ||||
| share: "Partager" | ||||
| notFound: "Non trouvé" | ||||
| @@ -404,6 +410,7 @@ noMessagesYet: "Pas encore discuté" | ||||
| newMessageExists: "Vous avez un nouveau message" | ||||
| onlyOneFileCanBeAttached: "Vous ne pouvez joindre qu’un seul fichier au message" | ||||
| signinRequired: "Veuillez vous connecter" | ||||
| invitations: "Inviter" | ||||
| invitationCode: "Code d’invitation" | ||||
| checking: "Vérification" | ||||
| available: "Disponible" | ||||
| @@ -445,7 +452,7 @@ total: "Total" | ||||
| weekOverWeekChanges: "Diff hebdo" | ||||
| dayOverDayChanges: "Diff quotidien" | ||||
| appearance: "Aspect" | ||||
| clinetSettings: "Paramètres du client" | ||||
| clientSettings: "Paramètres du client" | ||||
| accountSettings: "Paramètres du compte" | ||||
| promotion: "Promu" | ||||
| promote: "Promouvoir" | ||||
| @@ -475,6 +482,7 @@ newNoteRecived: "Vous avez une nouvelle note" | ||||
| sounds: "Sons" | ||||
| listen: "Écouter" | ||||
| none: "Rien" | ||||
| popout: "Fenêtre contextuelle" | ||||
| volume: "Volume" | ||||
| details: "Détails" | ||||
| chooseEmoji: "Choisissez un émoji" | ||||
| @@ -517,7 +525,6 @@ enableInfiniteScroll: "Activer le défilement infini" | ||||
| visibility: "Visibilité" | ||||
| poll: "Sondage" | ||||
| useCw: "Masquer le contenu" | ||||
| fixedWidgetsPosition: "Rendre la position du widget fixe" | ||||
| enablePlayer: "Activer le lecteur vidéo" | ||||
| disablePlayer: "Désactiver le lecteur vidéo" | ||||
| expandTweet: "Étendre le tweet" | ||||
| @@ -539,13 +546,68 @@ tokenRequested: "Autoriser l'accès au compte" | ||||
| pluginTokenRequestedDescription: "Ce plugin pourra utiliser les autorisations définies ici." | ||||
| notificationType: "Type de notifications" | ||||
| edit: "Editer" | ||||
| useStarForReactionFallback: "Utiliser ★ comme alternative si l’émoji de réaction est inconnu" | ||||
| emailConfig: "Configuration du serveur email" | ||||
| enableEmail: "Activer la distribution de courriel" | ||||
| emailConfigInfo: "Utilisé pour confirmer votre adresse de courriel et la réinitialisation de votre mot de passe en cas d’oubli." | ||||
| email: "Adresse de courrier électronique" | ||||
| smtpConfig: "Paramètres du serveur SMTP" | ||||
| smtpHost: "Hôte" | ||||
| smtpPort: "Port" | ||||
| smtpUser: "Nom d’utilisateur·rice" | ||||
| smtpPass: "Mot de passe" | ||||
| emptyToDisableSmtpAuth: "Laisser le nom d’utilisateur et le mot de passe vides pour désactiver la vérification SMTP" | ||||
| smtpSecure: "Utiliser SSL/TLS implicitement dans les connexions SMTP" | ||||
| smtpSecureInfo: "Désactiver cette option lorsque STARTTLS est utilisé" | ||||
| testEmail: "Tester la distribution de courriel" | ||||
| wordMute: "Filtre de mots" | ||||
| userSaysSomething: "{name} a dit quelque chose" | ||||
| makeActive: "Activer" | ||||
| display: "Affichage" | ||||
| copy: "Copier" | ||||
| metrics: "Métriques" | ||||
| overview: "Aperçu" | ||||
| logs: "Journaux" | ||||
| delayed: "en retard" | ||||
| database: "Base de données" | ||||
| channel: "Canaux" | ||||
| create: "Créer" | ||||
| notificationSetting: "Paramètres des notifications " | ||||
| notificationSettingDesc: "Sélectionnez le type de notification à afficher" | ||||
| useGlobalSetting: "Utiliser paramètre général" | ||||
| other: "Autre" | ||||
| regenerateLoginToken: "Régénérer le jeton de connexion" | ||||
| setMultipleBySeparatingWithSpace: "Vous pouvez définir plus d’un, séparés par des espaces." | ||||
| fileIdOrUrl: "ID du fichier ou URL" | ||||
| chatOpenBehavior: "Comportement de la fenêtre de discussion lors de son ouverture" | ||||
| random: "Aléatoire" | ||||
| _mfm: | ||||
|   mention: "Mentionner" | ||||
|   hashtag: "Hashtags" | ||||
|   link: "Lien" | ||||
|   center: "Centrée" | ||||
|   quote: "Citer" | ||||
|   emoji: "Émojis personnalisés" | ||||
|   search: "Rechercher" | ||||
| _reversi: | ||||
|   total: "Total" | ||||
| _serverDisconnectedBehavior: | ||||
|   reload: "Rechargement automatique" | ||||
| _channel: | ||||
|   create: "Créer un canal" | ||||
|   edit: "Éditer le canal" | ||||
|   removeBanner: "Supprimer la bannière" | ||||
|   featured: "Tendances" | ||||
|   usersCount: "{n} Participants" | ||||
|   notesCount: "{n} Notes" | ||||
| _sidebar: | ||||
|   full: "Complet" | ||||
|   icon: "Avatar" | ||||
|   hide: "Masquer" | ||||
| _wordMute: | ||||
|   muteWords: "Mot à mettre en sourdine" | ||||
|   muteWordsDescription: "Séparer avec des espaces pour la condition AND. Séparer avec un saut de ligne pour une condition OR." | ||||
|   mutedNotes: "Notes mises en sourdine" | ||||
| _theme: | ||||
|   explore: "Explorer les thèmes" | ||||
|   install: "Installer un thème" | ||||
| @@ -556,6 +618,8 @@ _theme: | ||||
|   invalid: "Le format du thème n'est pas valide" | ||||
|   make: "Créer un thème" | ||||
|   base: "Base" | ||||
|   addConstant: "Ajouter une constante" | ||||
|   constant: "Constante" | ||||
|   defaultValue: "Valeur par défaut" | ||||
|   color: "Couleur" | ||||
|   key: "Clé " | ||||
| @@ -581,6 +645,7 @@ _theme: | ||||
|     renote: "Renote" | ||||
|     divider: "Séparateur" | ||||
|     infoWarnFg: "Texte d’avertissement" | ||||
|     cwBg: "Arrière-plan du CW" | ||||
|     badge: "Badge" | ||||
|     messageBg: "Arrière plan de la discussion" | ||||
| _sfx: | ||||
| @@ -665,6 +730,8 @@ _permissions: | ||||
|   "write:page-likes": "Mettre à jour les favoris sur les Pages" | ||||
|   "read:user-groups": "Voir les groupes d'utilisateur·rice·s" | ||||
|   "write:user-groups": "Éditer les groupes des utilisateur·rice·s" | ||||
|   "read:channels": "Lire les canaux" | ||||
|   "write:channels": "Modifier les canaux" | ||||
| _auth: | ||||
|   shareAccess: "Autoriser \"{name}\" à accéder à votre compte ?" | ||||
|   shareAccessAsk: "Voulez-vous vraiment autoriser cette application à accéder à votre compte?" | ||||
| @@ -698,6 +765,7 @@ _widgets: | ||||
|   photos: "Photos" | ||||
|   digitalClock: "Horloge numérique" | ||||
|   federation: "Fédération" | ||||
|   postForm: "Formulaire à publier" | ||||
| _cw: | ||||
|   hide: "Masquer" | ||||
|   show: "Afficher plus …" | ||||
| @@ -739,6 +807,7 @@ _visibility: | ||||
| _postForm: | ||||
|   replyPlaceholder: "Répondre à cette note ..." | ||||
|   quotePlaceholder: "Citez cette note ..." | ||||
|   channelPlaceholder: "Publier vers le canal" | ||||
|   _placeholders: | ||||
|     a: "Quoi de neuf ?" | ||||
|     b: "Quoi de neuf ?" | ||||
| @@ -871,6 +940,7 @@ _pages: | ||||
|   my: "Mes pages" | ||||
|   liked: "Pages favorites" | ||||
|   inspector: "Inspecteur" | ||||
|   contents: "Contenu" | ||||
|   content: "Bloc de page" | ||||
|   variables: "Variables" | ||||
|   title: "Titre" | ||||
| @@ -1152,15 +1222,22 @@ _notification: | ||||
|   yourFollowRequestAccepted: "Votre demande d’abonnement a été accepté" | ||||
|   youWereInvitedToGroup: "Invité au groupe" | ||||
|   _types: | ||||
|     all: "Toutes" | ||||
|     follow: "Abonnements" | ||||
|     mention: "Mentionner" | ||||
|     reply: "Réponses" | ||||
|     renote: "Renote" | ||||
|     quote: "Citer" | ||||
|     reaction: "Réactions" | ||||
|     groupInvited: "Invité aux groupes" | ||||
|     app: "Notifications provenant des apps" | ||||
| _deck: | ||||
|   alwaysShowMainColumn: "Toujours afficher la colonne principale" | ||||
|   columnAlign: "Aligner les colonnes" | ||||
|   addColumn: "Ajouter une colonne" | ||||
|   swapLeft: "Déplacer à gauche" | ||||
|   swapRight: "Déplacer à droite" | ||||
|   stackLeft: "Empiler à gauche" | ||||
|   _columns: | ||||
|     widgets: "Widgets" | ||||
|     notifications: "Notifications" | ||||
|   | ||||
| @@ -15,17 +15,24 @@ const merge = (...args) => args.reduce((a, c) => ({ | ||||
|  | ||||
| const languages = [ | ||||
| 	'ar-SA', | ||||
| 	//'cs-CZ', | ||||
| 	//'da-DK', | ||||
| 	'cs-CZ', | ||||
| 	'da-DK', | ||||
| 	'de-DE', | ||||
| 	'en-US', | ||||
| 	'es-ES', | ||||
| 	'fr-FR', | ||||
| 	'ja-JP', | ||||
| 	'ja-KS', | ||||
| 	'kab-KAB', | ||||
| 	'kn-IN', | ||||
| 	'ko-KR', | ||||
| 	//'nl-NL', | ||||
| 	//'pl-PL', | ||||
| 	'nl-NL', | ||||
| 	'no-NO', | ||||
| 	'pl-PL', | ||||
| 	'pt-PT', | ||||
| 	'ru-RU', | ||||
| 	'ug-CN', | ||||
| 	'uk-UA', | ||||
| 	'zh-CN', | ||||
| 	'zh-TW', | ||||
| ]; | ||||
|   | ||||
| @@ -16,6 +16,9 @@ noNotes: "ノートはありません" | ||||
| noNotifications: "通知はありません" | ||||
| instance: "インスタンス" | ||||
| settings: "設定" | ||||
| basicSettings: "基本設定" | ||||
| otherSettings: "その他の設定" | ||||
| openInWindow: "ウィンドウで開く" | ||||
| profile: "プロフィール" | ||||
| timeline: "タイムライン" | ||||
| noAccountDescription: "自己紹介はありません" | ||||
| @@ -40,6 +43,7 @@ deleteAndEditConfirm: "このノートを削除してもう一度編集します | ||||
| addToList: "リストに追加" | ||||
| sendMessage: "メッセージを送信" | ||||
| copyUsername: "ユーザー名をコピー" | ||||
| searchUser: "ユーザーを検索" | ||||
| reply: "返信" | ||||
| loadMore: "もっと見る" | ||||
| youGotNewFollower: "フォローされました" | ||||
| @@ -66,8 +70,11 @@ followers: "フォロワー" | ||||
| followsYou: "フォローされています" | ||||
| createList: "リスト作成" | ||||
| manageLists: "リストの管理" | ||||
| error: "問題が発生しました" | ||||
| error: "エラー" | ||||
| somethingHappened: "問題が発生しました" | ||||
| retry: "再試行" | ||||
| pageLoadError: "ページの読み込みに失敗しました。" | ||||
| pageLoadErrorDescription: "これは通常、ネットワークまたはブラウザキャッシュが原因です。キャッシュをクリアするか、しばらく待ってから再度試してください。" | ||||
| enterListName: "リスト名を入力" | ||||
| privacy: "プライバシー" | ||||
| makeFollowManuallyApprove: "フォローを承認制にする" | ||||
| @@ -88,6 +95,7 @@ sensitive: "閲覧注意" | ||||
| add: "追加" | ||||
| reaction: "リアクション" | ||||
| reactionSettingDescription: "リアクションピッカーに表示するリアクションを設定します。" | ||||
| reactionSettingDescription2: "ドラッグして並び替えます。クリックして削除します。" | ||||
| rememberNoteVisibility: "公開範囲を記憶する" | ||||
| attachCancel: "添付取り消し" | ||||
| markAsSensitive: "閲覧注意にする" | ||||
| @@ -106,6 +114,8 @@ unsuspendConfirm: "解凍しますか?" | ||||
| selectList: "リストを選択" | ||||
| selectAntenna: "アンテナを選択" | ||||
| selectWidget: "ウィジェットを選択" | ||||
| editWidgets: "ウィジェットを編集" | ||||
| editWidgetsExit: "編集を終了" | ||||
| customEmojis: "カスタム絵文字" | ||||
| emoji: "絵文字" | ||||
| emojiName: "絵文字名" | ||||
| @@ -177,7 +187,6 @@ processing: "処理中" | ||||
| preview: "プレビュー" | ||||
| default: "デフォルト" | ||||
| noCustomEmojis: "絵文字はありません" | ||||
| customEmojisOfRemote: "リモートの絵文字" | ||||
| noJobs: "ジョブはありません" | ||||
| federating: "連合中" | ||||
| blocked: "ブロック中" | ||||
| @@ -206,6 +215,8 @@ imageUrl: "画像URL" | ||||
| remove: "削除" | ||||
| removed: "削除しました" | ||||
| removeAreYouSure: "「{x}」を削除しますか?" | ||||
| deleteAreYouSure: "「{x}」を削除しますか?" | ||||
| resetAreYouSure: "リセットしますか?" | ||||
| saved: "保存しました" | ||||
| messaging: "チャット" | ||||
| upload: "アップロード" | ||||
| @@ -264,6 +275,7 @@ rename: "名前を変更" | ||||
| avatar: "アイコン" | ||||
| banner: "バナー" | ||||
| nsfw: "閲覧注意" | ||||
| whenServerDisconnected: "サーバーとの接続が失われたとき" | ||||
| disconnectedFromServer: "サーバーから切断されました" | ||||
| reload: "リロード" | ||||
| doNothing: "なにもしない" | ||||
| @@ -364,9 +376,6 @@ unregister: "登録を解除" | ||||
| passwordLessLogin: "パスワード無しログイン" | ||||
| resetPassword: "パスワードをリセット" | ||||
| newPasswordIs: "新しいパスワードは「{password}」です" | ||||
| autoReloadWhenDisconnected: "サーバー切断時に自動リロード" | ||||
| autoNoteWatch: "ノートの自動ウォッチ" | ||||
| autoNoteWatchDescription: "あなたがリアクションしたり返信したりした他のユーザーのノートに関する通知を受け取るようにします。" | ||||
| reduceUiAnimation: "UIのアニメーションを減らす" | ||||
| share: "共有" | ||||
| notFound: "見つかりません" | ||||
| @@ -404,6 +413,7 @@ noMessagesYet: "まだチャットはありません" | ||||
| newMessageExists: "新しいメッセージがあります" | ||||
| onlyOneFileCanBeAttached: "メッセージに添付できるファイルはひとつです" | ||||
| signinRequired: "ログインしてください" | ||||
| invitations: "招待" | ||||
| invitationCode: "招待コード" | ||||
| checking: "確認しています" | ||||
| available: "利用できます" | ||||
| @@ -445,7 +455,7 @@ total: "合計" | ||||
| weekOverWeekChanges: "前週比" | ||||
| dayOverDayChanges: "前日比" | ||||
| appearance: "アピアランス" | ||||
| clinetSettings: "クライアント設定" | ||||
| clientSettings: "クライアント設定" | ||||
| accountSettings: "アカウント設定" | ||||
| promotion: "プロモーション" | ||||
| promote: "プロモート" | ||||
| @@ -468,6 +478,7 @@ objectStorageUseSSL: "SSLを使用する" | ||||
| objectStorageUseSSLDesc: "API接続にhttpsを使用しない場合はオフにしてください" | ||||
| objectStorageUseProxy: "Proxyを利用する" | ||||
| objectStorageUseProxyDesc: "API接続にproxyを利用しない場合はオフにしてください" | ||||
| objectStorageSetPublicRead: "アップロード時に'public-read'を設定する" | ||||
| serverLogs: "サーバーログ" | ||||
| deleteAll: "全て削除" | ||||
| showFixedPostForm: "タイムライン上部に投稿フォームを表示する" | ||||
| @@ -475,6 +486,8 @@ newNoteRecived: "新しいノートがあります" | ||||
| sounds: "サウンド" | ||||
| listen: "聴く" | ||||
| none: "なし" | ||||
| showInPage: "ページで表示" | ||||
| popout: "ポップアウト" | ||||
| volume: "音量" | ||||
| details: "詳細" | ||||
| chooseEmoji: "絵文字を選択" | ||||
| @@ -517,7 +530,6 @@ enableInfiniteScroll: "自動でもっと見る" | ||||
| visibility: "公開範囲" | ||||
| poll: "アンケート" | ||||
| useCw: "内容を隠す" | ||||
| fixedWidgetsPosition: "ウィジェットの位置を固定する" | ||||
| enablePlayer: "プレイヤーを開く" | ||||
| disablePlayer: "プレイヤーを閉じる" | ||||
| expandTweet: "ツイートを展開する" | ||||
| @@ -531,6 +543,7 @@ pluginInstallWarn: "信頼できないプラグインはインストールしな | ||||
| deck: "デッキ" | ||||
| undeck: "デッキ解除" | ||||
| useBlurEffectForModal: "モーダルにぼかし効果を使用" | ||||
| useFullReactionPicker: "フル機能リアクションピッカーを使用" | ||||
| generateAccessToken: "アクセストークンの発行" | ||||
| permission: "権限" | ||||
| enableAll: "全て有効にする" | ||||
| @@ -557,6 +570,149 @@ wordMute: "ワードミュート" | ||||
| userSaysSomething: "{name}が何かを言いました" | ||||
| makeActive: "アクティブにする" | ||||
| display: "表示" | ||||
| copy: "コピー" | ||||
| metrics: "メトリクス" | ||||
| overview: "概要" | ||||
| logs: "ログ" | ||||
| delayed: "遅延" | ||||
| database: "データベース" | ||||
| channel: "チャンネル" | ||||
| create: "作成" | ||||
| notificationSetting: "通知設定" | ||||
| notificationSettingDesc: "表示する通知の種別を選択してください。" | ||||
| useGlobalSetting: "グローバル設定を使う" | ||||
| useGlobalSettingDesc: "オンにすると、アカウントの通知設定が使用されます。オフにすると、個別に設定できるようになります。" | ||||
| other: "その他" | ||||
| regenerateLoginToken: "ログイントークンを再生成" | ||||
| regenerateLoginTokenDescription: "ログインに使用される内部トークンを再生成します。通常この操作を行う必要はありません。再生成すると、全てのデバイスでログアウトされます。" | ||||
| setMultipleBySeparatingWithSpace: "スペースで区切って複数設定できます。" | ||||
| fileIdOrUrl: "ファイルIDまたはURL" | ||||
| chatOpenBehavior: "チャットを開くときの動作" | ||||
| sample: "サンプル" | ||||
| abuseReports: "通報" | ||||
| reportAbuse: "通報" | ||||
| reportAbuseOf: "{name}を通報する" | ||||
| fillAbuseReportDescription: "通報理由の詳細を記入してください。対象のノートがある場合はそのURLも記入してください。" | ||||
| abuseReported: "内容が送信されました。ご報告ありがとうございました。" | ||||
| send: "送信" | ||||
| abuseMarkAsResolved: "対応済みにする" | ||||
| openInNewTab: "新しいタブで開く" | ||||
| openInSideView: "サイドビューで開く" | ||||
| defaultNavigationBehaviour: "デフォルトのナビゲーション" | ||||
| editTheseSettingsMayBreakAccount: "これらの設定を編集するとアカウントが破損する可能性があります。" | ||||
| instanceTicker: "ノートのインスタンス情報" | ||||
| waitingFor: "{x}を待っています" | ||||
| random: "ランダム" | ||||
| system: "システム" | ||||
| switchUi: "UI切り替え" | ||||
| desktop: "デスクトップ" | ||||
| clip: "クリップ" | ||||
| createNew: "新規作成" | ||||
| optional: "任意" | ||||
|  | ||||
| _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: "回転するアニメーションを与えます。" | ||||
|  | ||||
| _reversi: | ||||
|   reversi: "リバーシ" | ||||
|   gameSettings: "対局の設定" | ||||
|   chooseBoard: "ボードを選択" | ||||
|   blackOrWhite: "先行/後攻" | ||||
|   blackIs: "{name}が黒(先行)" | ||||
|   rules: "ルール" | ||||
|   botSettings: "Botのオプション" | ||||
|   thisGameIsStartedSoon: "対局は数秒後に開始されます" | ||||
|   waitingForOther: "相手の準備が完了するのを待っています" | ||||
|   waitingForMe: "あなたの準備が完了するのを待っています" | ||||
|   waitingBoth: "準備してください" | ||||
|   ready: "準備完了" | ||||
|   cancelReady: "準備を再開" | ||||
|   opponentTurn: "相手のターンです" | ||||
|   myTurn: "あなたのターンです" | ||||
|   turnOf: "{name}のターンです" | ||||
|   pastTurnOf: "{name}のターン" | ||||
|   surrender: "投了" | ||||
|   surrendered: "投了により" | ||||
|   drawn: "引き分け" | ||||
|   won: "{name}の勝ち" | ||||
|   black: "黒" | ||||
|   white: "白" | ||||
|   total: "合計" | ||||
|   turnCount: "{count}ターン目" | ||||
|   myGames: "自分の対局" | ||||
|   allGames: "みんなの対局" | ||||
|   ended: "終了" | ||||
|   playing: "対局中" | ||||
|   isLlotheo: "石の少ない方が勝ち(ロセオ)" | ||||
|   loopedMap: "ループマップ" | ||||
|   canPutEverywhere: "どこでも置けるモード" | ||||
|  | ||||
| _instanceTicker: | ||||
|   none: "表示しない" | ||||
|   remote: "リモートユーザーに表示" | ||||
|   always: "常に表示" | ||||
|  | ||||
| _serverDisconnectedBehavior: | ||||
|   reload: "自動でリロード" | ||||
|   dialog: "ダイアログで警告" | ||||
|   quiet: "控えめに警告" | ||||
|  | ||||
| _channel: | ||||
|   create: "チャンネルを作成" | ||||
|   edit: "チャンネルを編集" | ||||
|   setBanner: "バナーを設定" | ||||
|   removeBanner: "バナーを削除" | ||||
|   featured: "トレンド" | ||||
|   owned: "管理中" | ||||
|   following: "フォロー中" | ||||
|   usersCount: "{n}人が参加中" | ||||
|   notesCount: "{n}投稿があります" | ||||
|  | ||||
| _sidebar: | ||||
|   full: "フル" | ||||
| @@ -571,6 +727,7 @@ _wordMute: | ||||
|   hardDescription: "指定した条件のノートをタイムラインに追加しないようにします。追加されなかったノートは、条件を変更しても除外されたままになります。" | ||||
|   soft: "ソフト" | ||||
|   hard: "ハード" | ||||
|   mutedNotes: "ミュートされたノート" | ||||
|  | ||||
| _theme: | ||||
|   explore: "テーマを探す" | ||||
| @@ -652,6 +809,7 @@ _sfx: | ||||
|   chat: "チャット" | ||||
|   chatBg: "チャット(バックグラウンド)" | ||||
|   antenna: "アンテナ受信" | ||||
|   channel: "チャンネル通知" | ||||
|  | ||||
| _ago: | ||||
|   unknown: "謎" | ||||
| @@ -732,6 +890,8 @@ _permissions: | ||||
|   "write:page-likes": "ページのいいねを操作する" | ||||
|   "read:user-groups": "ユーザーグループを見る" | ||||
|   "write:user-groups": "ユーザーグループを操作する" | ||||
|   "read:channels": "チャンネルを見る" | ||||
|   "write:channels": "チャンネルを操作する" | ||||
|  | ||||
| _auth: | ||||
|   shareAccess: "「{name}」がアカウントにアクセスすることを許可しますか?" | ||||
| @@ -769,6 +929,7 @@ _widgets: | ||||
|   photos: "フォト" | ||||
|   digitalClock: "デジタル時計" | ||||
|   federation: "連合" | ||||
|   postForm: "投稿フォーム" | ||||
|  | ||||
| _cw: | ||||
|   hide: "隠す" | ||||
| @@ -814,6 +975,7 @@ _visibility: | ||||
| _postForm: | ||||
|   replyPlaceholder: "このノートに返信..." | ||||
|   quotePlaceholder: "このノートを引用..." | ||||
|   channelPlaceholder: "チャンネルに投稿..." | ||||
|   _placeholders: | ||||
|     a: "いまどうしてる?" | ||||
|     b: "何かありましたか?" | ||||
| @@ -942,6 +1104,7 @@ _pages: | ||||
|   created: "ページを作成しました" | ||||
|   updated: "ページを更新しました" | ||||
|   deleted: "ページを削除しました" | ||||
|   pageSetting: "ページ設定" | ||||
|   nameAlreadyExists: "指定されたページURLは既に存在しています" | ||||
|   invalidNameTitle: "不正なページURLです" | ||||
|   invalidNameText: "空白でないか確認してください" | ||||
| @@ -953,6 +1116,7 @@ _pages: | ||||
|   my: "自分のページ" | ||||
|   liked: "いいねしたページ" | ||||
|   inspector: "インスペクター" | ||||
|   contents: "コンテンツ" | ||||
|   content: "ページブロック" | ||||
|   variables: "変数" | ||||
|   title: "タイトル" | ||||
| @@ -1013,6 +1177,12 @@ _pages: | ||||
|       width: "幅" | ||||
|       height: "高さ" | ||||
|  | ||||
|     note: "ノート埋め込み" | ||||
|     _note: | ||||
|       id: "ノートID" | ||||
|       idDescription: "ノートURLをペーストして設定することもできます。" | ||||
|       detailed: "詳細な表示" | ||||
|  | ||||
|     switch: "スイッチ" | ||||
|     _switch: | ||||
|       name: "変数名" | ||||
| @@ -1255,8 +1425,11 @@ _notification: | ||||
|     renote: "Renote" | ||||
|     quote: "引用" | ||||
|     reaction: "リアクション" | ||||
|     pollVote: "投票" | ||||
|     receiveFollowRequest: "フォローリクエスト" | ||||
|     pollVote: "アンケートに投票された" | ||||
|     receiveFollowRequest: "フォロー申請を受け取った" | ||||
|     followRequestAccepted: "フォローが受理された" | ||||
|     groupInvited: "グループに招待された" | ||||
|     app: "連携アプリからの通知" | ||||
|  | ||||
| _deck: | ||||
|   alwaysShowMainColumn: "常にメインカラムを表示" | ||||
|   | ||||
| @@ -1,12 +1,12 @@ | ||||
| --- | ||||
| _lang_: "日本語 (関西弁)" | ||||
| introMisskey: "ようこそ!Misskeyは、オープンソースの分散型マイクロブログサービスやねん。\n「ノート」を作成しぃ、いま起こっとることを共有したり、あんたについて皆に発信しよう📡\n「リアクション」機能で、皆のノートに素はよ反応を追加することもできます✌\n新しい世界を探検しよう🚀" | ||||
| introMisskey: "ようこそ!Misskeyってのは、オープンソースの分散型マイクロブログサービスやねん。\n「ノート」を作成し、いま起こっとることを共有したり、あんたんこととか皆に伝えていこう📡\n「リアクション」機能で、皆のノートに素はよ反応を追加することもできるんやで✌\n新しい世界を探検してみらん?🚀" | ||||
| monthAndDay: "{month}月 {day}日" | ||||
| search: "探す" | ||||
| notifications: "通知" | ||||
| username: "ユーザー名" | ||||
| password: "パスワード" | ||||
| fetchingAsApObject: "連合に照会中" | ||||
| fetchingAsApObject: "今ちと連合に照会しとるで" | ||||
| ok: "おっけー" | ||||
| gotIt: "ほい" | ||||
| cancel: "やめとくわ" | ||||
| @@ -16,35 +16,40 @@ noNotes: "ノートはあらへん" | ||||
| noNotifications: "通知はあらへん" | ||||
| instance: "インスタンス" | ||||
| settings: "設定" | ||||
| basicSettings: "基本設定" | ||||
| otherSettings: "その他の設定" | ||||
| openInWindow: "ウィンドウで開いてや" | ||||
| profile: "プロフィール" | ||||
| timeline: "タイムライン" | ||||
| noAccountDescription: "自己紹介はあらへん" | ||||
| login: "ログイン" | ||||
| loggingIn: "ログインしとります" | ||||
| loggingIn: "ログインしよるで" | ||||
| logout: "ログアウト" | ||||
| signup: "新規登録" | ||||
| uploading: "アップロードしとります" | ||||
| save: "保存" | ||||
| uploading: "アップロードしよるで" | ||||
| save: "とっとく" | ||||
| users: "ユーザー" | ||||
| addUser: "ユーザー増やす" | ||||
| addUser: "ユーザーを追加や" | ||||
| favorite: "お気に入り" | ||||
| favorites: "お気に入り" | ||||
| unfavorite: "お気に入りやめる" | ||||
| pin: "ピン留め" | ||||
| unpin: "ピン留めやめる" | ||||
| unfavorite: "やっぱ気に入らん" | ||||
| pin: "ピン留めしとく" | ||||
| unpin: "やっぱピン留めせん" | ||||
| copyContent: "内容をコピー" | ||||
| copyLink: "リンクをコピー" | ||||
| delete: "ほかす" | ||||
| deleteAndEdit: "ほかして直す" | ||||
| deleteAndEditConfirm: "このノートをほかしてもっかい直す?このノートへのリアクション、Remote、返信も全部消えんで" | ||||
| deleteAndEditConfirm: "このノートをほかしてもっかい直す?このノートへのリアクション、Renote、返信も全部消えるんやけどそれでもええん?" | ||||
| addToList: "リストに入れたる" | ||||
| sendMessage: "メッセージを送る" | ||||
| copyUsername: "ユーザー名をコピー" | ||||
| searchUser: "ユーザーを検索" | ||||
| reply: "返す" | ||||
| loadMore: "もっとあるやろ!" | ||||
| youGotNewFollower: "フォローされたで" | ||||
| receiveFollowRequest: "フォローリクエストされたで" | ||||
| followRequestAccepted: "フォローが承認されたで" | ||||
| mention: "メンション" | ||||
| mentions: "あんた宛て" | ||||
| directNotes: "ダイレクト投稿" | ||||
| importAndExport: "インポートとエクスポート" | ||||
| @@ -57,7 +62,7 @@ unfollowConfirm: "{name}のフォローを解除してもええんか?" | ||||
| exportRequested: "エクスポートしてな、ってリクエストしたけど、これ多分めっちゃ時間かかるで。エクスポート終わったら「ドライブ」に突っ込んどくで。" | ||||
| importRequested: "インポートしてな、ってリクエストしたけど、これ多分めっちゃ時間かかるで。" | ||||
| lists: "リスト" | ||||
| noLists: "リストはあらへん" | ||||
| noLists: "リストなんてあらへんで" | ||||
| note: "ノート" | ||||
| notes: "ノート" | ||||
| following: "フォロー" | ||||
| @@ -65,32 +70,35 @@ followers: "フォロワー" | ||||
| followsYou: "フォローされとるで" | ||||
| createList: "リスト作る" | ||||
| manageLists: "リストの管理" | ||||
| error: "問題が発生してん" | ||||
| retry: "もっぺんやってみる" | ||||
| error: "エラー" | ||||
| somethingHappened: "なんかアカンことが起こったで" | ||||
| retry: "もっぺんやる?" | ||||
| pageLoadError: "ページの読み込みに失敗してしもうたで…" | ||||
| pageLoadErrorDescription: "これは普通、ネットワークかブラウザキャッシュが原因やからね。キャッシュをクリアするか、もうちっとだけ待ってくれへんか?" | ||||
| enterListName: "リスト名を入れてや" | ||||
| privacy: "プライバシーってなんや?オカンの年齢か?" | ||||
| makeFollowManuallyApprove: "他人のフォローは許可してからや!" | ||||
| privacy: "プライバシーってなんぞや?" | ||||
| makeFollowManuallyApprove: "他人からのフォローは自分が決める" | ||||
| defaultNoteVisibility: "もとからの公開範囲" | ||||
| follow: "フォロー" | ||||
| followRequest: "フォロー許してくれや!言うてみる" | ||||
| followRequests: "フォロー許してくれや!" | ||||
| followRequest: "フォローを頼む" | ||||
| followRequests: "フォローを頼む" | ||||
| unfollow: "フォローやめる" | ||||
| followRequestPending: "フォロー許してくれるん待っとる" | ||||
| enterEmoji: "絵文字を入れてや" | ||||
| renote: "Renote" | ||||
| unrenote: "Renoteやめる" | ||||
| quote: "引用" | ||||
| pinnedNote: "ピン留めされたノート" | ||||
| pinnedNote: "ピン留めされとるノート" | ||||
| you: "あんた" | ||||
| clickToShow: "押してみ、見せたるわ" | ||||
| sensitive: "見たらあかんで" | ||||
| clickToShow: "押したら見えるようになるで" | ||||
| sensitive: "ちょっとアカンやつやで" | ||||
| add: "増やす" | ||||
| reaction: "リアクション" | ||||
| reactionSettingDescription: "リアクションピッカーに出しとくリアクションを選んでや。" | ||||
| rememberNoteVisibility: "公開範囲覚えといて" | ||||
| attachCancel: "くっつけるのやめよか" | ||||
| markAsSensitive: "ちょっと見せられへんわ" | ||||
| unmarkAsSensitive: "別にええんじゃね?" | ||||
| attachCancel: "やっぱ添付やめてくれん?" | ||||
| markAsSensitive: "ちょっとこれはアカン" | ||||
| unmarkAsSensitive: "そこまでアカンことないやろ" | ||||
| enterFileName: "ファイル名を入れてや" | ||||
| mute: "ミュート" | ||||
| unmute: "ミュートやめたる" | ||||
| @@ -98,30 +106,35 @@ block: "ブロック" | ||||
| unblock: "ブロックやめたる" | ||||
| suspend: "凍結" | ||||
| unsuspend: "溶かす" | ||||
| blockConfirm: "ブロックしてしもうてええか?" | ||||
| unblockConfirm: "ブロックすんのやめるけどええか?" | ||||
| blockConfirm: "ブロックしてもええんか?" | ||||
| unblockConfirm: "ブロックやめたるってほんまか?" | ||||
| suspendConfirm: "凍結してしもうてええか?" | ||||
| unsuspendConfirm: "解凍するけどええか?" | ||||
| selectList: "リストを選ぶ" | ||||
| selectAntenna: "アンテナを選ぶ" | ||||
| selectWidget: "ウィジェットを選ぶ" | ||||
| editWidgets: "ウィジェットをいじる" | ||||
| editWidgetsExit: "編集終ったで" | ||||
| customEmojis: "カスタム絵文字" | ||||
| emoji: "絵文字" | ||||
| emojiName: "絵文字名" | ||||
| emojiUrl: "絵文字画像URL" | ||||
| addEmoji: "絵文字を追加" | ||||
| settingGuide: "ええ感じの設定" | ||||
| cacheRemoteFiles: "リモートのファイルをキャッシュする" | ||||
| cacheRemoteFilesDescription: "この設定をチャラにすると、リモートファイルをキャッシュせず直リンクするようになります。サーバーのストレージを節約できますが、サムネイルが生成されへんので通信量が増加します。" | ||||
| flagAsBot: "Botやでと言っとく" | ||||
| flagAsCat: "Catやでと言っとく" | ||||
| autoAcceptFollowed: "フォローしとるユーザーからのフォロリクは全部勝手にええでって言うで" | ||||
| cacheRemoteFilesDescription: "この設定を切っとくと、リモートファイルをキャッシュせず直リンクするようになってしまうんやで? サーバーのストレージは節約できるんやけど、かわりにサムネイルが作られんくなるから通信量が増えるで?" | ||||
| flagAsBot: "Botやで" | ||||
| flagAsCat: "Catやで" | ||||
| autoAcceptFollowed: "フォローしとるユーザーからのフォローリクエストには勝手に許可しとくで。" | ||||
| addAcount: "アカウント追加" | ||||
| loginFailed: "ログインに失敗してん" | ||||
| loginFailed: "ログインに失敗してしもうた…" | ||||
| showOnRemote: "リモートで見る" | ||||
| general: "全般" | ||||
| wallpaper: "壁紙" | ||||
| setWallpaper: "壁紙を設定" | ||||
| removeWallpaper: "壁紙ほかす" | ||||
| removeWallpaper: "壁紙を削除" | ||||
| searchWith: "検索: {q}" | ||||
| youHaveNoLists: "リストはあらへん" | ||||
| youHaveNoLists: "リストがあらへんで?" | ||||
| followConfirm: "{name}をフォローしてええか?" | ||||
| proxyAccount: "プロキシアカウント" | ||||
| proxyAccountDescription: "プロキシアカウントは、代わりにフォローしてくれるアカウントや。例えば、551に豚まんが無いときやったり、ユーザーがリモートユーザーをアカウントに入れたとき、リストに入れられたユーザーが誰からもフォローされてないと寂しいやん。寂しいし、アクティビティも配達されへんから、プロキシアカウントがフォローしてくれるで。ええやつやん…" | ||||
| @@ -131,7 +144,7 @@ recipient: "宛先" | ||||
| annotation: "注釈" | ||||
| federation: "連合" | ||||
| instances: "インスタンス" | ||||
| registeredAt: "一見さんになった日" | ||||
| registeredAt: "初観測" | ||||
| latestRequestSentAt: "ちょっと前のリクエスト送信" | ||||
| latestRequestReceivedAt: "ちょっと前のリクエスト受信" | ||||
| latestStatus: "ちょっと前のステータス" | ||||
| @@ -173,7 +186,6 @@ processing: "処理しとる" | ||||
| preview: "プレビュー" | ||||
| default: "デフォルト" | ||||
| noCustomEmojis: "絵文字はあらへん" | ||||
| customEmojisOfRemote: "リモートの絵文字" | ||||
| noJobs: "ジョブはあらへん" | ||||
| federating: "連合しとる" | ||||
| blocked: "ブロックしとる" | ||||
| @@ -202,6 +214,7 @@ imageUrl: "画像URL" | ||||
| remove: "ほかす" | ||||
| removed: "削除したで!" | ||||
| removeAreYouSure: "「{x}」はなおしてしもてええか?" | ||||
| deleteAreYouSure: "「{x}」はなおしてしもてええか?" | ||||
| saved: "保存したで!" | ||||
| messaging: "チャット" | ||||
| upload: "アップロード" | ||||
| @@ -259,7 +272,8 @@ copyUrl: "URLをコピー" | ||||
| rename: "名前を変えるで" | ||||
| avatar: "アイコン" | ||||
| banner: "バナー" | ||||
| nsfw: "見たらあかんで" | ||||
| nsfw: "ちょっとアカンやつやで" | ||||
| whenServerDisconnected: "サーバーとの接続が失くなってしもうたとき" | ||||
| disconnectedFromServer: "サーバーが機嫌悪いねん" | ||||
| reload: "リロード" | ||||
| doNothing: "何もせんとく" | ||||
| @@ -295,7 +309,19 @@ proxyRemoteFilesDescription: "この設定を入れると、保存しとらん | ||||
| driveCapacityPerLocalAccount: "ローカルユーザーひとりあたりのドライブ容量" | ||||
| driveCapacityPerRemoteAccount: "リモートユーザーひとりあたりのドライブ容量" | ||||
| inMb: "メガバイト単位" | ||||
| iconUrl: "アイコン画像のURL" | ||||
| bannerUrl: "バナー画像のURL" | ||||
| basicInfo: "基本情報" | ||||
| pinnedUsers: "ピン留めしたユーザー" | ||||
| pinnedUsersDescription: "「みつける」ページとかにピン留めしたいユーザーをここに書けばええんやで。他ん人との名前は改行で区切ればええんやで。" | ||||
| hcaptcha: "hCaptcha(キャプチャ)" | ||||
| enableHcaptcha: "hCaptcha(キャプチャ)をつけとく" | ||||
| hcaptchaSiteKey: "サイトキー" | ||||
| hcaptchaSecretKey: "シークレットキー" | ||||
| recaptcha: "reCAPTCHA" | ||||
| enableRecaptcha: "reCAPTCHA(リキャプチャ)を有効にする" | ||||
| recaptchaSiteKey: "サイトキー" | ||||
| recaptchaSecretKey: "シークレットキー" | ||||
| avoidMultiCaptchaConfirm: "ぎょうさんのCaptchaをつこてしまうと、仲良うせんことがあるんや。他のCaptchaをなおしとこか?別にキャンセルしてもろうたらCaptchaは消されへんで済むけど知らんで。" | ||||
| antennas: "アンテナ" | ||||
| manageAntennas: "アンテナいじる" | ||||
| @@ -348,18 +374,58 @@ unregister: "登録やめる" | ||||
| passwordLessLogin: "パスワード無くてもログインできるようにする" | ||||
| resetPassword: "パスワードをリセット" | ||||
| newPasswordIs: "今度のパスワードは「{password}」や" | ||||
| autoReloadWhenDisconnected: "サーバーが調子悪いときには自動でリロードしたる" | ||||
| reduceUiAnimation: "UIの動きやアニメーションを減らしてくれや。" | ||||
| share: "わけわけ" | ||||
| notFound: "見つからへんね" | ||||
| notFoundDescription: "指定されたURLに該当するページはあらへんやった。" | ||||
| uploadFolder: "とりあえずここへアップロード" | ||||
| cacheClear: "キャッシュをほかす" | ||||
| markAsReadAllNotifications: "通知はもう全て読んだわっ" | ||||
| markAsReadAllUnreadNotes: "投稿は全て読んだわっ" | ||||
| markAsReadAllTalkMessages: "チャットはもうぜんぶ読んだわっ" | ||||
| help: "ヘルプ" | ||||
| inputMessageHere: "ここにメッセージ書いてや" | ||||
| close: "さいなら" | ||||
| group: "グループ" | ||||
| groups: "グループ" | ||||
| createGroup: "グループを作るで" | ||||
| ownedGroups: "所有しとるグループ" | ||||
| joinedGroups: "参加しとるグループ" | ||||
| invites: "来てや" | ||||
| groupName: "グループ名" | ||||
| members: "メンバー" | ||||
| transfer: "譲渡" | ||||
| messagingWithUser: "ユーザーとチャット" | ||||
| messagingWithGroup: "グループでチャット" | ||||
| title: "タイトル" | ||||
| text: "テキスト" | ||||
| enable: "有効にするで" | ||||
| next: "次" | ||||
| retype: "もっかい入力" | ||||
| noteOf: "{user}のノート" | ||||
| inviteToGroup: "グループに招く" | ||||
| maxNoteTextLength: "ノートの文字数制限" | ||||
| quoteAttached: "引用付いとるで" | ||||
| quoteQuestion: "引用として添付してもええか?" | ||||
| noMessagesYet: "まだチャットはあらへんで" | ||||
| newMessageExists: "新しいメッセージがきたで" | ||||
| onlyOneFileCanBeAttached: "すまん、メッセージに添付できるファイルはひとつだけなんや。" | ||||
| invitations: "来てや" | ||||
| invitationCode: "招待コード" | ||||
| checking: "確認しとるで" | ||||
| smtpHost: "ホスト" | ||||
| smtpUser: "ユーザー名" | ||||
| smtpPass: "パスワード" | ||||
| _mfm: | ||||
|   mention: "メンション" | ||||
|   quote: "引用" | ||||
|   emoji: "カスタム絵文字" | ||||
|   search: "探す" | ||||
| _sidebar: | ||||
|   icon: "アイコン" | ||||
| _theme: | ||||
|   keys: | ||||
|     mention: "メンション" | ||||
|     renote: "Renote" | ||||
| _sfx: | ||||
|   note: "ノート" | ||||
| @@ -443,6 +509,7 @@ _notification: | ||||
|   youWereFollowed: "フォローされたで" | ||||
|   _types: | ||||
|     follow: "フォロー" | ||||
|     mention: "メンション" | ||||
|     renote: "Renote" | ||||
|     quote: "引用" | ||||
|     reaction: "リアクション" | ||||
|   | ||||
| @@ -35,6 +35,9 @@ userList: "Tibdarin" | ||||
| uiLanguage: "Tutlayt n wegrudem" | ||||
| smtpUser: "Isem n umseqdac" | ||||
| smtpPass: "Awal uffir" | ||||
| _mfm: | ||||
|   mention: "Bder" | ||||
|   search: "Nadi" | ||||
| _theme: | ||||
|   keys: | ||||
|     mention: "Bder" | ||||
| @@ -54,6 +57,7 @@ _exportOrImport: | ||||
|   blockingList: "Seḥbes" | ||||
|   userLists: "Tibdarin" | ||||
| _pages: | ||||
|   contents: "Agbur" | ||||
|   font: "Tasefsit" | ||||
|   fontSerif: "Serif" | ||||
|   fontSansSerif: "Sans Serif" | ||||
|   | ||||
| @@ -56,6 +56,8 @@ instances: "ನಿದರ್ಶನ" | ||||
| remove: "ಅಳಿಸು" | ||||
| smtpUser: "ಬಳಕೆಹೆಸರು" | ||||
| smtpPass: "ಗುಪ್ತಪದ" | ||||
| _mfm: | ||||
|   search: "ಹುಡುಕು" | ||||
| _sfx: | ||||
|   notification: "ಅಧಿಸೂಚನೆಗಳು" | ||||
| _widgets: | ||||
|   | ||||
| @@ -66,7 +66,6 @@ followers: "팔로워" | ||||
| followsYou: "당신을 팔로우합니다" | ||||
| createList: "리스트 만들기" | ||||
| manageLists: "리스트 관리" | ||||
| error: "오류가 발생했습니다" | ||||
| retry: "다시 시도" | ||||
| enterListName: "리스트 이름을 입력" | ||||
| privacy: "프라이버시" | ||||
| @@ -177,7 +176,6 @@ processing: "처리중" | ||||
| preview: "미리보기" | ||||
| default: "기본값" | ||||
| noCustomEmojis: "이모지가 없습니다" | ||||
| customEmojisOfRemote: "다른 인스턴스들의 이모지" | ||||
| noJobs: "작업이 없습니다" | ||||
| federating: "연합 중" | ||||
| blocked: "차단됨" | ||||
| @@ -206,6 +204,7 @@ imageUrl: "이미지 URL" | ||||
| remove: "삭제" | ||||
| removed: "삭제하였습니다" | ||||
| removeAreYouSure: "\"{x}\" 을(를) 삭제하시겠습니까?" | ||||
| deleteAreYouSure: "\"{x}\" 을(를) 삭제하시겠습니까?" | ||||
| saved: "저장하였습니다" | ||||
| messaging: "대화" | ||||
| upload: "업로드" | ||||
| @@ -364,9 +363,6 @@ unregister: "등록 해제" | ||||
| passwordLessLogin: "비밀번호 없이 로그인" | ||||
| resetPassword: "비밀번호 재설정" | ||||
| newPasswordIs: "새로운 비밀번호는 \"{password}\" 입니다" | ||||
| autoReloadWhenDisconnected: "서버와의 연결이 끊기면 자동 새로고침" | ||||
| autoNoteWatch: "노트를 자동으로 지켜보기" | ||||
| autoNoteWatchDescription: "리액션하거나 답글을 남긴 다른 유저의 노트에 대한 알림을 받습니다." | ||||
| reduceUiAnimation: "UI의 애니메이션을 줄이기" | ||||
| share: "공유" | ||||
| notFound: "찾을 수 없습니다" | ||||
| @@ -404,6 +400,7 @@ noMessagesYet: "아직 대화가 없습니다" | ||||
| newMessageExists: "새 메시지가 있습니다" | ||||
| onlyOneFileCanBeAttached: "메시지에 첨부할 수 있는 파일은 하나까지입니다" | ||||
| signinRequired: "로그인 해주세요" | ||||
| invitations: "초대" | ||||
| invitationCode: "초대 코드" | ||||
| checking: "확인하는 중입니다" | ||||
| available: "사용 가능합니다" | ||||
| @@ -444,7 +441,6 @@ remote: "리모트" | ||||
| total: "합계" | ||||
| weekOverWeekChanges: "지난주보다" | ||||
| dayOverDayChanges: "어제보다" | ||||
| clinetSettings: "클라이언트 설정" | ||||
| accountSettings: "계정 설정" | ||||
| promotion: "프로모션" | ||||
| promote: "프로모션하기" | ||||
| @@ -516,7 +512,6 @@ enableInfiniteScroll: "자동으로 좀 더 보기" | ||||
| visibility: "공개 범위" | ||||
| poll: "투표" | ||||
| useCw: "내용 숨기기" | ||||
| fixedWidgetsPosition: "위젯의 위치 고정" | ||||
| enablePlayer: "플레이어 열기" | ||||
| disablePlayer: "플레이어 닫기" | ||||
| expandTweet: "트윗 확장하기" | ||||
| @@ -533,21 +528,51 @@ generateAccessToken: "액세스 토큰 생성" | ||||
| permission: "권한" | ||||
| enableAll: "전체 선택" | ||||
| disableAll: "전체 해제" | ||||
| tokenRequested: "계정 접근 허용" | ||||
| edit: "편집" | ||||
| useStarForReactionFallback: "알 수 없는 리액션 이모지 대신 ★ 사용" | ||||
| emailConfig: "메일 서버 설정" | ||||
| emailConfigInfo: "가입 시 메일 주소 확인이나 비밀번호 초기화 시에 사용합니다." | ||||
| email: "메일 주소" | ||||
| smtpConfig: "SMTP 서버 설정" | ||||
| smtpHost: "호스트" | ||||
| smtpPort: "포트" | ||||
| smtpUser: "유저명" | ||||
| smtpPass: "비밀번호" | ||||
| emptyToDisableSmtpAuth: "SMTP 인증을 사용하지 않으려면 공란으로 비워둡니다." | ||||
| smtpSecure: "SMTP 연결에 Implicit SSL/TTS 사용" | ||||
| smtpSecureInfo: "STARTTLS 사용 시에는 해제합니다." | ||||
| wordMute: "단어 뮤트" | ||||
| makeActive: "활성화" | ||||
| copy: "복사" | ||||
| logs: "로그" | ||||
| database: "데이터베이스" | ||||
| channel: "채널" | ||||
| random: "랜덤" | ||||
| _mfm: | ||||
|   mention: "멘션" | ||||
|   hashtag: "해시태그" | ||||
|   link: "링크" | ||||
|   center: "가운데 정렬" | ||||
|   quote: "인용" | ||||
|   emoji: "커스텀 이모지" | ||||
|   search: "검색" | ||||
| _reversi: | ||||
|   total: "합계" | ||||
| _channel: | ||||
|   create: "채널 생성" | ||||
|   setBanner: "배너 설정" | ||||
|   removeBanner: "배너 삭제" | ||||
|   featured: "트렌드" | ||||
|   following: "팔로잉" | ||||
|   usersCount: "{n}명 참여 중" | ||||
|   notesCount: "{n}노트" | ||||
| _sidebar: | ||||
|   icon: "아바타" | ||||
|   hide: "숨기기" | ||||
| _wordMute: | ||||
|   muteWords: "뮤트할 단어" | ||||
|   mutedNotes: "뮤트된 노트" | ||||
| _theme: | ||||
|   explore: "테마 찾아보기" | ||||
|   install: "테마 설치" | ||||
| @@ -568,7 +593,10 @@ _theme: | ||||
|   func: "함수" | ||||
|   funcKind: "함수 종류" | ||||
|   argument: "매개변수" | ||||
|   importInfo: "여기에 테마 코드를 붙여 넣어 에디터로 불러올 수 있습니다." | ||||
|   keys: | ||||
|     link: "링크" | ||||
|     hashtag: "해시태그" | ||||
|     mention: "멘션" | ||||
|     renote: "Renote" | ||||
|     divider: "구분선" | ||||
| @@ -687,6 +715,7 @@ _widgets: | ||||
|   photos: "사진" | ||||
|   digitalClock: "디지털 시계" | ||||
|   federation: "연합" | ||||
|   postForm: "글 입력란" | ||||
| _cw: | ||||
|   hide: "숨기기" | ||||
|   show: "더 보기" | ||||
| @@ -860,6 +889,7 @@ _pages: | ||||
|   my: "내 페이지" | ||||
|   liked: "좋아요한 페이지" | ||||
|   inspector: "인스펙터" | ||||
|   contents: "콘텐츠" | ||||
|   content: "페이지 블록" | ||||
|   variables: "변수" | ||||
|   title: "제목" | ||||
| @@ -1145,6 +1175,9 @@ _notification: | ||||
|     quote: "인용" | ||||
|     reaction: "리액션" | ||||
| _deck: | ||||
|   alwaysShowMainColumn: "메인 칼럼 항상 표시" | ||||
|   columnAlign: "칼럼 정렬" | ||||
|   addColumn: "칼럼 추가" | ||||
|   swapLeft: "왼쪽으로 이동" | ||||
|   swapRight: "오른쪽으로 이동" | ||||
|   swapUp: "위로 이동" | ||||
|   | ||||
| @@ -1,2 +1,23 @@ | ||||
| --- | ||||
| _lang_: "język polski" | ||||
| search: "Szukaj" | ||||
| notifications: "Powiadomienia" | ||||
| username: "Nazwa użytkownika" | ||||
| password: "Hasło" | ||||
| ok: "OK" | ||||
| gotIt: "Rozumiem!" | ||||
| cancel: "Anuluj" | ||||
| enterUsername: "Wprowadź nazwę użytkownika" | ||||
| smtpUser: "Nazwa użytkownika" | ||||
| smtpPass: "Hasło" | ||||
| _mfm: | ||||
|   search: "Szukaj" | ||||
| _sfx: | ||||
|   notification: "Powiadomienia" | ||||
| _widgets: | ||||
|   notifications: "Powiadomienia" | ||||
| _profile: | ||||
|   username: "Nazwa użytkownika" | ||||
| _deck: | ||||
|   _columns: | ||||
|     notifications: "Powiadomienia" | ||||
|   | ||||
							
								
								
									
										1374
									
								
								locales/ru-RU.yml
									
									
									
									
									
								
							
							
						
						
									
										1374
									
								
								locales/ru-RU.yml
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -1,3 +1,5 @@ | ||||
| --- | ||||
| _lang_: "ياپونچە" | ||||
| search: "ئىزدەش" | ||||
| _mfm: | ||||
|   search: "ئىزدەش" | ||||
|   | ||||
							
								
								
									
										496
									
								
								locales/uk-UA.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										496
									
								
								locales/uk-UA.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,496 @@ | ||||
| --- | ||||
| _lang_: "Українська" | ||||
| monthAndDay: "{month}/{day}" | ||||
| search: "Пошук" | ||||
| notifications: "Сповіщення" | ||||
| username: "Ім'я користувача" | ||||
| password: "Пароль" | ||||
| fetchingAsApObject: "Отримуємо з федіверсу..." | ||||
| ok: "OK" | ||||
| gotIt: "Зрозуміло!" | ||||
| cancel: "Скасувати" | ||||
| enterUsername: "Введіть ім'я користувача" | ||||
| renotedBy: "Поширено {user}" | ||||
| noNotes: "Немає дописів" | ||||
| noNotifications: "Немає сповіщень" | ||||
| instance: "Інстанс" | ||||
| settings: "Налаштування" | ||||
| basicSettings: "Основні налаштування" | ||||
| otherSettings: "Інші налаштування" | ||||
| openInWindow: "Відкрити у вікні" | ||||
| profile: "Профіль" | ||||
| timeline: "Стрічка" | ||||
| noAccountDescription: "Цей користувач ще нічого не написав про себе" | ||||
| login: "Увійти" | ||||
| loggingIn: "Здійснюємо вхід..." | ||||
| logout: "Вийти" | ||||
| signup: "Реєстрація" | ||||
| uploading: "Завантаження..." | ||||
| save: "Зберегти" | ||||
| users: "Користувачі" | ||||
| addUser: "Додати користувача" | ||||
| favorite: "Обране" | ||||
| favorites: "Обране" | ||||
| unfavorite: "Видалити з обраного" | ||||
| pin: "Закріпити" | ||||
| unpin: "Відкріпити" | ||||
| copyContent: "Скопіювати контент" | ||||
| copyLink: "Скопіювати посилання" | ||||
| delete: "Видалити" | ||||
| deleteAndEdit: "Видалити й редагувати" | ||||
| addToList: "Додати до списку" | ||||
| sendMessage: "Надіслати повідомлення" | ||||
| copyUsername: "Скопіювати ім’я користувача" | ||||
| searchUser: "Пошук користувачів" | ||||
| reply: "Відповісти" | ||||
| loadMore: "Показати більше" | ||||
| youGotNewFollower: "У вас новий підписник" | ||||
| receiveFollowRequest: "Отримано запит на підписку" | ||||
| followRequestAccepted: "Запит на підписку прийнято" | ||||
| mention: "Згадка" | ||||
| mentions: "Згадки" | ||||
| directNotes: "Прямі повідомлення" | ||||
| importAndExport: "Імпорт та експорт" | ||||
| import: "Імпорт" | ||||
| export: "Експорт" | ||||
| files: "Файли" | ||||
| download: "Завантажити" | ||||
| unfollowConfirm: "Ви впевнені, що хочете відписатися від {name}?" | ||||
| lists: "Списки" | ||||
| noLists: "Немає списків" | ||||
| note: "Дописи" | ||||
| notes: "Дописи" | ||||
| following: "Підписки" | ||||
| followers: "Підписники" | ||||
| followsYou: "Підписаний(-а) на вас" | ||||
| createList: "Створити список" | ||||
| manageLists: "Управління списками" | ||||
| error: "Помилка" | ||||
| somethingHappened: "Щось пішло не так" | ||||
| retry: "Спробувати знову" | ||||
| pageLoadError: "Помилка при завантаженні сторінки" | ||||
| enterListName: "Введіть назву списку" | ||||
| privacy: "Приватність" | ||||
| makeFollowManuallyApprove: "Підтверджувати підписників уручну" | ||||
| defaultNoteVisibility: "Видимість допису за замовчуванням" | ||||
| follow: "Підписка" | ||||
| followRequest: "Запит на підписку" | ||||
| followRequests: "Запити на підписку" | ||||
| unfollow: "Відписатися" | ||||
| followRequestPending: "Очікуючі запити на підписку" | ||||
| enterEmoji: "Введіть емодзі" | ||||
| renote: "Поширити" | ||||
| unrenote: "Відміна поширення" | ||||
| quote: "Цитата" | ||||
| pinnedNote: "Закріплений допис" | ||||
| you: "Ви" | ||||
| clickToShow: "Натисніть для перегляду" | ||||
| sensitive: "NSFW" | ||||
| add: "Додати" | ||||
| reaction: "Реакції" | ||||
| rememberNoteVisibility: "Пам’ятати видимість дописів" | ||||
| attachCancel: "Видалити вкладення" | ||||
| markAsSensitive: "Позначити як NSFW" | ||||
| unmarkAsSensitive: "Зняти позначку NSFW" | ||||
| enterFileName: "Введіть ім'я файлу" | ||||
| mute: "Ігнорувати" | ||||
| unmute: "Показувати" | ||||
| block: "Заблокувати" | ||||
| unblock: "Розблокувати" | ||||
| suspend: "Призупинити" | ||||
| unsuspend: "Відновити" | ||||
| blockConfirm: "Ви впевнені, що хочете заблокувати цей акаунт?" | ||||
| unblockConfirm: "Ви впевнені, що хочете розблокувати цей акаунт?" | ||||
| suspendConfirm: "Ви впевнені, що хочете призупинити цей акаунт?" | ||||
| unsuspendConfirm: "Ви впевнені, що хочете відновити цей акаунт?" | ||||
| selectList: "Виберіть список" | ||||
| selectAntenna: "Виберіть антену" | ||||
| selectWidget: "Виберіть віджет" | ||||
| editWidgets: "Редагувати віджети" | ||||
| editWidgetsExit: "Готово" | ||||
| customEmojis: "Кастомні емоджі" | ||||
| emoji: "Емоджі" | ||||
| emojiName: "Назва емоджі" | ||||
| emojiUrl: "URL емодзі" | ||||
| addEmoji: "Додати емодзі" | ||||
| settingGuide: "Рекомендована конфігурація" | ||||
| cacheRemoteFiles: "Кешувати дані з інших інстансів" | ||||
| flagAsBot: "Акаунт бота" | ||||
| flagAsCat: "Акаунт кота" | ||||
| autoAcceptFollowed: "Автоматично приймати запити на підписку від користувачів, на яких ви підписані" | ||||
| addAcount: "Додати акаунт" | ||||
| loginFailed: "Не вдалося увійти" | ||||
| showOnRemote: "Переглянути в оригіналі" | ||||
| general: "Загальне" | ||||
| wallpaper: "Шпалери" | ||||
| setWallpaper: "Встановити шпалери" | ||||
| removeWallpaper: "Прибрати шпалери" | ||||
| searchWith: "Шукати з {q}" | ||||
| youHaveNoLists: "У вас немає списків" | ||||
| followConfirm: "Підписатися на {name}?" | ||||
| proxyAccount: "Проксі-акаунт" | ||||
| host: "Хост" | ||||
| selectUser: "Виберіть користувача" | ||||
| recipient: "Кому" | ||||
| annotation: "Коментар" | ||||
| federation: "Федіверс" | ||||
| instances: "Інстанс" | ||||
| registeredAt: "Приєднався(-лась)" | ||||
| latestRequestSentAt: "Останній запит надіслано" | ||||
| latestRequestReceivedAt: "Останній запит прийнято" | ||||
| latestStatus: "Останній статус" | ||||
| storageUsage: "Використання простіру" | ||||
| charts: "Графіки" | ||||
| perHour: "Щогодини" | ||||
| perDay: "Щоденно" | ||||
| stopActivityDelivery: "Припинити розсилання активності" | ||||
| blockThisInstance: "Заблокувати цей інстанс" | ||||
| operations: "Операції" | ||||
| software: "Програмне забезпечення" | ||||
| version: "Версія" | ||||
| metadata: "Метадані" | ||||
| withNFiles: "файли: {n}" | ||||
| monitor: "Монітор" | ||||
| jobQueue: "Черга завдань" | ||||
| cpuAndMemory: "ЦП та пам'ять" | ||||
| network: "Мережа" | ||||
| disk: "Диск" | ||||
| instanceInfo: "Про цей інстанс" | ||||
| statistics: "Статистика" | ||||
| clearQueue: "Очистити чергу" | ||||
| clearQueueConfirmTitle: "Ви впевнені, що хочете очистити чергу?" | ||||
| clearCachedFiles: "Очистити кеш" | ||||
| clearCachedFilesConfirm: "Ви впевнені, що хочете видалити всі кешовані файли?" | ||||
| blockedInstances: "Заблоковані інстанси" | ||||
| muteAndBlock: "Ігнор і блокування" | ||||
| mutedUsers: "Ігноровані користувачі" | ||||
| blockedUsers: "Заблоковані користувачі" | ||||
| noUsers: "Немає користувачів" | ||||
| editProfile: "Редагувати профіль" | ||||
| noteDeleteConfirm: "Ви дійсно хочете видалити цей допис?" | ||||
| pinLimitExceeded: "Більше дописів не можна закріпити" | ||||
| done: "Готово" | ||||
| processing: "Обробка" | ||||
| preview: "Передогляд" | ||||
| default: "За умовчанням" | ||||
| noCustomEmojis: "Немає кастомних емоджі" | ||||
| noJobs: "Немає завдань" | ||||
| federating: "Федерується" | ||||
| blocked: "Заблоковано" | ||||
| suspended: "Призупинено" | ||||
| notResponding: "Не відповідає" | ||||
| changePassword: "Змінити пароль" | ||||
| security: "Безпека" | ||||
| currentPassword: "Поточний пароль" | ||||
| newPassword: "Новий пароль" | ||||
| newPasswordRetype: "Новий пароль (повторно)" | ||||
| attachFile: "Вкласти файл" | ||||
| more: "Бiльше!" | ||||
| featured: "Виділено" | ||||
| noSuchUser: "Користувача не знайдено" | ||||
| lookup: "Пошук" | ||||
| announcements: "Оголошення" | ||||
| imageUrl: "URL зображення" | ||||
| remove: "Видалити" | ||||
| removed: "Видалено" | ||||
| saved: "Збережено" | ||||
| messaging: "Чати" | ||||
| upload: "Завантажити" | ||||
| fromDrive: "З диска" | ||||
| fromUrl: "З URL" | ||||
| uploadFromUrl: "Завантажити з URL" | ||||
| explore: "Огляд" | ||||
| games: "Ігри Misskey" | ||||
| noMoreHistory: "Подальшої історії немає" | ||||
| start: "Розпочати" | ||||
| home: "Домівка" | ||||
| activity: "Активність" | ||||
| images: "Зображення" | ||||
| birthday: "День народження" | ||||
| yearsOld: "{age} років" | ||||
| registeredDate: "Приєднався(-лась)" | ||||
| location: "Локація" | ||||
| theme: "Тема" | ||||
| themeForLightMode: "Світла тема" | ||||
| themeForDarkMode: "Темна тема" | ||||
| light: "Світла" | ||||
| dark: "Темна" | ||||
| lightThemes: "Світлі теми" | ||||
| darkThemes: "Темні теми" | ||||
| drive: "Диск" | ||||
| fileName: "Ім'я файлу" | ||||
| selectFile: "Вибрати файл" | ||||
| selectFiles: "Вибрати файли" | ||||
| selectFolder: "Вибрати теку" | ||||
| selectFolders: "Вибрати теки" | ||||
| renameFile: "Перейменувати файл" | ||||
| folderName: "Ім'я теки" | ||||
| createFolder: "Створити теку" | ||||
| renameFolder: "Перейменувати теку" | ||||
| deleteFolder: "Видалити теку" | ||||
| addFile: "Додати файл" | ||||
| emptyDrive: "Диск порожній" | ||||
| emptyFolder: "Тека порожня" | ||||
| unableToDelete: "Видалення неможливе" | ||||
| inputNewFileName: "Введіть ім'я нового файлу" | ||||
| inputNewFolderName: "Введіть ім'я нової теки" | ||||
| hasChildFilesOrFolders: "Ця тека не порожня і не може бути видалена" | ||||
| copyUrl: "Копіювати URL" | ||||
| rename: "Перейменувати" | ||||
| avatar: "Аватар" | ||||
| banner: "Банер" | ||||
| nsfw: "NSFW" | ||||
| disconnectedFromServer: "Підключення до сервера було перервано" | ||||
| reload: "Оновити" | ||||
| doNothing: "Нічого не робити" | ||||
| reloadConfirm: "Перезавантажити стрічку?" | ||||
| watch: "Стежити" | ||||
| unwatch: "Не стежити" | ||||
| accept: "Прийняти" | ||||
| reject: "Відхилити" | ||||
| instanceName: "Назва інстансу" | ||||
| instanceDescription: "Описання інстансу" | ||||
| maintainerName: "Ім'я адміністратора" | ||||
| maintainerEmail: "Email адміністратора" | ||||
| tosUrl: "URL умов використання" | ||||
| thisYear: "Рік" | ||||
| thisMonth: "Місяць" | ||||
| today: "День" | ||||
| dayX: "{day}" | ||||
| monthX: "{month}" | ||||
| yearX: "{year}" | ||||
| pages: "Сторінки" | ||||
| integration: "Інтеграція" | ||||
| connectSerice: "Під’єднатися" | ||||
| disconnectSerice: "Відключитися" | ||||
| enableLocalTimeline: "Увімкнути локальну стрічку" | ||||
| enableGlobalTimeline: "Увімкнути глобальну стрічку" | ||||
| registration: "Реєстрація" | ||||
| enableRegistration: "Дозволити реєстрацію" | ||||
| invite: "Запрошення" | ||||
| proxyRemoteFiles: "Проксувати файли з інших інстансів" | ||||
| iconUrl: "URL аватара" | ||||
| bannerUrl: "URL банера" | ||||
| basicInfo: "Основна інформація" | ||||
| pinnedUsers: "Закріплені користувачі" | ||||
| hcaptcha: "hCaptcha" | ||||
| enableHcaptcha: "Увімкнути hCaptcha" | ||||
| hcaptchaSiteKey: "Ключ сайту" | ||||
| hcaptchaSecretKey: "Секретний ключ" | ||||
| recaptcha: "reCAPTCHA" | ||||
| enableRecaptcha: "Увімкнути reCAPTCHA" | ||||
| recaptchaSiteKey: "Ключ сайту" | ||||
| recaptchaSecretKey: "Секретний ключ" | ||||
| antennas: "Антени" | ||||
| manageAntennas: "Налаштування антен" | ||||
| name: "Ім'я" | ||||
| antennaSource: "Джерело антени" | ||||
| antennaKeywords: "Ключові слова антени" | ||||
| antennaExcludeKeywords: "Винятки" | ||||
| serviceworker: "ServiceWorker" | ||||
| enableServiceworker: "Ввімкнути ServiceWorker" | ||||
| caseSensitive: "З урахуванням регістру" | ||||
| notesAndReplies: "Дописи та відповіді" | ||||
| popularUsers: "Популярні користувачі" | ||||
| recentlyUpdatedUsers: "Нещодавно активні користувачі" | ||||
| recentlyRegisteredUsers: "Нещодавно зареєстровані користувачі" | ||||
| recentlyDiscoveredUsers: "Нещодавно знайдені користувачі" | ||||
| exploreUsersCount: "{count} користувачів" | ||||
| exploreFediverse: "Огляд федіверсу" | ||||
| popularTags: "Популярні теги" | ||||
| userList: "Списки" | ||||
| about: "Інформація" | ||||
| aboutMisskey: "Про Misskey" | ||||
| administrator: "Адмін" | ||||
| token: "Токен" | ||||
| twoStepAuthentication: "Двохфакторна аутентифікація" | ||||
| moderator: "Модератор" | ||||
| securityKey: "Ключ захисту" | ||||
| securityKeyName: "Назва ключа" | ||||
| registerSecurityKey: "Зареєструвати ключ захисту" | ||||
| lastUsed: "Востаннє використано" | ||||
| unregister: "Скасувати реєстрацію" | ||||
| passwordLessLogin: "Налаштувати вхід без пароля" | ||||
| resetPassword: "Скинути пароль" | ||||
| newPasswordIs: "Новий пароль: {password}" | ||||
| share: "Поділитись" | ||||
| notFound: "Не знайдено" | ||||
| cacheClear: "Очистити кеш" | ||||
| help: "Допомога" | ||||
| inputMessageHere: "Введіть повідомлення тут" | ||||
| close: "Закрити" | ||||
| group: "Група" | ||||
| groups: "Групи" | ||||
| createGroup: "Створити групу" | ||||
| ownedGroups: "Власні групи" | ||||
| invites: "Запрошення" | ||||
| groupName: "Назва групи" | ||||
| transfer: "Передача" | ||||
| messagingWithUser: "Чат з користувачами" | ||||
| messagingWithGroup: "Чат з групою" | ||||
| title: "Тема" | ||||
| text: "Текст" | ||||
| next: "Далі" | ||||
| retype: "Введіть ще раз" | ||||
| noteOf: "Допис {user}" | ||||
| inviteToGroup: "Запрошення до групи" | ||||
| maxNoteTextLength: "Максимальна довжина допису" | ||||
| quoteAttached: "Цитата" | ||||
| quoteQuestion: "Ви хочете додати цитату?" | ||||
| noMessagesYet: "Ще немає повідомлень" | ||||
| newMessageExists: "Є нові повідомлення" | ||||
| onlyOneFileCanBeAttached: "До повідомлення можна вкласти лише один файл" | ||||
| signinRequired: "Будь ласка, авторизуйтесь" | ||||
| invitations: "Запрошення" | ||||
| invitationCode: "Код запрошення" | ||||
| checking: "Перевірка…" | ||||
| available: "Доступно" | ||||
| unavailable: "Недоступно" | ||||
| usernameInvalidFormat: "літери, цифри та _ є прийнятними" | ||||
| tooShort: "Занадто короткий" | ||||
| tooLong: "Занадто довгий" | ||||
| weakPassword: "Слабкий пароль" | ||||
| strongPassword: "Міцний пароль" | ||||
| passwordMatched: "Все вірно" | ||||
| passwordNotMatched: "Паролі не співпадають" | ||||
| signinWith: "Увійти за допомогою {x}" | ||||
| uiLanguage: "Мова інтерфейсу" | ||||
| aboutX: "Про {x}" | ||||
| useOsNativeEmojis: "Використовувати емодзі ОС" | ||||
| youHaveNoGroups: "Немає груп" | ||||
| noHistory: "Історія порожня" | ||||
| disableAnimatedMfm: "Відключити анімації MFM" | ||||
| doing: "Виконується" | ||||
| category: "Категорія" | ||||
| tags: "Теги" | ||||
| docSource: "Джерело цього документа" | ||||
| createAccount: "Створити акаунт" | ||||
| existingAcount: "Існуючий акаунт" | ||||
| regenerate: "Оновити" | ||||
| fontSize: "Розмір шрифту" | ||||
| noFollowRequests: "Немає запитів на підписку" | ||||
| dashboard: "Панель приладів" | ||||
| local: "Локальні" | ||||
| remote: "Віддалений" | ||||
| total: "Всього" | ||||
| appearance: "Вигляд" | ||||
| clientSettings: "Налаштування клієнта" | ||||
| accountSettings: "Налаштування акаунта" | ||||
| promotion: "Просування" | ||||
| promote: "Просунути" | ||||
| numberOfDays: "Кількість днів" | ||||
| hideThisNote: "Сховати цей допис" | ||||
| objectStorageBaseUrl: "Base URL" | ||||
| objectStorageBucket: "Bucket" | ||||
| objectStoragePrefix: "Prefix" | ||||
| objectStorageEndpoint: "Endpoint" | ||||
| objectStorageRegion: "Region" | ||||
| objectStorageUseSSL: "Використовувати SSL" | ||||
| objectStorageUseProxy: "Використовувати Proxy" | ||||
| deleteAll: "Видалити все" | ||||
| newNoteRecived: "Є нові дописи" | ||||
| sounds: "Звук" | ||||
| listen: "Слухати" | ||||
| showInPage: "Показати на сторінці" | ||||
| volume: "Гучність" | ||||
| install: "Інсталювати" | ||||
| uninstall: "Видалити" | ||||
| sort: "Сортування" | ||||
| ascendingOrder: "За зростанням" | ||||
| descendingOrder: "За спаданням" | ||||
| script: "Скрипт" | ||||
| deleteAllFiles: "Видалити всі файли" | ||||
| deleteAllFilesConfirm: "Ви дійсно хочете видалити всі файли?" | ||||
| sidebar: "Бокова панель" | ||||
| rooms: "Кімнати" | ||||
| relays: "Ретранслятори" | ||||
| addRelay: "Додати ретранслятор" | ||||
| smtpHost: "Хост" | ||||
| smtpUser: "Ім'я користувача" | ||||
| smtpPass: "Пароль" | ||||
| regenerateLoginToken: "Оновити Login Token" | ||||
| _mfm: | ||||
|   cheatSheet: " Довідка MFM" | ||||
|   mention: "Згадка" | ||||
|   quote: "Цитата" | ||||
|   emoji: "Кастомні емоджі" | ||||
|   search: "Пошук" | ||||
| _reversi: | ||||
|   total: "Всього" | ||||
| _sidebar: | ||||
|   icon: "Аватар" | ||||
| _theme: | ||||
|   keys: | ||||
|     mention: "Згадка" | ||||
|     renote: "Поширити" | ||||
| _sfx: | ||||
|   note: "Дописи" | ||||
|   notification: "Сповіщення" | ||||
|   chat: "Чати" | ||||
| _antennaSources: | ||||
|   homeTimeline: "Дописи тих, на кого ви підписані" | ||||
| _widgets: | ||||
|   notifications: "Сповіщення" | ||||
|   timeline: "Стрічка" | ||||
|   activity: "Активність" | ||||
|   federation: "Федіверс" | ||||
| _cw: | ||||
|   show: "Показати більше" | ||||
| _visibility: | ||||
|   home: "Домівка" | ||||
|   followers: "Підписники" | ||||
|   localOnly: "Лише локально" | ||||
| _postForm: | ||||
|   replyPlaceholder: "Відповідь на допис..." | ||||
| _profile: | ||||
|   name: "Ім'я" | ||||
|   username: "Ім'я користувача" | ||||
| _exportOrImport: | ||||
|   followingList: "Підписки" | ||||
|   muteList: "Ігнорувати" | ||||
|   blockingList: "Заблокувати" | ||||
|   userLists: "Списки" | ||||
| _timelines: | ||||
|   home: "Домівка" | ||||
| _rooms: | ||||
|   _roomType: | ||||
|     default: "За умовчанням" | ||||
|   _furnitures: | ||||
|     monitor: "Монітор" | ||||
| _pages: | ||||
|   blocks: | ||||
|     image: "Зображення" | ||||
|   script: | ||||
|     categories: | ||||
|       list: "Списки" | ||||
|     blocks: | ||||
|       _join: | ||||
|         arg1: "Списки" | ||||
|       _randomPick: | ||||
|         arg1: "Списки" | ||||
|       _dailyRandomPick: | ||||
|         arg1: "Списки" | ||||
|       _seedRandomPick: | ||||
|         arg2: "Списки" | ||||
|       _pick: | ||||
|         arg1: "Списки" | ||||
|       _listLen: | ||||
|         arg1: "Списки" | ||||
|     types: | ||||
|       array: "Списки" | ||||
| _notification: | ||||
|   youRenoted: "{name} поширив(ла) ваш допис" | ||||
|   youWereFollowed: "У вас новий підписник" | ||||
|   _types: | ||||
|     follow: "Підписки" | ||||
|     mention: "Згадка" | ||||
|     renote: "Поширити" | ||||
|     quote: "Цитата" | ||||
|     reaction: "Реакції" | ||||
| _deck: | ||||
|   _columns: | ||||
|     notifications: "Сповіщення" | ||||
|     tl: "Стрічка" | ||||
|     antenna: "Антени" | ||||
|     list: "Списки" | ||||
|     mentions: "Згадки" | ||||
| @@ -16,6 +16,9 @@ noNotes: "没有帖文" | ||||
| noNotifications: "无通知" | ||||
| instance: "实例" | ||||
| settings: "设置" | ||||
| basicSettings: "基本设置" | ||||
| otherSettings: "其他设置" | ||||
| openInWindow: "在新窗口中打开" | ||||
| profile: "个人资料" | ||||
| timeline: "时间线" | ||||
| noAccountDescription: "这个人很懒,没有写自我介绍" | ||||
| @@ -40,6 +43,7 @@ deleteAndEditConfirm: "要删除此帖并再次编辑吗?对此帖的所有回 | ||||
| addToList: "添加至列表" | ||||
| sendMessage: "发送" | ||||
| copyUsername: "复制用户名" | ||||
| searchUser: "搜索用户" | ||||
| reply: "回复" | ||||
| loadMore: "查看更多" | ||||
| youGotNewFollower: "你有新的关注者" | ||||
| @@ -66,8 +70,11 @@ followers: "关注者" | ||||
| followsYou: "关注了你" | ||||
| createList: "创建列表" | ||||
| manageLists: "管理列表" | ||||
| error: "有点小问题" | ||||
| error: "错误" | ||||
| somethingHappened: "出现了问题" | ||||
| retry: "重试" | ||||
| pageLoadError: "页面加载失败。" | ||||
| pageLoadErrorDescription: "这通常是由于网络或浏览器缓存的原因。请清除缓存或等待片刻后重试。" | ||||
| enterListName: "输入列表名称" | ||||
| privacy: "隐私" | ||||
| makeFollowManuallyApprove: "关注者请求需要批准" | ||||
| @@ -77,7 +84,7 @@ followRequest: "关注申请" | ||||
| followRequests: "关注申请" | ||||
| unfollow: "取消关注" | ||||
| followRequestPending: "发送关注申请" | ||||
| enterEmoji: "输入Emoji" | ||||
| enterEmoji: "输入表情符号" | ||||
| renote: "转发" | ||||
| unrenote: "取消转发" | ||||
| quote: "引用" | ||||
| @@ -106,11 +113,13 @@ unsuspendConfirm: "要解除冻结吗?" | ||||
| selectList: "选择列表" | ||||
| selectAntenna: "天线选择" | ||||
| selectWidget: "选择小工具" | ||||
| editWidgets: "编辑小工具" | ||||
| editWidgetsExit: "完成编辑" | ||||
| customEmojis: "自定义Emoji" | ||||
| emoji: "表情符号" | ||||
| emojiName: "Emoji 名称" | ||||
| emojiUrl: "emoji 地址" | ||||
| addEmoji: "添加Emoji" | ||||
| emojiName: "表情符号名称" | ||||
| emojiUrl: "表情符号地址" | ||||
| addEmoji: "添加表情符号" | ||||
| settingGuide: "推荐配置" | ||||
| cacheRemoteFiles: "远程文件缓存" | ||||
| cacheRemoteFilesDescription: "当禁用此设定时远程文件将直接从远程实例载入。禁用后会减小储存空间需求,但是会增加流量,因为缩略图不会被生成。" | ||||
| @@ -155,7 +164,7 @@ jobQueue: "作业队列" | ||||
| cpuAndMemory: "CPU使用量" | ||||
| network: "网络" | ||||
| disk: "存储" | ||||
| instanceInfo: "实例情报" | ||||
| instanceInfo: "实例信息" | ||||
| statistics: "统计" | ||||
| clearQueue: "清除队列" | ||||
| clearQueueConfirmTitle: "确定清除队列?" | ||||
| @@ -176,8 +185,7 @@ done: "完成" | ||||
| processing: "处理中" | ||||
| preview: "预览" | ||||
| default: "默认" | ||||
| noCustomEmojis: "无自定义Emoji" | ||||
| customEmojisOfRemote: "远程Emoji" | ||||
| noCustomEmojis: "没有自定义表情符号" | ||||
| noJobs: "没有任务" | ||||
| federating: "联合中" | ||||
| blocked: "已拦截" | ||||
| @@ -206,6 +214,8 @@ imageUrl: "图片URL" | ||||
| remove: "删除" | ||||
| removed: "已删除" | ||||
| removeAreYouSure: "要删掉「{x}」吗?" | ||||
| deleteAreYouSure: "要删掉「{x}」吗?" | ||||
| resetAreYouSure: "恢复默认设置?" | ||||
| saved: "已保存" | ||||
| messaging: "聊天" | ||||
| upload: "上传" | ||||
| @@ -264,6 +274,7 @@ rename: "重命名" | ||||
| avatar: "头像" | ||||
| banner: "Banner" | ||||
| nsfw: "阅读注意" | ||||
| whenServerDisconnected: "与服务器连接中断时" | ||||
| disconnectedFromServer: "已从服务器断开连接" | ||||
| reload: "重新加载" | ||||
| doNothing: "什么都不做" | ||||
| @@ -285,7 +296,7 @@ dayX: "{day}日" | ||||
| monthX: "{month}月" | ||||
| yearX: "{year}年" | ||||
| pages: "页面" | ||||
| integration: "连携" | ||||
| integration: "关联" | ||||
| connectSerice: "已连接" | ||||
| disconnectSerice: "断开连接" | ||||
| enableLocalTimeline: "启用本地时间线功能" | ||||
| @@ -344,7 +355,7 @@ popularTags: "热门标签" | ||||
| userList: "列表" | ||||
| about: "关于" | ||||
| aboutMisskey: "关于 Misskey" | ||||
| aboutMisskeyText: "Misskey是由syuilo于2014年开发的开放源代码软件。" | ||||
| aboutMisskeyText: "Misskey是由syuilo于2014年开发的开源软件。" | ||||
| misskeyMembers: "现在由以下成员进行开发和维护:" | ||||
| misskeySource: "源代码在这里公开:" | ||||
| misskeyTranslation: "与我们一同进行Misskey的翻译工作:" | ||||
| @@ -358,15 +369,12 @@ moderator: "版主" | ||||
| nUsersMentioned: "{n} 被提到" | ||||
| securityKey: "安全密钥" | ||||
| securityKeyName: "密钥名称" | ||||
| registerSecurityKey: "注册安全密钥" | ||||
| registerSecurityKey: "注册硬件安全密钥" | ||||
| lastUsed: "最后使用:" | ||||
| unregister: "删除账户" | ||||
| passwordLessLogin: "无密码登录" | ||||
| resetPassword: "重置密码" | ||||
| newPasswordIs: "新的密码是「{password}」" | ||||
| autoReloadWhenDisconnected: "断开连接时自动重新加载" | ||||
| autoNoteWatch: "自动关注帖子" | ||||
| autoNoteWatchDescription: "让您能够收到关于「回应」和回复其他用户的帖子的通知。" | ||||
| reduceUiAnimation: "减少UI动画" | ||||
| share: "分享" | ||||
| notFound: "未找到" | ||||
| @@ -404,6 +412,7 @@ noMessagesYet: "现在没有新的聊天" | ||||
| newMessageExists: "新信息" | ||||
| onlyOneFileCanBeAttached: "只能添加一个附件" | ||||
| signinRequired: "请先登录" | ||||
| invitations: "邀请" | ||||
| invitationCode: "邀请码" | ||||
| checking: "正在确认" | ||||
| available: "可用" | ||||
| @@ -423,7 +432,7 @@ or: "或者" | ||||
| uiLanguage: "显示语言" | ||||
| groupInvited: "群组招待" | ||||
| aboutX: "关于 {x}" | ||||
| useOsNativeEmojis: "使用OS原生Emoji" | ||||
| useOsNativeEmojis: "使用OS原生表情符号" | ||||
| youHaveNoGroups: "没有群组" | ||||
| joinOrCreateGroup: "请加入一个现有的群组,或者创建新群组。" | ||||
| noHistory: "没有历史记录" | ||||
| @@ -445,7 +454,7 @@ total: "总计" | ||||
| weekOverWeekChanges: "与前一周相比" | ||||
| dayOverDayChanges: "与前一日相比" | ||||
| appearance: "外观" | ||||
| clinetSettings: "客户端设置" | ||||
| clientSettings: "客户端设置" | ||||
| accountSettings: "账户设置" | ||||
| promotion: "推广" | ||||
| promote: "推广" | ||||
| @@ -455,19 +464,20 @@ showFeaturedNotesInTimeline: "在时间线上显示热门推荐" | ||||
| objectStorage: "对象存储" | ||||
| useObjectStorage: "使用对象存储" | ||||
| objectStorageBaseUrl: "基本网址" | ||||
| objectStorageBaseUrlDesc: "供参考的URL。如果使用CDN或Proxy,则其URL为S3:\"https://<bucket>.s3.amazonaws.com\"、GCS等:\"https://storage-googleapis.proxy.ustclug.org/<bucket>\"。" | ||||
| objectStorageBaseUrlDesc: "URL前缀,用于构造URL到对象(媒体)的引用,如果您使用的是CDN或反向代理,请指定其URL,否则请根据您使用的服务指定可公开访问的地址。例如“https://<bucket>.s3.amazonaws.com”用于AWS S3,“https://storage.googleapis.com/<bucket>”用于GCS" | ||||
| objectStorageBucket: "存储桶" | ||||
| objectStorageBucketDesc: "请指定使用的对象存储服务的存储桶名称。" | ||||
| objectStoragePrefix: "前缀" | ||||
| objectStoragePrefixDesc: "文件将存储在此前缀的目录下。" | ||||
| objectStorageEndpoint: "端点" | ||||
| objectStorageEndpointDesc: "S3默认情况下为空,否则请为每个服务指定端点。 指定为“<host>”或“<host>:<port>”。" | ||||
| objectStorageEndpointDesc: "如果你希望使用AWS S3请留空。否则请根据你使用的服务来进行设置,指定端点形式为“<host>”或“<host>:<port>”。" | ||||
| objectStorageRegion: "可用区" | ||||
| objectStorageRegionDesc: "指定一个可用区,例如“xx-east-1”。 如果您的对象存储服务没有可用区概念,请将其留空或填写“us-east-1”。" | ||||
| objectStorageUseSSL: "使用SSL" | ||||
| objectStorageUseSSLDesc: "如果不使用https进行API连接,请关闭。" | ||||
| objectStorageUseProxy: "使用代理" | ||||
| objectStorageUseProxyDesc: "如果您不使用代理进行API连接,请将其关闭。" | ||||
| objectStorageSetPublicRead: "上传时设置为public-read" | ||||
| serverLogs: "服务器日志" | ||||
| deleteAll: "删除全部" | ||||
| showFixedPostForm: "在时间线顶部显示帖子表单" | ||||
| @@ -475,6 +485,8 @@ newNoteRecived: "有新的帖子" | ||||
| sounds: "声音" | ||||
| listen: "听" | ||||
| none: "空" | ||||
| showInPage: "在页面中显示" | ||||
| popout: "弹窗" | ||||
| volume: "音量" | ||||
| details: "详情" | ||||
| chooseEmoji: "选择表情符号" | ||||
| @@ -490,8 +502,8 @@ state: "状态" | ||||
| sort: "排序" | ||||
| ascendingOrder: "升序" | ||||
| descendingOrder: "降序" | ||||
| scratchpad: "暂存器" | ||||
| scratchpadDescription: "暂存器为AiScript提供了实验环境。您可以编写代码以与Misskey交互,运行它并查看结果。" | ||||
| scratchpad: "便签本" | ||||
| scratchpadDescription: "便签本为AiScript提供了实验环境。您可以编写代码以与Misskey交互,运行它并查看结果。" | ||||
| output: "输出" | ||||
| script: "脚本" | ||||
| disablePagesScript: "禁用页面脚本" | ||||
| @@ -517,7 +529,6 @@ enableInfiniteScroll: "启用自动滚动页面模式" | ||||
| visibility: "可见性" | ||||
| poll: "调查问卷" | ||||
| useCw: "隐藏内容" | ||||
| fixedWidgetsPosition: "固定小工具的位置" | ||||
| enablePlayer: "打开播放器" | ||||
| disablePlayer: "关闭播放器" | ||||
| expandTweet: "展开推文" | ||||
| @@ -531,6 +542,7 @@ pluginInstallWarn: "请不要安装不明来源的插件" | ||||
| deck: "Deck" | ||||
| undeck: "取消Deck" | ||||
| useBlurEffectForModal: "模态框使用模糊效果" | ||||
| useFullReactionPicker: "使用全功能的回应工具栏" | ||||
| generateAccessToken: "生成访问令牌" | ||||
| permission: "权限" | ||||
| enableAll: "启用全部" | ||||
| @@ -539,7 +551,7 @@ tokenRequested: "允许访问账户" | ||||
| pluginTokenRequestedDescription: "此插件将能够拥有此处设置的权限" | ||||
| notificationType: "通知类型" | ||||
| edit: "编辑" | ||||
| useStarForReactionFallback: "如果回应的颜文字未知,则使用★作为代替" | ||||
| useStarForReactionFallback: "如果回应的是未知表情符号,则使用★作为代替" | ||||
| emailConfig: "邮件服务器设置" | ||||
| enableEmail: "启用发送邮件功能" | ||||
| emailConfigInfo: "用于确认电子邮件和密码重置" | ||||
| @@ -557,8 +569,144 @@ wordMute: "文字屏蔽" | ||||
| userSaysSomething: "{name}说了什么" | ||||
| makeActive: "激活" | ||||
| display: "显示" | ||||
| copy: "复制" | ||||
| metrics: "指标" | ||||
| overview: "概述" | ||||
| logs: "日志" | ||||
| delayed: "延迟" | ||||
| database: "数据库" | ||||
| channel: "频道" | ||||
| create: "创建" | ||||
| notificationSetting: "通知设置" | ||||
| notificationSettingDesc: "选择要显示的通知类型。" | ||||
| useGlobalSetting: "使用全局设置" | ||||
| useGlobalSettingDesc: "启用时,将使用帐户通知设置。关闭时,则可以单独设置。" | ||||
| other: "其他" | ||||
| regenerateLoginToken: "重新生成登录令牌" | ||||
| regenerateLoginTokenDescription: "重新生成用于登录的内部令牌。通常您不需要这样做。重新生成后,您将在所有设备上登出。" | ||||
| setMultipleBySeparatingWithSpace: "您可以使用空格分隔多个项目。" | ||||
| fileIdOrUrl: "文件ID或者URL" | ||||
| chatOpenBehavior: "聊天窗口打开时的行为" | ||||
| sample: "示例" | ||||
| abuseReports: "举报" | ||||
| reportAbuse: "举报" | ||||
| reportAbuseOf: "举报{name}" | ||||
| fillAbuseReportDescription: "请填写举报的详细原因。如果有对方发的帖子,请同时填写URL地址。" | ||||
| abuseReported: "内容已发送。感谢您的报告。" | ||||
| send: "发送" | ||||
| abuseMarkAsResolved: "处理完毕" | ||||
| openInNewTab: "在新标签页中打开" | ||||
| openInSideView: "在侧边栏中打开" | ||||
| defaultNavigationBehaviour: "默认导航" | ||||
| editTheseSettingsMayBreakAccount: "编辑这些设置可以会损坏您的账号" | ||||
| instanceTicker: "帖子的实例信息" | ||||
| waitingFor: "等待{x}" | ||||
| random: "随机" | ||||
| system: "系统" | ||||
| switchUi: "切换界面" | ||||
| desktop: "桌面" | ||||
| _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: "显示回转的动画效果。" | ||||
| _reversi: | ||||
|   reversi: "黑白棋" | ||||
|   gameSettings: "对局设置" | ||||
|   chooseBoard: "棋盘选择" | ||||
|   blackOrWhite: "先手/后手" | ||||
|   blackIs: "{name}执黑(先走)" | ||||
|   rules: "规则" | ||||
|   botSettings: "机器人设置" | ||||
|   thisGameIsStartedSoon: "对局在几秒后开始" | ||||
|   waitingForOther: "等待对手准备" | ||||
|   waitingForMe: "等待您的准备" | ||||
|   waitingBoth: "请准备" | ||||
|   ready: "准备就绪" | ||||
|   cancelReady: "重新准备" | ||||
|   opponentTurn: "对手的会合" | ||||
|   myTurn: "您的回合" | ||||
|   turnOf: "{name}的回合" | ||||
|   pastTurnOf: "{name}的回合" | ||||
|   surrender: "认输 " | ||||
|   surrendered: "对手认输" | ||||
|   drawn: "平局" | ||||
|   won: "{name}获胜" | ||||
|   black: "黑" | ||||
|   white: "白" | ||||
|   total: "总计" | ||||
|   turnCount: "{count}回合" | ||||
|   myGames: "我的对局" | ||||
|   allGames: "所有对局" | ||||
|   ended: "结束" | ||||
|   playing: "对局中" | ||||
|   isLlotheo: "棋子较少一方获胜(LLoTheO规则)" | ||||
|   loopedMap: "循环棋盘" | ||||
|   canPutEverywhere: "可以下在任意位置" | ||||
| _instanceTicker: | ||||
|   none: "不显示" | ||||
|   remote: "显示给远程用户" | ||||
|   always: "始终显示" | ||||
| _serverDisconnectedBehavior: | ||||
|   reload: "自动重载" | ||||
|   dialog: "对话框警告" | ||||
|   quiet: "安静警告" | ||||
| _channel: | ||||
|   create: "创建频道" | ||||
|   edit: "编辑频道" | ||||
|   setBanner: "设置横幅" | ||||
|   removeBanner: "删除横幅" | ||||
|   featured: "热点" | ||||
|   owned: "管理中" | ||||
|   following: "正在关注" | ||||
|   usersCount: "有{n}人参与" | ||||
|   notesCount: "有{n}个帖子" | ||||
| _sidebar: | ||||
|   icon: "头像" | ||||
|   full: "全部" | ||||
|   icon: "图标" | ||||
|   hide: "隐藏" | ||||
| _wordMute: | ||||
|   muteWords: "禁用词" | ||||
| @@ -566,6 +714,9 @@ _wordMute: | ||||
|   muteWordsDescription2: "将关键字用斜线括起来表示正则表达式。" | ||||
|   softDescription: "隐藏时间轴中指定条件的帖文。" | ||||
|   hardDescription: "防止将具有指定条件的帖文添加到时间线。 即使您更改条件,未添加的帖文也会被排除在外。" | ||||
|   soft: "软屏蔽" | ||||
|   hard: "硬屏蔽" | ||||
|   mutedNotes: "被屏蔽的帖文" | ||||
| _theme: | ||||
|   explore: "寻找主题" | ||||
|   install: "安装主题" | ||||
| @@ -625,7 +776,7 @@ _theme: | ||||
|     cwFg: "CW 按钮文本" | ||||
|     cwHoverBg: "CW 按钮背景(悬停)" | ||||
|     toastBg: "吐司提示背景" | ||||
|     toastFg: "土司提示文本" | ||||
|     toastFg: "吐司提示文本" | ||||
|     buttonBg: "按钮背景" | ||||
|     buttonHoverBg: "按钮背景(悬停)" | ||||
|     inputBorder: "输入框边框" | ||||
| @@ -644,6 +795,7 @@ _sfx: | ||||
|   chat: "聊天" | ||||
|   chatBg: "聊天背景" | ||||
|   antenna: "天线接收" | ||||
|   channel: "频道通知" | ||||
| _ago: | ||||
|   unknown: "未知" | ||||
|   future: "未来" | ||||
| @@ -680,7 +832,7 @@ _tutorial: | ||||
|   step6_1: "现在,您将可以在时间线上看到其他用户的帖子。" | ||||
|   step6_2: "您还可以在其他人的帖子上进行「回应」,以快速做出简单回复。" | ||||
|   step6_3: "在他人的贴子上按下「+」图标,即可选择想要的表情来进行「回应」。" | ||||
|   step7_1: "对Misskey基本操作的简单介绍,到此结束了。 辛苦了!" | ||||
|   step7_1: "对Misskey基本操作的简单介绍,就到此结束了。 辛苦了!" | ||||
|   step7_2: "如果你想了解更多有关Misskey的信息,请参见{help}。" | ||||
|   step7_3: "接下来,享受Misskey带来的乐趣吧🚀" | ||||
| _2fa: | ||||
| @@ -691,7 +843,7 @@ _2fa: | ||||
|   step2: "然后,扫描屏幕上显示的二维码。" | ||||
|   step3: "输入您的应用提供的动态口令以完成设置。" | ||||
|   step4: "从现在开始,任何登录操作都将要求您提供动态口令。" | ||||
|   securityKeyInfo: "您可以设置使用支持FIDO2的硬件安全密钥、指纹或设备上的PIN来保护您的登录过程。" | ||||
|   securityKeyInfo: "您可以设置使用支持FIDO2的硬件安全密钥、设备上的指纹或PIN来保护您的登录过程。" | ||||
| _permissions: | ||||
|   "read:account": "查看账户信息" | ||||
|   "write:account": "更改帐户信息" | ||||
| @@ -719,6 +871,8 @@ _permissions: | ||||
|   "write:page-likes": "操作喜欢的页面" | ||||
|   "read:user-groups": "查看用户组" | ||||
|   "write:user-groups": "操作用户组" | ||||
|   "read:channels": "查看频道" | ||||
|   "write:channels": "管理频道" | ||||
| _auth: | ||||
|   shareAccess: "您要授权允许“{name}”访问您的帐户吗?" | ||||
|   shareAccessAsk: "您确定要授权此应用程序访问您的帐户吗?" | ||||
| @@ -752,6 +906,7 @@ _widgets: | ||||
|   photos: "照片" | ||||
|   digitalClock: "数字时钟" | ||||
|   federation: "联邦宇宙" | ||||
|   postForm: "投稿窗口" | ||||
| _cw: | ||||
|   hide: "隐藏" | ||||
|   show: "查看更多" | ||||
| @@ -793,6 +948,7 @@ _visibility: | ||||
| _postForm: | ||||
|   replyPlaceholder: "回复这个帖子..." | ||||
|   quotePlaceholder: "引用这个帖子..." | ||||
|   channelPlaceholder: "发布到频道…" | ||||
|   _placeholders: | ||||
|     a: "现在如何?" | ||||
|     b: "发生了什么?" | ||||
| @@ -925,6 +1081,7 @@ _pages: | ||||
|   my: "我的页面" | ||||
|   liked: "喜欢的页面" | ||||
|   inspector: "检查器" | ||||
|   contents: "内容" | ||||
|   content: "页面内容" | ||||
|   variables: "变量" | ||||
|   title: "标题" | ||||
| @@ -1213,8 +1370,11 @@ _notification: | ||||
|     renote: "转发" | ||||
|     quote: "引用" | ||||
|     reaction: "回应" | ||||
|     pollVote: "投票" | ||||
|     receiveFollowRequest: "关注请求" | ||||
|     pollVote: "问卷调查已投票" | ||||
|     receiveFollowRequest: "收到关注请求" | ||||
|     followRequestAccepted: "关注请求已接受" | ||||
|     groupInvited: "加入群组邀请" | ||||
|     app: "关联应用的通知" | ||||
| _deck: | ||||
|   alwaysShowMainColumn: "总是显示主列" | ||||
|   columnAlign: "列对齐" | ||||
|   | ||||
| @@ -1,21 +1,24 @@ | ||||
| --- | ||||
| _lang_: "中文(繁體)" | ||||
| _lang_: "繁體中文" | ||||
| introMisskey: "歡迎! Misskey是一個開源的去中心化的社群網站。\n通過「貼文」來分享現在發生的事情吧! 📡\n「反應」功能,可以讓你快速的對大家的「帖子」來表達感情👍\n一起來探索新的世界吧! 🚀" | ||||
| monthAndDay: "{month}月 {day}日" | ||||
| search: "搜尋" | ||||
| notifications: "通知" | ||||
| username: "使用名稱" | ||||
| username: "使用者名稱" | ||||
| password: "密碼" | ||||
| fetchingAsApObject: "從 Fediverse 查詢中..." | ||||
| ok: "確定" | ||||
| ok: "OK" | ||||
| gotIt: "知道了" | ||||
| cancel: "取消" | ||||
| enterUsername: "輸入使用者名稱" | ||||
| renotedBy: "由{user}轉發" | ||||
| renotedBy: "{user} 轉發了" | ||||
| noNotes: "貼文不可用。" | ||||
| noNotifications: "沒有通知" | ||||
| instance: "實例" | ||||
| settings: "設定" | ||||
| basicSettings: "基本設定" | ||||
| otherSettings: "其他設定" | ||||
| openInWindow: "在新視窗開啟" | ||||
| profile: "個人檔案" | ||||
| timeline: "時間軸" | ||||
| noAccountDescription: "此用戶還沒有自我介紹" | ||||
| @@ -40,6 +43,7 @@ deleteAndEditConfirm: "要刪除並再次編輯嗎?此貼文的所有反應, | ||||
| addToList: "添加至清單" | ||||
| sendMessage: "發送訊息" | ||||
| copyUsername: "複製用戶名" | ||||
| searchUser: "搜尋用戶" | ||||
| reply: "回覆" | ||||
| loadMore: "瀏覽更多" | ||||
| youGotNewFollower: "您有新的追隨者" | ||||
| @@ -47,6 +51,7 @@ receiveFollowRequest: "收到追隨請求" | ||||
| followRequestAccepted: "追隨請求已接受" | ||||
| mention: "提及" | ||||
| mentions: "提及" | ||||
| directNotes: "私信" | ||||
| importAndExport: "匯入 / 匯出" | ||||
| import: "匯入" | ||||
| export: "匯出" | ||||
| @@ -65,8 +70,10 @@ followers: "追隨者" | ||||
| followsYou: "追隨你的人" | ||||
| createList: "建立清單" | ||||
| manageLists: "管理清單" | ||||
| error: "發生錯誤" | ||||
| error: "錯誤" | ||||
| somethingHappened: "發生錯誤" | ||||
| retry: "重試" | ||||
| pageLoadError: "載入頁面失敗" | ||||
| enterListName: "輸入清單名稱" | ||||
| privacy: "隱私" | ||||
| makeFollowManuallyApprove: "手動審核追隨請求" | ||||
| @@ -92,8 +99,8 @@ attachCancel: "移除附件" | ||||
| markAsSensitive: "標記為敏感內容" | ||||
| unmarkAsSensitive: "取消標記為敏感內容" | ||||
| enterFileName: "請輸入檔案名稱" | ||||
| mute: "消音" | ||||
| unmute: "解除消音" | ||||
| mute: "靜音" | ||||
| unmute: "解除靜音" | ||||
| block: "封鎖" | ||||
| unblock: "解除封鎖" | ||||
| suspend: "凍結" | ||||
| @@ -103,17 +110,24 @@ unblockConfirm: "確定解除封鎖此用戶?" | ||||
| suspendConfirm: "確定凍結此帳號?" | ||||
| unsuspendConfirm: "確定解凍此帳號?" | ||||
| selectList: "選擇清單" | ||||
| selectAntenna: "選擇天線" | ||||
| selectWidget: "選擇小工具" | ||||
| editWidgets: "編輯小工具" | ||||
| editWidgetsExit: "停止編輯" | ||||
| customEmojis: "自訂表情符號" | ||||
| emoji: "表情符號" | ||||
| emojiName: "表情符號名稱" | ||||
| emojiUrl: "表情符號URL" | ||||
| addEmoji: "新增表情符號" | ||||
| settingGuide: "推薦設定" | ||||
| flagAsBot: "此帳戶是Bot" | ||||
| flagAsCat: "此帳戶是Cat" | ||||
| cacheRemoteFiles: "緩存非遠程檔案" | ||||
| cacheRemoteFilesDescription: "禁用此設定會停止遠端檔案的緩存,從而節省儲存空間。但資料會因直接連線從而產生額外連接數據。" | ||||
| flagAsBot: "此使用者是機器人" | ||||
| flagAsCat: "此使用者是貓" | ||||
| autoAcceptFollowed: "自動許可追隨" | ||||
| addAcount: "新增帳號" | ||||
| loginFailed: "登入失敗" | ||||
| showOnRemote: "轉到所在實例顯示" | ||||
| general: "一般" | ||||
| wallpaper: "桌布" | ||||
| setWallpaper: "設定桌布" | ||||
| @@ -122,12 +136,16 @@ searchWith: "搜尋: {q}" | ||||
| youHaveNoLists: "沒有任何清單" | ||||
| followConfirm: "你真的要關注{name}嗎?" | ||||
| proxyAccount: "代理帳號" | ||||
| proxyAccountDescription: "代理帳號是在某些情況下充當其他服務器用戶的帳號。例如,當用戶將一個來自其他服務器的帳號放在列表中時,由於沒有其他用戶關注該帳號,該指令不會傳送到該服務器上,因此會由代理帳戶關注。" | ||||
| host: "主機" | ||||
| selectUser: "選取使用者" | ||||
| recipient: "發送至" | ||||
| annotation: "註解" | ||||
| federation: "聯邦宇宙" | ||||
| instances: "實例" | ||||
| registeredAt: "初次觀測" | ||||
| latestRequestSentAt: "上次發送的請求" | ||||
| latestRequestReceivedAt: "上次收到的請求" | ||||
| latestStatus: "最後狀態" | ||||
| storageUsage: "已使用容量" | ||||
| charts: "圖表" | ||||
| @@ -148,12 +166,13 @@ instanceInfo: "實例資訊" | ||||
| statistics: "統計" | ||||
| clearQueue: "清除佇列" | ||||
| clearQueueConfirmTitle: "確定要清除佇列嗎?" | ||||
| clearQueueConfirmText: "未發佈的帖子將不會發佈。您通常不需要確認。" | ||||
| clearCachedFiles: "清除快取資料" | ||||
| clearCachedFilesConfirm: "確定要清除緩存資料嗎?" | ||||
| blockedInstances: "已封鎖的實例" | ||||
| blockedInstancesDescription: "請逐行輸入需要封鎖的實例。已封鎖的實例將無法與本實例進行通訊。" | ||||
| muteAndBlock: "禁言 / 封鎖" | ||||
| mutedUsers: "已禁言用戶" | ||||
| muteAndBlock: "靜音/封鎖" | ||||
| mutedUsers: "已靜音用戶" | ||||
| blockedUsers: "已封鎖用戶" | ||||
| noUsers: "無用戶" | ||||
| editProfile: "編輯個人檔案" | ||||
| @@ -165,7 +184,6 @@ processing: "處理中" | ||||
| preview: "預覽" | ||||
| default: "預設" | ||||
| noCustomEmojis: "沒有表情符號" | ||||
| customEmojisOfRemote: "來自其他實例的表情符號" | ||||
| noJobs: "沒有任務" | ||||
| federating: "整合搜索中" | ||||
| blocked: "已封鎖" | ||||
| @@ -194,12 +212,15 @@ imageUrl: "圖片URL" | ||||
| remove: "刪除" | ||||
| removed: "成功移除" | ||||
| removeAreYouSure: "確定要刪掉「{x}」嗎?" | ||||
| deleteAreYouSure: "確定要刪掉「{x}」嗎?" | ||||
| saved: "已保存" | ||||
| messaging: "傳送訊息" | ||||
| upload: "上傳" | ||||
| fromDrive: "從雲端" | ||||
| fromUrl: "從URL" | ||||
| uploadFromUrl: "從網址上傳" | ||||
| uploadFromUrlDescription: "您要上傳的文件的URL" | ||||
| uploadFromUrlRequested: "已請求上傳" | ||||
| uploadFromUrlMayTakeTime: "還需要一些時間才能完成上傳。" | ||||
| explore: "探索" | ||||
| games: "Misskey 遊戲" | ||||
| @@ -207,9 +228,11 @@ messageRead: "已讀" | ||||
| noMoreHistory: "沒有更多歷史紀錄" | ||||
| startMessaging: "開始傳送訊息" | ||||
| nUsersRead: "{n}人已讀" | ||||
| agreeTo: "我同意{0}" | ||||
| tos: "使用條款" | ||||
| start: "開始" | ||||
| home: "首頁" | ||||
| remoteUserCaution: "由於該用戶來自遠端實例,因此資料用戶並未即時更新。" | ||||
| activity: "動態" | ||||
| images: "圖片" | ||||
| birthday: "生日" | ||||
| @@ -217,10 +240,13 @@ yearsOld: "{age}歲" | ||||
| registeredDate: "註冊日期" | ||||
| location: "位置" | ||||
| theme: "外觀主題" | ||||
| themeForLightMode: "在淺色模式下使用的主題" | ||||
| themeForDarkMode: "在深色模式下使用的主題" | ||||
| light: "淺色" | ||||
| dark: "灰暗" | ||||
| lightThemes: "明亮主題" | ||||
| darkThemes: "灰暗主題" | ||||
| syncDeviceDarkMode: "將深色模式與設備設置同步" | ||||
| drive: "雲端硬碟" | ||||
| fileName: "檔案名稱" | ||||
| selectFile: "選擇檔案" | ||||
| @@ -238,11 +264,14 @@ emptyFolder: "空的資料夾" | ||||
| unableToDelete: "無法刪除" | ||||
| inputNewFileName: "輸入檔案名稱" | ||||
| inputNewFolderName: "輸入新資料夾的名稱" | ||||
| circularReferenceFolder: "目標文件夾是您要移動的文件夾的子文件夾。" | ||||
| hasChildFilesOrFolders: "此文件夾不是空的,無法刪除。" | ||||
| copyUrl: "複製URL" | ||||
| rename: "重新命名" | ||||
| avatar: "頭像" | ||||
| banner: "橫幅" | ||||
| nsfw: "敏感內容" | ||||
| whenServerDisconnected: "與服務器的連接中斷時" | ||||
| disconnectedFromServer: "與伺服器中斷連線" | ||||
| reload: "重新載入" | ||||
| doNothing: "無視" | ||||
| @@ -269,10 +298,12 @@ connectSerice: "連線" | ||||
| disconnectSerice: "中斷連線" | ||||
| enableLocalTimeline: "開啟本地時間軸" | ||||
| enableGlobalTimeline: "開啟全球時間軸" | ||||
| disablingTimelinesInfo: "即使您禁用了時間線功能,管理員和協調人仍可以繼續使用,以方便您。" | ||||
| registration: "註冊" | ||||
| enableRegistration: "開啟新用戶註冊" | ||||
| invite: "邀請" | ||||
| proxyRemoteFiles: "代理遠程檔案" | ||||
| proxyRemoteFiles: "遠端代理檔案" | ||||
| proxyRemoteFilesDescription: "啟用此設置後,由於超出存儲容量而未保存或刪除的遠程文件將被本地代理,並且將生成預覽圖。這不影響服務器的存儲。" | ||||
| driveCapacityPerLocalAccount: "每個本地用戶的雲端容量" | ||||
| driveCapacityPerRemoteAccount: "每個非本地用戶的雲端容量" | ||||
| inMb: "以Mbps為單位" | ||||
| @@ -280,6 +311,7 @@ iconUrl: "圖像URL" | ||||
| bannerUrl: "橫幅圖片URL" | ||||
| basicInfo: "基本資訊" | ||||
| pinnedUsers: "置頂用戶" | ||||
| pinnedUsersDescription: "在「發現」頁面中使用換行標記想要置頂的用戶。" | ||||
| hcaptcha: "hCaptcha" | ||||
| enableHcaptcha: "啟用 hCaptcha" | ||||
| hcaptchaSiteKey: "網站金鑰" | ||||
| @@ -288,13 +320,21 @@ recaptcha: "reCAPTCHA" | ||||
| enableRecaptcha: "啟用 reCAPTCHA" | ||||
| recaptchaSiteKey: "網站金鑰" | ||||
| recaptchaSecretKey: "金鑰" | ||||
| avoidMultiCaptchaConfirm: "使用多種驗證方式可能會造成干擾,您要禁用其他驗證方式嗎?您可以按“取消”保留多種驗證方式。" | ||||
| antennas: "天線" | ||||
| manageAntennas: "管理天線" | ||||
| name: "名稱" | ||||
| antennaKeywords: "包含的關鍵字" | ||||
| antennaSource: "接收來源" | ||||
| antennaKeywords: "包含關鍵字" | ||||
| antennaExcludeKeywords: "排除關鍵字" | ||||
| antennaKeywordsDescription: "用空格分隔指定AND、用換行符分隔指定OR" | ||||
| notifyAntenna: "通知我有新的貼文" | ||||
| serviceworker: "ServiceWorker" | ||||
| enableServiceworker: "開啟 ServiceWorker" | ||||
| antennaUsersDescription: "指定用換行符分隔的用戶名" | ||||
| caseSensitive: "區分大小寫" | ||||
| withReplies: "包含回覆" | ||||
| connectedTo: "您的帳號已連接到以下社交帳號" | ||||
| notesAndReplies: "貼文與回覆" | ||||
| withFiles: "附件" | ||||
| silence: "禁言" | ||||
| @@ -311,6 +351,9 @@ popularTags: "熱門標籤" | ||||
| userList: "清單" | ||||
| about: "資訊" | ||||
| aboutMisskey: "關於 Misskey" | ||||
| aboutMisskeyText: "Misskey是由syuilo於2014年開發的開放源代碼軟件。" | ||||
| misskeyMembers: "現在由以下成員開發及維護:" | ||||
| misskeySource: "源代碼在這裡公開:" | ||||
| misskeyTranslation: "幫助我們為Misskey的翻譯工作出一分力:" | ||||
| misskeyDonate: "向Misskey捐款以支援我們開發工作:" | ||||
| morePatrons: "感激你們的支持、 幫助。 🥰" | ||||
| @@ -328,9 +371,6 @@ unregister: "刪除賬戶" | ||||
| passwordLessLogin: "設置無密碼登入" | ||||
| resetPassword: "重置密碼" | ||||
| newPasswordIs: "新密碼為「{password}」" | ||||
| autoReloadWhenDisconnected: "和伺服器斷線時自動重新載入" | ||||
| autoNoteWatch: "自動追隨貼文" | ||||
| autoNoteWatchDescription: "收到反應或回覆過的貼文的通知" | ||||
| reduceUiAnimation: "減少介面的動態視覺" | ||||
| share: "分享" | ||||
| notFound: "找不到" | ||||
| @@ -346,6 +386,7 @@ close: "關閉" | ||||
| group: "群組" | ||||
| groups: "群組" | ||||
| createGroup: "創建群組" | ||||
| ownedGroups: "擁有的群組" | ||||
| joinedGroups: "群組成員" | ||||
| invites: "邀請" | ||||
| groupName: "群組名稱" | ||||
| @@ -367,6 +408,7 @@ noMessagesYet: "沒有訊息" | ||||
| newMessageExists: "有新的訊息" | ||||
| onlyOneFileCanBeAttached: "只能添加一個附件" | ||||
| signinRequired: "請先登入" | ||||
| invitations: "邀請" | ||||
| invitationCode: "邀請碼" | ||||
| checking: "確認中" | ||||
| available: "可用的" | ||||
| @@ -379,16 +421,53 @@ normalPassword: "密碼強度普通" | ||||
| strongPassword: "密碼強度堅強" | ||||
| passwordMatched: "密碼一致" | ||||
| passwordNotMatched: "密碼不一致" | ||||
| signinWith: "以{x}登錄" | ||||
| signinFailed: "登入失敗。 請檢查用戶名和密碼。" | ||||
| tapSecurityKey: "點擊安全密鑰" | ||||
| or: "或者" | ||||
| uiLanguage: "介面語言" | ||||
| groupInvited: "您有新的群組邀請" | ||||
| aboutX: "關於{x}" | ||||
| useOsNativeEmojis: "使用OS原生表情符號" | ||||
| youHaveNoGroups: "找不到群組" | ||||
| joinOrCreateGroup: "請加入現有群組,或創建新群組。" | ||||
| noHistory: "沒有歷史紀錄" | ||||
| disableAnimatedMfm: "禁用MFM動畫" | ||||
| doing: "正在進行" | ||||
| category: "類別" | ||||
| tags: "標籤" | ||||
| docSource: "文件來源" | ||||
| createAccount: "建立帳戶" | ||||
| existingAcount: "現有帳戶" | ||||
| regenerate: "再生" | ||||
| fontSize: "字體大小" | ||||
| openImageInNewTab: "於新分頁中開啟圖片" | ||||
| local: "本地" | ||||
| remote: "遠端" | ||||
| total: "合計" | ||||
| clinetSettings: "用戶端設定" | ||||
| weekOverWeekChanges: "與上週相比" | ||||
| dayOverDayChanges: "與前一日相比" | ||||
| appearance: "外觀" | ||||
| clientSettings: "用戶端設定" | ||||
| accountSettings: "帳號設定" | ||||
| promotion: "推廣貼文" | ||||
| promote: "推廣" | ||||
| numberOfDays: "有效天數" | ||||
| hideThisNote: "隱藏此貼文" | ||||
| showFeaturedNotesInTimeline: "在時間軸上顯示熱門推薦" | ||||
| objectStorageBaseUrl: "Base URL" | ||||
| objectStorageBucket: "儲存空間(Bucket)" | ||||
| objectStoragePrefix: "前綴" | ||||
| objectStorageEndpoint: "訪問網域名稱(Endpoint)" | ||||
| objectStorageEndpointDesc: "如要使用AWS S3,請留空。否則請根據伺服器要求以'<host>'或 '<host>:<port>'的形式設定訪問網域名稱(Endpoint)。" | ||||
| objectStorageRegion: "地域(Region)" | ||||
| objectStorageUseSSL: "使用SSL" | ||||
| objectStorageUseProxy: "使用網路代理" | ||||
| serverLogs: "伺服器日誌" | ||||
| deleteAll: "刪除所有記錄" | ||||
| sounds: "音效" | ||||
| none: "無" | ||||
| showInPage: "在頁面中顯示" | ||||
| volume: "音量" | ||||
| details: "詳細資訊" | ||||
| chooseEmoji: "選擇您的表情符號\n" | ||||
| @@ -396,31 +475,163 @@ unableToProcess: "操作無法完成" | ||||
| recentUsed: "最近使用" | ||||
| install: "安裝" | ||||
| uninstall: "解除安裝" | ||||
| installedApps: "已授權的應用程式" | ||||
| nothing: "未發現" | ||||
| installedDate: "安裝時間" | ||||
| lastUsedDate: "最後上線日期" | ||||
| state: "狀態" | ||||
| sort: "排序" | ||||
| ascendingOrder: "昇冪" | ||||
| descendingOrder: "降冪" | ||||
| scratchpad: "暫存記憶體" | ||||
| output: "輸出" | ||||
| script: "腳本" | ||||
| updateRemoteUser: "更新非本地用戶資料" | ||||
| deleteAllFiles: "刪除所有檔案" | ||||
| deleteAllFilesConfirm: "要删除所有檔案吗?" | ||||
| removeAllFollowing: "解除所有追隨" | ||||
| userSuspended: "該用戶已被凍結" | ||||
| userSilenced: "該用戶已被禁言。" | ||||
| sidebar: "側邊列" | ||||
| divider: "分割線" | ||||
| addItem: "新增項目" | ||||
| rooms: "房間" | ||||
| relays: "中繼" | ||||
| addRelay: "添加中繼" | ||||
| inboxUrl: "私信URL" | ||||
| addedRelays: "已添加的中繼" | ||||
| serviceworkerInfo: "您需要啟用推送通知" | ||||
| deletedNote: "已删除的貼文" | ||||
| invisibleNote: "隱藏的帖子" | ||||
| enableInfiniteScroll: "啟用自動滾動頁面模式" | ||||
| visibility: "公開範圍" | ||||
| poll: "投票" | ||||
| useCw: "隱藏內容" | ||||
| enablePlayer: "打開播放器" | ||||
| disablePlayer: "關閉播放器" | ||||
| expandTweet: "展開推文" | ||||
| themeEditor: "主題編輯器" | ||||
| description: "描述" | ||||
| author: "作者" | ||||
| leaveConfirm: "有未保存的更改。要放棄嗎?" | ||||
| manage: "管理" | ||||
| plugins: "插件" | ||||
| pluginInstallWarn: "請不要安裝來源不明的插件。" | ||||
| deck: "多欄模式" | ||||
| undeck: "取消多欄模式" | ||||
| permission: "權限" | ||||
| enableAll: "啟用全部" | ||||
| disableAll: "停用全部" | ||||
| tokenRequested: "允許訪問帳號" | ||||
| notificationType: "通知形式" | ||||
| edit: "編輯" | ||||
| useStarForReactionFallback: "以★代替未知的表情符號" | ||||
| emailConfig: "電子郵件伺服器設定" | ||||
| enableEmail: "啟用發送電郵功能" | ||||
| emailConfigInfo: "用於確認電郵地址及密碼重置" | ||||
| email: "電郵地址" | ||||
| smtpConfig: "SMTP伺服器設定" | ||||
| smtpHost: "主機" | ||||
| smtpUser: "使用名稱" | ||||
| smtpPort: "端口" | ||||
| smtpUser: "使用者名稱" | ||||
| smtpPass: "密碼" | ||||
| emptyToDisableSmtpAuth: "留空使用者名稱和密碼以禁用SMTP驗證。" | ||||
| testEmail: "郵件測試發送" | ||||
| wordMute: "靜音文字" | ||||
| display: "檢視" | ||||
| copy: "複製" | ||||
| metrics: "指標" | ||||
| overview: "概覽" | ||||
| logs: "日誌" | ||||
| delayed: "延遲" | ||||
| database: "資料庫" | ||||
| channel: "頻道" | ||||
| create: "新增" | ||||
| notificationSetting: "通知設定" | ||||
| other: "其他" | ||||
| regenerateLoginTokenDescription: "再生用於登入的內部權杖。一般情況下是不需要這樣做的。一旦再生,所有裝置將會被登出。" | ||||
| sample: "範例 " | ||||
| abuseReports: "檢舉" | ||||
| reportAbuse: "檢舉" | ||||
| reportAbuseOf: "檢舉{name}" | ||||
| send: "發送" | ||||
| openInNewTab: "在新分頁中開啟" | ||||
| random: "隨機" | ||||
| system: "系統" | ||||
| _mfm: | ||||
|   mention: "提及" | ||||
|   hashtag: "#tag" | ||||
|   link: "鏈接" | ||||
|   quote: "引用" | ||||
|   emoji: "自訂表情符號" | ||||
|   search: "搜尋" | ||||
| _reversi: | ||||
|   reversi: "黑白棋" | ||||
|   gameSettings: "對弈設定" | ||||
|   chooseBoard: "選擇棋盤" | ||||
|   rules: "規則" | ||||
|   botSettings: "機器人設定" | ||||
|   opponentTurn: "對手回合" | ||||
|   myTurn: "你的回合" | ||||
|   turnOf: "{name}的回合" | ||||
|   pastTurnOf: "{name}的回合" | ||||
|   surrender: "認輸" | ||||
|   black: "黑" | ||||
|   white: "白" | ||||
|   total: "合計" | ||||
|   ended: "已結束" | ||||
|   playing: "正在對弈" | ||||
| _instanceTicker: | ||||
|   always: "總是顯示" | ||||
| _serverDisconnectedBehavior: | ||||
|   reload: "自動重載" | ||||
|   dialog: "以對話框警告" | ||||
|   quiet: "適當地警告" | ||||
| _channel: | ||||
|   create: "建立頻道" | ||||
|   edit: "編輯頻道" | ||||
|   setBanner: "設定橫幅" | ||||
|   removeBanner: "移除封面圖" | ||||
|   featured: "流行" | ||||
|   owned: "管理中" | ||||
|   following: "關注中" | ||||
|   usersCount: "有{n}人參與" | ||||
|   notesCount: "有{n}個帖子" | ||||
| _sidebar: | ||||
|   icon: "頭像" | ||||
|   hide: "隱藏" | ||||
| _wordMute: | ||||
|   muteWords: "加入靜音文字" | ||||
|   mutedNotes: "已靜音的貼文" | ||||
| _theme: | ||||
|   constant: "常數" | ||||
|   defaultValue: "預設值" | ||||
|   color: "顏色" | ||||
|   func: "函数" | ||||
|   argument: "引數" | ||||
|   alpha: "透明度" | ||||
|   darken: "暗度" | ||||
|   lighten: "亮度" | ||||
|   keys: | ||||
|     bg: "背景" | ||||
|     fg: "文本" | ||||
|     shadow: "陰影" | ||||
|     link: "鏈接" | ||||
|     hashtag: "#tag" | ||||
|     mention: "提及" | ||||
|     mentionMe: "提及我" | ||||
|     renote: "轉發貼文" | ||||
|     divider: "分割線" | ||||
|     infoBg: "資訊背景" | ||||
|     infoFg: "資訊內容" | ||||
|     infoWarnBg: "警告背景" | ||||
|     infoWarnFg: "警告字元" | ||||
| _sfx: | ||||
|   note: "貼文" | ||||
|   noteMy: "我的貼文" | ||||
|   notification: "通知" | ||||
|   chat: "傳送訊息" | ||||
|   channel: "頻道通知" | ||||
| _ago: | ||||
|   unknown: "未知" | ||||
|   future: "未來" | ||||
| @@ -451,6 +662,7 @@ _tutorial: | ||||
|   step4_1: "筆記發出去了嗎?" | ||||
|   step4_2: "如果你的貼文有顯示在時間軸上,就代表已經發文成功。" | ||||
|   step5_1: "現在試試看追隨其他人來讓你的時間軸變得更生動吧。" | ||||
|   step5_2: "你可以在{featured}上看到受歡迎的貼文,你也可以選擇從列表中追隨你喜歡的人,或者在{explore}上找到熱門使用者。" | ||||
|   step5_3: "想要追隨其他人,只要點擊他們的頭像並按「追隨」即可。" | ||||
|   step5_4: "如果使用者的名字旁有鎖頭的圖示,代表他們需要手動核准你的追隨請求。" | ||||
|   step6_1: "現在你可以在時間軸上看到其他用戶的貼文" | ||||
| @@ -458,6 +670,8 @@ _tutorial: | ||||
|   step6_3: "在他人的貼文按下「+」的圖示即可選擇想要的表情符號來進行「反應」。" | ||||
|   step7_1: "以上為Misskey的基本操作說明,教學在此告一段落。辛苦了。" | ||||
|   step7_2: "歡迎到{help}來瞭解更多Misskey相關介紹。" | ||||
| _2fa: | ||||
|   registerDevice: "註冊裝置" | ||||
| _permissions: | ||||
|   "read:blocks": "已封鎖用戶名單" | ||||
|   "write:blocks": "編輯已封鎖用戶名單" | ||||
| @@ -466,11 +680,32 @@ _permissions: | ||||
|   "read:favorites": "瀏覽已收藏" | ||||
|   "write:favorites": "編輯收藏清單" | ||||
|   "write:following": "追隨/解除追隨" | ||||
|   "read:messaging": "顯示訊息" | ||||
|   "write:messaging": "撰寫或刪除私人訊息" | ||||
|   "read:mutes": "顯示已靜音列表" | ||||
|   "write:mutes": "編輯已靜音列表" | ||||
|   "write:notes": "撰寫或刪除貼文" | ||||
|   "read:notifications": "查看通知" | ||||
|   "write:notifications": "編輯通知" | ||||
|   "read:reactions": "查看反應" | ||||
|   "write:reactions": "編輯反應" | ||||
|   "write:votes": "投票" | ||||
|   "read:pages": "顯示頁面" | ||||
|   "write:pages": "編輯頁面" | ||||
|   "read:page-likes": "顯示頁面的已喜歡" | ||||
|   "write:page-likes": "編輯頁面上喜歡" | ||||
|   "read:user-groups": "顯示使用者群組" | ||||
|   "write:user-groups": "編輯使用者群組" | ||||
|   "read:channels": "已查看的頻道" | ||||
|   "write:channels": "編輯頻道" | ||||
| _auth: | ||||
|   shareAccess: "要授權「“{name}”」存取您的帳戶嗎?" | ||||
| _antennaSources: | ||||
|   all: "全部貼文" | ||||
|   homeTimeline: "來自已追隨使用者的貼文" | ||||
|   users: "來自特定使用者的貼文" | ||||
|   userList: "來自特定清單中的貼文" | ||||
|   userGroup: "來自特定群組的貼文" | ||||
| _weekday: | ||||
|   sunday: "週日" | ||||
|   monday: "週一" | ||||
| @@ -480,61 +715,236 @@ _weekday: | ||||
|   friday: "週五" | ||||
|   saturday: "週六" | ||||
| _widgets: | ||||
|   memo: "備忘錄" | ||||
|   notifications: "通知" | ||||
|   timeline: "時間軸" | ||||
|   calendar: "行事曆" | ||||
|   trends: "發燒貼文" | ||||
|   clock: "時鐘" | ||||
|   rss: "RSS閱讀器" | ||||
|   activity: "動態" | ||||
|   photos: "照片" | ||||
|   digitalClock: "電子時鐘" | ||||
|   federation: "聯邦宇宙" | ||||
| _cw: | ||||
|   hide: "隱藏" | ||||
|   show: "瀏覽更多" | ||||
|   chars: "{count}字元" | ||||
|   files: "{count} 個檔案" | ||||
| _poll: | ||||
|   noOnlyOneChoice: "至少需要兩個選項。" | ||||
|   expiration: "期限" | ||||
|   infinite: "無期限" | ||||
|   at: "結束時間" | ||||
|   deadlineDate: "截止日期" | ||||
|   deadlineTime: "小時" | ||||
|   duration: "時長" | ||||
|   votesCount: "{n}票" | ||||
|   totalVotes: "一共{n}票" | ||||
|   vote: "投票" | ||||
|   showResult: "顯示結果" | ||||
|   voted: "已投票" | ||||
|   closed: "已結束" | ||||
|   remainingDays: "{d}天{h}小時後結束" | ||||
| _visibility: | ||||
|   public: "公開" | ||||
|   home: "首頁" | ||||
|   followers: "追隨者" | ||||
|   specified: "指定使用者" | ||||
|   specifiedDescription: "僅發送至指定使用者" | ||||
|   localOnly: "僅限本地" | ||||
|   localOnlyDescription: "對遠端使用者隱藏" | ||||
| _postForm: | ||||
|   replyPlaceholder: "回覆此貼文..." | ||||
|   quotePlaceholder: "引用此貼文..." | ||||
|   channelPlaceholder: "發佈到頻道" | ||||
|   _placeholders: | ||||
|     a: "今天過得如何?" | ||||
|     b: "有什麼新鮮事嗎?" | ||||
|     c: "有什麼新鮮想法嗎?" | ||||
|     d: "想要發布些什麼嗎?" | ||||
|     e: "寫些什麼吧..." | ||||
|     f: "期待你發佈的內容..." | ||||
| _profile: | ||||
|   name: "名稱" | ||||
|   username: "使用名稱" | ||||
|   username: "使用者名稱" | ||||
|   description: "關於我" | ||||
|   youCanIncludeHashtags: "你也可以在「關於我」中加上 #tag" | ||||
|   metadata: "更多資訊" | ||||
|   metadataLabel: "標籤" | ||||
|   metadataContent: "内容" | ||||
| _exportOrImport: | ||||
|   allNotes: "全部貼文" | ||||
|   followingList: "追隨中" | ||||
|   muteList: "消音" | ||||
|   muteList: "靜音" | ||||
|   blockingList: "封鎖" | ||||
|   userLists: "清單" | ||||
| _charts: | ||||
|   usersIncDec: "使用者増減" | ||||
|   usersTotal: "使用者合共" | ||||
|   activeUsers: "活躍使用者" | ||||
|   notesIncDec: "貼文増減" | ||||
|   localNotesIncDec: "本地貼文増減" | ||||
|   remoteNotesIncDec: "非本地貼文的數目增减" | ||||
|   notesTotal: "貼文合共" | ||||
|   filesIncDec: "檔案増減" | ||||
|   filesTotal: "累計檔案" | ||||
|   storageUsageIncDec: "儲存空間的増減" | ||||
|   storageUsageTotal: "已使用的儲存空間合共" | ||||
| _instanceCharts: | ||||
|   requests: "請求" | ||||
|   users: "使用者増減" | ||||
|   usersTotal: "總計使用者" | ||||
|   notes: "貼文増減" | ||||
|   notesTotal: "累計貼文" | ||||
|   ff: "追隨/追隨者的増減" | ||||
|   ffTotal: "追隨/追隨者累計" | ||||
|   cacheSize: "增加或減少快取用量" | ||||
|   cacheSizeTotal: "快取大小總計" | ||||
|   files: "檔案數量的増減" | ||||
|   filesTotal: "檔案數量總計" | ||||
| _timelines: | ||||
|   home: "首頁" | ||||
|   local: "本地" | ||||
|   social: "社群" | ||||
|   global: "全域" | ||||
| _rooms: | ||||
|   roomOf: "{user}的房間" | ||||
|   addFurniture: "擺放家具" | ||||
|   translate: "移動 " | ||||
|   rotate: "旋轉" | ||||
|   exit: "返回" | ||||
|   remove: "移除" | ||||
|   clear: "全部移除" | ||||
|   clearConfirm: "確定要移除全部家具嗎?" | ||||
|   leaveConfirm: "修改未儲存,是否要離開?" | ||||
|   chooseImage: "選擇圖像" | ||||
|   roomType: "房間種類" | ||||
|   carpetColor: "地板顏色" | ||||
|   _roomType: | ||||
|     default: "預設" | ||||
|     washitsu: "和室" | ||||
|   _furnitures: | ||||
|     milk: "牛奶盒" | ||||
|     bed: "床" | ||||
|     low-table: "咖啡桌" | ||||
|     desk: "書桌" | ||||
|     chair: "椅子" | ||||
|     chair2: "椅子2" | ||||
|     fan: "通風機" | ||||
|     pc: "電腦" | ||||
|     plant: "觀葉植物" | ||||
|     plant2: "觀葉植物2" | ||||
|     eraser: "橡皮擦" | ||||
|     pencil: "鉛筆" | ||||
|     pudding: "布丁" | ||||
|     cardboard-box: "紙板箱" | ||||
|     cardboard-box2: "紙板箱2" | ||||
|     cardboard-box3: "紙板箱3" | ||||
|     book: "讀物" | ||||
|     book2: "讀物2" | ||||
|     piano: "鋼琴" | ||||
|     moon: "月亮" | ||||
|     corkboard: "木栓板" | ||||
|     mousepad: "滑鼠墊" | ||||
|     monitor: "監視器" | ||||
|     keyboard: "鍵盤" | ||||
|     carpet-stripe: "條紋地毯" | ||||
|     bin: "垃圾箱" | ||||
|     cup-noodle: "杯面" | ||||
|     holo-display: "投影機" | ||||
|     energy-drink: "能量飲料" | ||||
|     doll-ai: "小藍的人偶公仔" | ||||
|     banknote: "大疊鈔票" | ||||
| _pages: | ||||
|   newPage: "建立頁面" | ||||
|   editPage: "編輯頁面" | ||||
|   created: "頁面已建立" | ||||
|   updated: "頁面已更新" | ||||
|   deleted: "頁面已被刪除" | ||||
|   editThisPage: "編輯此頁面" | ||||
|   viewSource: "檢視原始碼" | ||||
|   viewPage: "顯示頁面" | ||||
|   like: "喜歡" | ||||
|   unlike: "收回喜歡" | ||||
|   my: "我的頁面" | ||||
|   liked: "已喜歡的頁面" | ||||
|   inspector: "面板檢查" | ||||
|   variables: "變數" | ||||
|   title: "標題" | ||||
|   url: "頁面網址" | ||||
|   font: "字型" | ||||
|   fontSerif: "襯線體" | ||||
|   fontSansSerif: "無襯線體" | ||||
|   inputBlocks: "輸入" | ||||
|   blocks: | ||||
|     text: "文本" | ||||
|     textarea: "文字區域" | ||||
|     section: "區段" | ||||
|     image: "圖片" | ||||
|     button: "按鈕" | ||||
|     if: "如果" | ||||
|     _if: | ||||
|       variable: "變數" | ||||
|     _post: | ||||
|       text: "内容" | ||||
|       canvasId: "畫布ID" | ||||
|     textInput: "插入文字" | ||||
|     _textInput: | ||||
|       name: "變數名稱" | ||||
|       text: "標題" | ||||
|       default: "預設值" | ||||
|     textareaInput: "多行文字输入" | ||||
|     _textareaInput: | ||||
|       name: "變數名稱" | ||||
|       text: "標題" | ||||
|       default: "預設值" | ||||
|     numberInput: "輸入數值" | ||||
|     _numberInput: | ||||
|       name: "變數名稱" | ||||
|     _canvas: | ||||
|       width: "寬度" | ||||
|     _counter: | ||||
|       text: "標題" | ||||
|       default: "預設值" | ||||
|     canvas: "畫布" | ||||
|     _canvas: | ||||
|       id: "畫布ID" | ||||
|       width: "寬度" | ||||
|       height: "高度" | ||||
|     switch: "開關" | ||||
|     _switch: | ||||
|       name: "變數名稱" | ||||
|       text: "標題" | ||||
|       default: "預設值" | ||||
|     counter: "計數器" | ||||
|     _counter: | ||||
|       name: "變數名稱" | ||||
|       text: "標題" | ||||
|       inc: "増加値" | ||||
|     _button: | ||||
|       text: "標題" | ||||
|       colored: "彩色" | ||||
|       action: "按下按鈕後發生的行為" | ||||
|       _action: | ||||
|         _dialog: | ||||
|           content: "内容" | ||||
|         resetRandom: "重設亂數" | ||||
|         pushEvent: "發送事件" | ||||
|         _pushEvent: | ||||
|           event: "事件名稱" | ||||
|           no-variable: "沒有" | ||||
|         callAiScript: "調用AiScript" | ||||
|         _callAiScript: | ||||
|           functionName: "函數名稱" | ||||
|     radioButton: "選項" | ||||
|     _radioButton: | ||||
|       name: "變數名稱" | ||||
|       title: "標題" | ||||
|       default: "預設值" | ||||
|   script: | ||||
|     categories: | ||||
|       logical: "邏輯運算" | ||||
|       operation: "計算" | ||||
|       comparison: "對比" | ||||
|       random: "隨機" | ||||
|       value: "數值 " | ||||
|       fn: "函数" | ||||
|       text: "文本操作" | ||||
| @@ -544,6 +954,8 @@ _pages: | ||||
|       text: "文本" | ||||
|       multiLineText: "文本 (多行)" | ||||
|       textList: "文本列表" | ||||
|       _strLen: | ||||
|         arg1: "文本" | ||||
|       _strPick: | ||||
|         arg1: "文本" | ||||
|         arg2: "字元位置" | ||||
| @@ -557,18 +969,22 @@ _pages: | ||||
|       _add: | ||||
|         arg1: "A" | ||||
|         arg2: "B" | ||||
|       subtract: "减去" | ||||
|       _subtract: | ||||
|         arg1: "A" | ||||
|         arg2: "B" | ||||
|       multiply: "乘" | ||||
|       _multiply: | ||||
|         arg1: "A" | ||||
|         arg2: "B" | ||||
|       divide: "除" | ||||
|       _divide: | ||||
|         arg1: "A" | ||||
|         arg2: "B" | ||||
|       _mod: | ||||
|         arg1: "A" | ||||
|         arg2: "B" | ||||
|       round: "四舍五入" | ||||
|       _round: | ||||
|         arg1: "數值" | ||||
|       eq: "A和B相等" | ||||
| @@ -610,6 +1026,7 @@ _pages: | ||||
|       not: "否" | ||||
|       _not: | ||||
|         arg1: "否" | ||||
|       random: "隨機" | ||||
|       _random: | ||||
|         arg1: "機率" | ||||
|       rannum: "亂數" | ||||
| @@ -652,6 +1069,8 @@ _pages: | ||||
|         arg1: "文字" | ||||
|       _numberToString: | ||||
|         arg1: "數值" | ||||
|       _splitStrByLine: | ||||
|         arg1: "文本" | ||||
|       ref: "變數" | ||||
|       aiScriptVar: "AiScript的變數" | ||||
|       fn: "函数" | ||||
| @@ -667,23 +1086,43 @@ _pages: | ||||
|       array: "清單" | ||||
|       stringArray: "文本列表" | ||||
|     enviromentVariables: "環境變數" | ||||
|     pageVariables: "頁面元素" | ||||
| _relayStatus: | ||||
|   requesting: "等待核准" | ||||
|   accepted: "已通過核准" | ||||
|   rejected: "已拒絕" | ||||
| _notification: | ||||
|   youRenoted: "{name} 轉發了你的貼文" | ||||
|   youGotPoll: "{name}已投票" | ||||
|   youWereFollowed: "您有新的追隨者" | ||||
|   yourFollowRequestAccepted: "您的追隨請求已通過" | ||||
|   youWereInvitedToGroup: "您有新的群組邀請" | ||||
|   _types: | ||||
|     all: "全部 " | ||||
|     follow: "追隨中" | ||||
|     mention: "提及" | ||||
|     reply: "回覆" | ||||
|     renote: "轉發貼文" | ||||
|     quote: "引用" | ||||
|     reaction: "反應" | ||||
|     receiveFollowRequest: "已收到追隨請求" | ||||
|     followRequestAccepted: "追隨請求已接受" | ||||
|     app: "應用程式通知" | ||||
| _deck: | ||||
|   alwaysShowMainColumn: "總是顯示主欄" | ||||
|   columnAlign: "對齊欄位" | ||||
|   addColumn: "新增欄位" | ||||
|   swapLeft: "向左移動" | ||||
|   swapRight: "向右移動" | ||||
|   swapUp: "往上移動" | ||||
|   swapDown: "往下移動" | ||||
|   stackLeft: "向左折疊" | ||||
|   popRight: "向右彈出" | ||||
|   _columns: | ||||
|     widgets: "小工具" | ||||
|     notifications: "通知" | ||||
|     tl: "時間軸" | ||||
|     antenna: "天線" | ||||
|     list: "清單" | ||||
|     mentions: "提及" | ||||
|     direct: "指定使用者" | ||||
|   | ||||
							
								
								
									
										58
									
								
								migration/1596548170836-channel.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										58
									
								
								migration/1596548170836-channel.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,58 @@ | ||||
| import {MigrationInterface, QueryRunner} from "typeorm"; | ||||
|  | ||||
| export class channel1596548170836 implements MigrationInterface { | ||||
|     name = 'channel1596548170836' | ||||
|  | ||||
|     public async up(queryRunner: QueryRunner): Promise<void> { | ||||
|         await queryRunner.query(`CREATE TABLE "channel" ("id" character varying(32) NOT NULL, "createdAt" TIMESTAMP WITH TIME ZONE NOT NULL, "lastNotedAt" TIMESTAMP WITH TIME ZONE, "userId" character varying(32) NOT NULL, "name" character varying(128) NOT NULL, "description" character varying(2048), "bannerId" character varying(32), "notesCount" integer NOT NULL DEFAULT 0, "usersCount" integer NOT NULL DEFAULT 0, CONSTRAINT "PK_590f33ee6ee7d76437acf362e39" PRIMARY KEY ("id"))`); | ||||
|         await queryRunner.query(`CREATE INDEX "IDX_71cb7b435b7c0d4843317e7e16" ON "channel" ("createdAt") `); | ||||
|         await queryRunner.query(`CREATE INDEX "IDX_29ef80c6f13bcea998447fce43" ON "channel" ("lastNotedAt") `); | ||||
|         await queryRunner.query(`CREATE INDEX "IDX_823bae55bd81b3be6e05cff438" ON "channel" ("userId") `); | ||||
|         await queryRunner.query(`CREATE INDEX "IDX_0f58c11241e649d2a638a8de94" ON "channel" ("notesCount") `); | ||||
|         await queryRunner.query(`CREATE INDEX "IDX_094b86cd36bb805d1aa1e8cc9a" ON "channel" ("usersCount") `); | ||||
|         await queryRunner.query(`CREATE TABLE "channel_following" ("id" character varying(32) NOT NULL, "createdAt" TIMESTAMP WITH TIME ZONE NOT NULL, "followeeId" character varying(32) NOT NULL, "followerId" character varying(32) NOT NULL, CONSTRAINT "PK_8b104be7f7415113f2a02cd5bdd" PRIMARY KEY ("id"))`); | ||||
|         await queryRunner.query(`CREATE INDEX "IDX_11e71f2511589dcc8a4d3214f9" ON "channel_following" ("createdAt") `); | ||||
|         await queryRunner.query(`CREATE INDEX "IDX_0e43068c3f92cab197c3d3cd86" ON "channel_following" ("followeeId") `); | ||||
|         await queryRunner.query(`CREATE INDEX "IDX_6d8084ec9496e7334a4602707e" ON "channel_following" ("followerId") `); | ||||
|         await queryRunner.query(`CREATE UNIQUE INDEX "IDX_2e230dd45a10e671d781d99f3e" ON "channel_following" ("followerId", "followeeId") `); | ||||
|         await queryRunner.query(`CREATE TABLE "channel_note_pining" ("id" character varying(32) NOT NULL, "createdAt" TIMESTAMP WITH TIME ZONE NOT NULL, "channelId" character varying(32) NOT NULL, "noteId" character varying(32) NOT NULL, CONSTRAINT "PK_44f7474496bcf2e4b741681146d" PRIMARY KEY ("id"))`); | ||||
|         await queryRunner.query(`CREATE INDEX "IDX_8125f950afd3093acb10d2db8a" ON "channel_note_pining" ("channelId") `); | ||||
|         await queryRunner.query(`CREATE UNIQUE INDEX "IDX_f36fed37d6d4cdcc68c803cd9c" ON "channel_note_pining" ("channelId", "noteId") `); | ||||
|         await queryRunner.query(`ALTER TABLE "note" ADD "channelId" character varying(32) DEFAULT null`); | ||||
|         await queryRunner.query(`CREATE INDEX "IDX_f22169eb10657bded6d875ac8f" ON "note" ("channelId") `); | ||||
|         await queryRunner.query(`ALTER TABLE "channel" ADD CONSTRAINT "FK_823bae55bd81b3be6e05cff4383" FOREIGN KEY ("userId") REFERENCES "user"("id") ON DELETE SET NULL ON UPDATE NO ACTION`); | ||||
|         await queryRunner.query(`ALTER TABLE "channel" ADD CONSTRAINT "FK_999da2bcc7efadbfe0e92d3bc19" FOREIGN KEY ("bannerId") REFERENCES "drive_file"("id") ON DELETE SET NULL ON UPDATE NO ACTION`); | ||||
|         await queryRunner.query(`ALTER TABLE "note" ADD CONSTRAINT "FK_f22169eb10657bded6d875ac8f9" FOREIGN KEY ("channelId") REFERENCES "channel"("id") ON DELETE CASCADE ON UPDATE NO ACTION`); | ||||
|         await queryRunner.query(`ALTER TABLE "channel_following" ADD CONSTRAINT "FK_0e43068c3f92cab197c3d3cd86e" FOREIGN KEY ("followeeId") REFERENCES "channel"("id") ON DELETE CASCADE ON UPDATE NO ACTION`); | ||||
|         await queryRunner.query(`ALTER TABLE "channel_following" ADD CONSTRAINT "FK_6d8084ec9496e7334a4602707e1" FOREIGN KEY ("followerId") REFERENCES "user"("id") ON DELETE CASCADE ON UPDATE NO ACTION`); | ||||
|         await queryRunner.query(`ALTER TABLE "channel_note_pining" ADD CONSTRAINT "FK_8125f950afd3093acb10d2db8a8" FOREIGN KEY ("channelId") REFERENCES "channel"("id") ON DELETE CASCADE ON UPDATE NO ACTION`); | ||||
|         await queryRunner.query(`ALTER TABLE "channel_note_pining" ADD CONSTRAINT "FK_10b19ef67d297ea9de325cd4502" FOREIGN KEY ("noteId") REFERENCES "note"("id") ON DELETE CASCADE ON UPDATE NO ACTION`); | ||||
|     } | ||||
|  | ||||
|     public async down(queryRunner: QueryRunner): Promise<void> { | ||||
|         await queryRunner.query(`ALTER TABLE "channel_note_pining" DROP CONSTRAINT "FK_10b19ef67d297ea9de325cd4502"`); | ||||
|         await queryRunner.query(`ALTER TABLE "channel_note_pining" DROP CONSTRAINT "FK_8125f950afd3093acb10d2db8a8"`); | ||||
|         await queryRunner.query(`ALTER TABLE "channel_following" DROP CONSTRAINT "FK_6d8084ec9496e7334a4602707e1"`); | ||||
|         await queryRunner.query(`ALTER TABLE "channel_following" DROP CONSTRAINT "FK_0e43068c3f92cab197c3d3cd86e"`); | ||||
|         await queryRunner.query(`ALTER TABLE "note" DROP CONSTRAINT "FK_f22169eb10657bded6d875ac8f9"`); | ||||
|         await queryRunner.query(`ALTER TABLE "channel" DROP CONSTRAINT "FK_999da2bcc7efadbfe0e92d3bc19"`); | ||||
|         await queryRunner.query(`ALTER TABLE "channel" DROP CONSTRAINT "FK_823bae55bd81b3be6e05cff4383"`); | ||||
|         await queryRunner.query(`DROP INDEX "IDX_f22169eb10657bded6d875ac8f"`); | ||||
|         await queryRunner.query(`ALTER TABLE "note" DROP COLUMN "channelId"`); | ||||
|         await queryRunner.query(`DROP INDEX "IDX_f36fed37d6d4cdcc68c803cd9c"`); | ||||
|         await queryRunner.query(`DROP INDEX "IDX_8125f950afd3093acb10d2db8a"`); | ||||
|         await queryRunner.query(`DROP TABLE "channel_note_pining"`); | ||||
|         await queryRunner.query(`DROP INDEX "IDX_2e230dd45a10e671d781d99f3e"`); | ||||
|         await queryRunner.query(`DROP INDEX "IDX_6d8084ec9496e7334a4602707e"`); | ||||
|         await queryRunner.query(`DROP INDEX "IDX_0e43068c3f92cab197c3d3cd86"`); | ||||
|         await queryRunner.query(`DROP INDEX "IDX_11e71f2511589dcc8a4d3214f9"`); | ||||
|         await queryRunner.query(`DROP TABLE "channel_following"`); | ||||
|         await queryRunner.query(`DROP INDEX "IDX_094b86cd36bb805d1aa1e8cc9a"`); | ||||
|         await queryRunner.query(`DROP INDEX "IDX_0f58c11241e649d2a638a8de94"`); | ||||
|         await queryRunner.query(`DROP INDEX "IDX_823bae55bd81b3be6e05cff438"`); | ||||
|         await queryRunner.query(`DROP INDEX "IDX_29ef80c6f13bcea998447fce43"`); | ||||
|         await queryRunner.query(`DROP INDEX "IDX_71cb7b435b7c0d4843317e7e16"`); | ||||
|         await queryRunner.query(`DROP TABLE "channel"`); | ||||
|     } | ||||
|  | ||||
| } | ||||
							
								
								
									
										14
									
								
								migration/1596786425167-channel2.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								migration/1596786425167-channel2.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | ||||
| import {MigrationInterface, QueryRunner} from "typeorm"; | ||||
|  | ||||
| export class channel21596786425167 implements MigrationInterface { | ||||
|     name = 'channel21596786425167' | ||||
|  | ||||
|     public async up(queryRunner: QueryRunner): Promise<void> { | ||||
|         await queryRunner.query(`ALTER TABLE "channel_following" ADD "readCursor" TIMESTAMP WITH TIME ZONE NOT NULL`); | ||||
|     } | ||||
|  | ||||
|     public async down(queryRunner: QueryRunner): Promise<void> { | ||||
|         await queryRunner.query(`ALTER TABLE "channel_following" DROP COLUMN "readCursor"`); | ||||
|     } | ||||
|  | ||||
| } | ||||
							
								
								
									
										14
									
								
								migration/1597230137744-objectStorageSetPublicRead.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								migration/1597230137744-objectStorageSetPublicRead.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | ||||
| import {MigrationInterface, QueryRunner} from "typeorm"; | ||||
|  | ||||
| export class objectStorageSetPublicRead1597230137744 implements MigrationInterface { | ||||
|     name = 'objectStorageSetPublicRead1597230137744' | ||||
|  | ||||
|     public async up(queryRunner: QueryRunner): Promise<void> { | ||||
|         await queryRunner.query(`ALTER TABLE "meta" ADD "objectStorageSetPublicRead" boolean NOT NULL DEFAULT false`); | ||||
|     } | ||||
|  | ||||
|     public async down(queryRunner: QueryRunner): Promise<void> { | ||||
|         await queryRunner.query(`ALTER TABLE "meta" DROP COLUMN "objectStorageSetPublicRead"`); | ||||
|     } | ||||
|  | ||||
| } | ||||
							
								
								
									
										16
									
								
								migration/1597236229720-IncludingNotificationTypes.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								migration/1597236229720-IncludingNotificationTypes.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,16 @@ | ||||
| import {MigrationInterface, QueryRunner} from "typeorm"; | ||||
|  | ||||
| export class IncludingNotificationTypes1597236229720 implements MigrationInterface { | ||||
|     name = 'IncludingNotificationTypes1597236229720' | ||||
|  | ||||
|     public async up(queryRunner: QueryRunner): Promise<void> { | ||||
|         await queryRunner.query(`CREATE TYPE "user_profile_includingnotificationtypes_enum" AS ENUM('follow', 'mention', 'reply', 'renote', 'quote', 'reaction', 'pollVote', 'receiveFollowRequest', 'followRequestAccepted', 'groupInvited', 'app')`); | ||||
|         await queryRunner.query(`ALTER TABLE "user_profile" ADD "includingNotificationTypes" "user_profile_includingnotificationtypes_enum" array`); | ||||
|     } | ||||
|  | ||||
|     public async down(queryRunner: QueryRunner): Promise<void> { | ||||
|         await queryRunner.query(`ALTER TABLE "user_profile" DROP COLUMN "includingNotificationTypes"`); | ||||
|         await queryRunner.query(`DROP TYPE "user_profile_includingnotificationtypes_enum"`); | ||||
|     } | ||||
|  | ||||
| } | ||||
							
								
								
									
										14
									
								
								migration/1597385880794-add-sensitive-index.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								migration/1597385880794-add-sensitive-index.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | ||||
| import {MigrationInterface, QueryRunner} from "typeorm"; | ||||
|  | ||||
| export class addSensitiveIndex1597385880794 implements MigrationInterface { | ||||
|     name = 'addSensitiveIndex1597385880794' | ||||
|  | ||||
|     public async up(queryRunner: QueryRunner): Promise<void> { | ||||
|         await queryRunner.query(`CREATE INDEX "IDX_a7eba67f8b3fa27271e85d2e26" ON "drive_file" ("isSensitive") `); | ||||
|     } | ||||
|  | ||||
|     public async down(queryRunner: QueryRunner): Promise<void> { | ||||
|         await queryRunner.query(`DROP INDEX "IDX_a7eba67f8b3fa27271e85d2e26"`); | ||||
|     } | ||||
|  | ||||
| } | ||||
							
								
								
									
										27
									
								
								migration/1597459042300-channel-unread.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								migration/1597459042300-channel-unread.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,27 @@ | ||||
| import {MigrationInterface, QueryRunner} from "typeorm"; | ||||
|  | ||||
| export class channelUnread1597459042300 implements MigrationInterface { | ||||
|     name = 'channelUnread1597459042300' | ||||
|  | ||||
|     public async up(queryRunner: QueryRunner): Promise<void> { | ||||
| 				await queryRunner.query(`TRUNCATE TABLE "note_unread"`, undefined); | ||||
|         await queryRunner.query(`ALTER TABLE "channel_following" DROP COLUMN "readCursor"`); | ||||
|         await queryRunner.query(`ALTER TABLE "note_unread" ADD "isMentioned" boolean NOT NULL`); | ||||
|         await queryRunner.query(`ALTER TABLE "note_unread" ADD "noteChannelId" character varying(32)`); | ||||
|         await queryRunner.query(`CREATE INDEX "IDX_25b1dd384bec391b07b74b861c" ON "note_unread" ("isMentioned") `); | ||||
|         await queryRunner.query(`CREATE INDEX "IDX_89a29c9237b8c3b6b3cbb4cb30" ON "note_unread" ("isSpecified") `); | ||||
|         await queryRunner.query(`CREATE INDEX "IDX_29e8c1d579af54d4232939f994" ON "note_unread" ("noteUserId") `); | ||||
|         await queryRunner.query(`CREATE INDEX "IDX_6a57f051d82c6d4036c141e107" ON "note_unread" ("noteChannelId") `); | ||||
|     } | ||||
|  | ||||
|     public async down(queryRunner: QueryRunner): Promise<void> { | ||||
|         await queryRunner.query(`DROP INDEX "IDX_6a57f051d82c6d4036c141e107"`); | ||||
|         await queryRunner.query(`DROP INDEX "IDX_29e8c1d579af54d4232939f994"`); | ||||
|         await queryRunner.query(`DROP INDEX "IDX_89a29c9237b8c3b6b3cbb4cb30"`); | ||||
|         await queryRunner.query(`DROP INDEX "IDX_25b1dd384bec391b07b74b861c"`); | ||||
|         await queryRunner.query(`ALTER TABLE "note_unread" DROP COLUMN "noteChannelId"`); | ||||
|         await queryRunner.query(`ALTER TABLE "note_unread" DROP COLUMN "isMentioned"`); | ||||
|         await queryRunner.query(`ALTER TABLE "channel_following" ADD "readCursor" TIMESTAMP WITH TIME ZONE NOT NULL`); | ||||
|     } | ||||
|  | ||||
| } | ||||
							
								
								
									
										16
									
								
								migration/1597893996136-ChannelNoteIdDescIndex.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								migration/1597893996136-ChannelNoteIdDescIndex.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,16 @@ | ||||
| import {MigrationInterface, QueryRunner} from "typeorm"; | ||||
|  | ||||
| export class ChannelNoteIdDescIndex1597893996136 implements MigrationInterface { | ||||
|     name = 'ChannelNoteIdDescIndex1597893996136' | ||||
|  | ||||
|     public async up(queryRunner: QueryRunner): Promise<void> { | ||||
| 				await queryRunner.query(`DROP INDEX "IDX_f22169eb10657bded6d875ac8f"`); | ||||
| 				await queryRunner.query(`CREATE INDEX "IDX_note_on_channelId_and_id_desc" ON "note" ("channelId", "id" desc)`); | ||||
| 		} | ||||
|  | ||||
|     public async down(queryRunner: QueryRunner): Promise<void> { | ||||
| 				await queryRunner.query(`DROP INDEX "IDX_note_on_channelId_and_id_desc"`); | ||||
|         await queryRunner.query(`CREATE INDEX "IDX_f22169eb10657bded6d875ac8f" ON "note" ("channelId") `); | ||||
|     } | ||||
|  | ||||
| } | ||||
							
								
								
									
										20
									
								
								migration/1600353287890-mutingNotificationTypes.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								migration/1600353287890-mutingNotificationTypes.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,20 @@ | ||||
| import {MigrationInterface, QueryRunner} from "typeorm"; | ||||
|  | ||||
| export class mutingNotificationTypes1600353287890 implements MigrationInterface { | ||||
|     name = 'mutingNotificationTypes1600353287890' | ||||
|  | ||||
|     public async up(queryRunner: QueryRunner): Promise<void> { | ||||
|         await queryRunner.query(`ALTER TABLE "user_profile" DROP COLUMN "includingNotificationTypes"`); | ||||
|         await queryRunner.query(`DROP TYPE "public"."user_profile_includingnotificationtypes_enum"`); | ||||
|         await queryRunner.query(`CREATE TYPE "user_profile_mutingnotificationtypes_enum" AS ENUM('follow', 'mention', 'reply', 'renote', 'quote', 'reaction', 'pollVote', 'receiveFollowRequest', 'followRequestAccepted', 'groupInvited', 'app')`); | ||||
|         await queryRunner.query(`ALTER TABLE "user_profile" ADD "mutingNotificationTypes" "user_profile_mutingnotificationtypes_enum" array NOT NULL DEFAULT '{}'`); | ||||
|     } | ||||
|  | ||||
|     public async down(queryRunner: QueryRunner): Promise<void> { | ||||
|         await queryRunner.query(`ALTER TABLE "user_profile" DROP COLUMN "mutingNotificationTypes"`); | ||||
|         await queryRunner.query(`DROP TYPE "user_profile_mutingnotificationtypes_enum"`); | ||||
|         await queryRunner.query(`CREATE TYPE "public"."user_profile_includingnotificationtypes_enum" AS ENUM('follow', 'mention', 'reply', 'renote', 'quote', 'reaction', 'pollVote', 'receiveFollowRequest', 'followRequestAccepted', 'groupInvited', 'app')`); | ||||
|         await queryRunner.query(`ALTER TABLE "user_profile" ADD "includingNotificationTypes" "user_profile_includingnotificationtypes_enum" array`); | ||||
|     } | ||||
|  | ||||
| } | ||||
							
								
								
									
										32
									
								
								migration/1603094348345-refine-abuse-user-report.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										32
									
								
								migration/1603094348345-refine-abuse-user-report.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,32 @@ | ||||
| import {MigrationInterface, QueryRunner} from "typeorm"; | ||||
|  | ||||
| export class refineAbuseUserReport1603094348345 implements MigrationInterface { | ||||
|     name = 'refineAbuseUserReport1603094348345' | ||||
|  | ||||
|     public async up(queryRunner: QueryRunner): Promise<void> { | ||||
|         await queryRunner.query(`ALTER TABLE "abuse_user_report" DROP CONSTRAINT "FK_d049123c413e68ca52abe734203"`); | ||||
|         await queryRunner.query(`DROP INDEX "IDX_d049123c413e68ca52abe73420"`); | ||||
|         await queryRunner.query(`DROP INDEX "IDX_5cd442c3b2e74fdd99dae20243"`); | ||||
|         await queryRunner.query(`ALTER TABLE "abuse_user_report" RENAME COLUMN "userId" TO "targetUserId"`); | ||||
|         await queryRunner.query(`ALTER TABLE "abuse_user_report" ADD "assigneeId" character varying(32)`); | ||||
|         await queryRunner.query(`ALTER TABLE "abuse_user_report" ADD "resolved" boolean NOT NULL DEFAULT false`); | ||||
|         await queryRunner.query(`ALTER TABLE "abuse_user_report" DROP COLUMN "comment"`); | ||||
|         await queryRunner.query(`ALTER TABLE "abuse_user_report" ADD "comment" character varying(2048) NOT NULL DEFAULT '{}'::varchar[]`); | ||||
|         await queryRunner.query(`CREATE INDEX "IDX_2b15aaf4a0dc5be3499af7ab6a" ON "abuse_user_report" ("resolved") `); | ||||
|         await queryRunner.query(`ALTER TABLE "abuse_user_report" ADD CONSTRAINT "FK_08b883dd5fdd6f9c4c1572b36de" FOREIGN KEY ("assigneeId") REFERENCES "user"("id") ON DELETE SET NULL ON UPDATE NO ACTION`); | ||||
|     } | ||||
|  | ||||
|     public async down(queryRunner: QueryRunner): Promise<void> { | ||||
|         await queryRunner.query(`ALTER TABLE "abuse_user_report" DROP CONSTRAINT "FK_08b883dd5fdd6f9c4c1572b36de"`); | ||||
|         await queryRunner.query(`DROP INDEX "IDX_2b15aaf4a0dc5be3499af7ab6a"`); | ||||
|         await queryRunner.query(`ALTER TABLE "abuse_user_report" DROP COLUMN "comment"`); | ||||
|         await queryRunner.query(`ALTER TABLE "abuse_user_report" ADD "comment" character varying(512) NOT NULL DEFAULT '{}'::varchar[]`); | ||||
|         await queryRunner.query(`ALTER TABLE "abuse_user_report" DROP COLUMN "resolved"`); | ||||
|         await queryRunner.query(`ALTER TABLE "abuse_user_report" DROP COLUMN "assigneeId"`); | ||||
|         await queryRunner.query(`ALTER TABLE "abuse_user_report" RENAME COLUMN "targetUserId" TO "userId"`); | ||||
|         await queryRunner.query(`CREATE UNIQUE INDEX "IDX_5cd442c3b2e74fdd99dae20243" ON "abuse_user_report" ("userId", "reporterId") `); | ||||
|         await queryRunner.query(`CREATE INDEX "IDX_d049123c413e68ca52abe73420" ON "abuse_user_report" ("userId") `); | ||||
|         await queryRunner.query(`ALTER TABLE "abuse_user_report" ADD CONSTRAINT "FK_d049123c413e68ca52abe734203" FOREIGN KEY ("userId") REFERENCES "user"("id") ON DELETE CASCADE ON UPDATE NO ACTION`); | ||||
|     } | ||||
|  | ||||
| } | ||||
							
								
								
									
										20
									
								
								migration/1603095701770-refine-abuse-user-report2.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								migration/1603095701770-refine-abuse-user-report2.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,20 @@ | ||||
| import {MigrationInterface, QueryRunner} from "typeorm"; | ||||
|  | ||||
| export class refineAbuseUserReport21603095701770 implements MigrationInterface { | ||||
|     name = 'refineAbuseUserReport21603095701770' | ||||
|  | ||||
|     public async up(queryRunner: QueryRunner): Promise<void> { | ||||
|         await queryRunner.query(`ALTER TABLE "abuse_user_report" ADD "targetUserHost" character varying(128)`); | ||||
|         await queryRunner.query(`ALTER TABLE "abuse_user_report" ADD "reporterHost" character varying(128)`); | ||||
|         await queryRunner.query(`CREATE INDEX "IDX_4ebbf7f93cdc10e8d1ef2fc6cd" ON "abuse_user_report" ("targetUserHost") `); | ||||
|         await queryRunner.query(`CREATE INDEX "IDX_f8d8b93740ad12c4ce8213a199" ON "abuse_user_report" ("reporterHost") `); | ||||
|     } | ||||
|  | ||||
|     public async down(queryRunner: QueryRunner): Promise<void> { | ||||
|         await queryRunner.query(`DROP INDEX "IDX_f8d8b93740ad12c4ce8213a199"`); | ||||
|         await queryRunner.query(`DROP INDEX "IDX_4ebbf7f93cdc10e8d1ef2fc6cd"`); | ||||
|         await queryRunner.query(`ALTER TABLE "abuse_user_report" DROP COLUMN "reporterHost"`); | ||||
|         await queryRunner.query(`ALTER TABLE "abuse_user_report" DROP COLUMN "targetUserHost"`); | ||||
|     } | ||||
|  | ||||
| } | ||||
							
								
								
									
										14
									
								
								migration/1603776877564-instance-theme-color.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								migration/1603776877564-instance-theme-color.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | ||||
| import {MigrationInterface, QueryRunner} from "typeorm"; | ||||
|  | ||||
| export class instanceThemeColor1603776877564 implements MigrationInterface { | ||||
|     name = 'instanceThemeColor1603776877564' | ||||
|  | ||||
|     public async up(queryRunner: QueryRunner): Promise<void> { | ||||
|         await queryRunner.query(`ALTER TABLE "instance" ADD "themeColor" character varying(64) DEFAULT null`); | ||||
|     } | ||||
|  | ||||
|     public async down(queryRunner: QueryRunner): Promise<void> { | ||||
|         await queryRunner.query(`ALTER TABLE "instance" DROP COLUMN "themeColor"`); | ||||
|     } | ||||
|  | ||||
| } | ||||
							
								
								
									
										14
									
								
								migration/1603781553011-instance-favicon.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								migration/1603781553011-instance-favicon.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | ||||
| import {MigrationInterface, QueryRunner} from "typeorm"; | ||||
|  | ||||
| export class instanceFavicon1603781553011 implements MigrationInterface { | ||||
|     name = 'instanceFavicon1603781553011' | ||||
|  | ||||
|     public async up(queryRunner: QueryRunner): Promise<void> { | ||||
|         await queryRunner.query(`ALTER TABLE "instance" ADD "faviconUrl" character varying(256) DEFAULT null`); | ||||
|     } | ||||
|  | ||||
|     public async down(queryRunner: QueryRunner): Promise<void> { | ||||
|         await queryRunner.query(`ALTER TABLE "instance" DROP COLUMN "faviconUrl"`); | ||||
|     } | ||||
|  | ||||
| } | ||||
							
								
								
									
										14
									
								
								migration/1604821689616-delete-auto-watch.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								migration/1604821689616-delete-auto-watch.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | ||||
| import {MigrationInterface, QueryRunner} from "typeorm"; | ||||
|  | ||||
| export class deleteAutoWatch1604821689616 implements MigrationInterface { | ||||
|     name = 'deleteAutoWatch1604821689616' | ||||
|  | ||||
|     public async up(queryRunner: QueryRunner): Promise<void> { | ||||
|         await queryRunner.query(`ALTER TABLE "user_profile" DROP COLUMN "autoWatch"`); | ||||
|     } | ||||
|  | ||||
|     public async down(queryRunner: QueryRunner): Promise<void> { | ||||
|         await queryRunner.query(`ALTER TABLE "user_profile" ADD "autoWatch" boolean NOT NULL DEFAULT false`); | ||||
|     } | ||||
|  | ||||
| } | ||||
							
								
								
									
										15
									
								
								migration/1605408848373-clip-description.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								migration/1605408848373-clip-description.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,15 @@ | ||||
| import {MigrationInterface, QueryRunner} from "typeorm"; | ||||
|  | ||||
| export class clipDescription1605408848373 implements MigrationInterface { | ||||
|     name = 'clipDescription1605408848373' | ||||
|  | ||||
|     public async up(queryRunner: QueryRunner): Promise<void> { | ||||
|         await queryRunner.query(`ALTER TABLE "clip" ADD "description" character varying(2048) DEFAULT null`); | ||||
|     } | ||||
|  | ||||
|     public async down(queryRunner: QueryRunner): Promise<void> { | ||||
|  | ||||
|         await queryRunner.query(`ALTER TABLE "clip" DROP COLUMN "description"`); | ||||
|     } | ||||
|  | ||||
| } | ||||
							
								
								
									
										434
									
								
								migration/1605408971051-comments.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										434
									
								
								migration/1605408971051-comments.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,434 @@ | ||||
| import {MigrationInterface, QueryRunner} from "typeorm"; | ||||
|  | ||||
| export class comments1605408971051 implements MigrationInterface { | ||||
|     name = 'comments1605408971051' | ||||
|  | ||||
|     public async up(queryRunner: QueryRunner): Promise<void> { | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "log"."createdAt" IS 'The created date of the Log.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "drive_folder"."createdAt" IS 'The created date of the DriveFolder.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "drive_folder"."name" IS 'The name of the DriveFolder.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "drive_folder"."userId" IS 'The owner ID.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "drive_folder"."parentId" IS 'The parent folder ID. If null, it means the DriveFolder is located in root.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "drive_file"."createdAt" IS 'The created date of the DriveFile.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "drive_file"."userId" IS 'The owner ID.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "drive_file"."userHost" IS 'The host of owner. It will be null if the user in local.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "drive_file"."md5" IS 'The MD5 hash of the DriveFile.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "drive_file"."name" IS 'The file name of the DriveFile.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "drive_file"."type" IS 'The content type (MIME) of the DriveFile.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "drive_file"."size" IS 'The file size (bytes) of the DriveFile.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "drive_file"."comment" IS 'The comment of the DriveFile.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "drive_file"."blurhash" IS 'The BlurHash string.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "drive_file"."properties" IS 'The any properties of the DriveFile. For example, it includes image width/height.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "drive_file"."url" IS 'The URL of the DriveFile.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "drive_file"."thumbnailUrl" IS 'The URL of the thumbnail of the DriveFile.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "drive_file"."webpublicUrl" IS 'The URL of the webpublic of the DriveFile.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "drive_file"."uri" IS 'The URI of the DriveFile. it will be null when the DriveFile is local.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "drive_file"."folderId" IS 'The parent folder ID. If null, it means the DriveFile is located in root.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "drive_file"."isSensitive" IS 'Whether the DriveFile is NSFW.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "drive_file"."isLink" IS 'Whether the DriveFile is direct link to remote server.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user"."createdAt" IS 'The created date of the User.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user"."updatedAt" IS 'The updated date of the User.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user"."username" IS 'The username of the User.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user"."usernameLower" IS 'The username (lowercased) of the User.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user"."name" IS 'The name of the User.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user"."followersCount" IS 'The count of followers.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user"."followingCount" IS 'The count of following.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user"."notesCount" IS 'The count of notes.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user"."avatarId" IS 'The ID of avatar DriveFile.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user"."bannerId" IS 'The ID of banner DriveFile.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user"."isSuspended" IS 'Whether the User is suspended.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user"."isSilenced" IS 'Whether the User is silenced.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user"."isLocked" IS 'Whether the User is locked.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user"."isBot" IS 'Whether the User is a bot.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user"."isCat" IS 'Whether the User is a cat.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user"."isAdmin" IS 'Whether the User is the admin.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user"."isModerator" IS 'Whether the User is a moderator.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user"."host" IS 'The host of the User. It will be null if the origin of the user is local.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user"."inbox" IS 'The inbox URL of the User. It will be null if the origin of the user is local.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user"."sharedInbox" IS 'The sharedInbox URL of the User. It will be null if the origin of the user is local.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user"."featured" IS 'The featured URL of the User. It will be null if the origin of the user is local.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user"."uri" IS 'The URI of the User. It will be null if the origin of the user is local.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user"."token" IS 'The native access token of the User. It will be null if the origin of the user is local.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "app"."createdAt" IS 'The created date of the App.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "app"."userId" IS 'The owner ID.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "app"."secret" IS 'The secret key of the App.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "app"."name" IS 'The name of the App.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "app"."description" IS 'The description of the App.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "app"."permission" IS 'The permission of the App.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "app"."callbackUrl" IS 'The callbackUrl of the App.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "access_token"."createdAt" IS 'The created date of the AccessToken.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "access_token"."lastUsedAt" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "access_token"."session" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "access_token"."appId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "access_token"."name" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "access_token"."description" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "access_token"."iconUrl" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "channel"."createdAt" IS 'The created date of the Channel.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "channel"."userId" IS 'The owner ID.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "channel"."name" IS 'The name of the Channel.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "channel"."description" IS 'The description of the Channel.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "channel"."bannerId" IS 'The ID of banner Channel.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "channel"."notesCount" IS 'The count of notes.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "channel"."usersCount" IS 'The count of users.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "note"."createdAt" IS 'The created date of the Note.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "note"."replyId" IS 'The ID of reply target.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "note"."renoteId" IS 'The ID of renote target.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "note"."userId" IS 'The ID of author.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "note"."uri" IS 'The URI of a note. it will be null when the note is local.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "note"."url" IS 'The human readable url of a note. it will be null when the note is local.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "note"."channelId" IS 'The ID of source channel.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "note"."userHost" IS '[Denormalized]'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "note"."replyUserId" IS '[Denormalized]'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "note"."replyUserHost" IS '[Denormalized]'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "note"."renoteUserId" IS '[Denormalized]'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "note"."renoteUserHost" IS '[Denormalized]'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "poll_vote"."createdAt" IS 'The created date of the PollVote.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "note_reaction"."createdAt" IS 'The created date of the NoteReaction.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "note_watching"."createdAt" IS 'The created date of the NoteWatching.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "note_watching"."userId" IS 'The watcher ID.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "note_watching"."noteId" IS 'The target Note ID.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "note_watching"."noteUserId" IS '[Denormalized]'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "note_unread"."noteUserId" IS '[Denormalized]'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "note_unread"."noteChannelId" IS '[Denormalized]'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "follow_request"."createdAt" IS 'The created date of the FollowRequest.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "follow_request"."followeeId" IS 'The followee user ID.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "follow_request"."followerId" IS 'The follower user ID.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "follow_request"."requestId" IS 'id of Follow Activity.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "follow_request"."followerHost" IS '[Denormalized]'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "follow_request"."followerInbox" IS '[Denormalized]'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "follow_request"."followerSharedInbox" IS '[Denormalized]'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "follow_request"."followeeHost" IS '[Denormalized]'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "follow_request"."followeeInbox" IS '[Denormalized]'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "follow_request"."followeeSharedInbox" IS '[Denormalized]'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_group"."createdAt" IS 'The created date of the UserGroup.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_group"."userId" IS 'The ID of owner.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_group_invitation"."createdAt" IS 'The created date of the UserGroupInvitation.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_group_invitation"."userId" IS 'The user ID.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_group_invitation"."userGroupId" IS 'The group ID.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "notification"."createdAt" IS 'The created date of the Notification.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "notification"."notifieeId" IS 'The ID of recipient user of the Notification.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "notification"."isRead" IS 'Whether the Notification is read.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "meta"."localDriveCapacityMb" IS 'Drive capacity of a local user (MB)'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "meta"."remoteDriveCapacityMb" IS 'Drive capacity of a remote user (MB)'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "meta"."maxNoteTextLength" IS 'Max allowed note text length in characters'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "following"."createdAt" IS 'The created date of the Following.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "following"."followeeId" IS 'The followee user ID.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "following"."followerId" IS 'The follower user ID.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "following"."followerHost" IS '[Denormalized]'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "following"."followerInbox" IS '[Denormalized]'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "following"."followerSharedInbox" IS '[Denormalized]'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "following"."followeeHost" IS '[Denormalized]'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "following"."followeeInbox" IS '[Denormalized]'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "following"."followeeSharedInbox" IS '[Denormalized]'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "instance"."caughtAt" IS 'The caught date of the Instance.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "instance"."host" IS 'The host of the Instance.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "instance"."usersCount" IS 'The count of the users of the Instance.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "instance"."notesCount" IS 'The count of the notes of the Instance.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "instance"."softwareName" IS 'The software of the Instance.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "instance"."softwareVersion" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "instance"."openRegistrations" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "instance"."name" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "instance"."description" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "instance"."maintainerName" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "instance"."maintainerEmail" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "instance"."iconUrl" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "instance"."faviconUrl" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "instance"."themeColor" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "muting"."createdAt" IS 'The created date of the Muting.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "muting"."muteeId" IS 'The mutee user ID.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "muting"."muterId" IS 'The muter user ID.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "blocking"."createdAt" IS 'The created date of the Blocking.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "blocking"."blockeeId" IS 'The blockee user ID.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "blocking"."blockerId" IS 'The blocker user ID.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_list"."createdAt" IS 'The created date of the UserList.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_list"."userId" IS 'The owner ID.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_list"."name" IS 'The name of the UserList.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_list_joining"."createdAt" IS 'The created date of the UserListJoining.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_list_joining"."userId" IS 'The user ID.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_list_joining"."userListId" IS 'The list ID.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_group_joining"."createdAt" IS 'The created date of the UserGroupJoining.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_group_joining"."userId" IS 'The user ID.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_group_joining"."userGroupId" IS 'The group ID.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "note_favorite"."createdAt" IS 'The created date of the NoteFavorite.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "abuse_user_report"."createdAt" IS 'The created date of the AbuseUserReport.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "abuse_user_report"."targetUserHost" IS '[Denormalized]'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "abuse_user_report"."reporterHost" IS '[Denormalized]'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "messaging_message"."createdAt" IS 'The created date of the MessagingMessage.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "messaging_message"."userId" IS 'The sender user ID.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "messaging_message"."groupId" IS 'The recipient group ID.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "signin"."createdAt" IS 'The created date of the Signin.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "auth_session"."createdAt" IS 'The created date of the AuthSession.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "reversi_game"."createdAt" IS 'The created date of the ReversiGame.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "reversi_game"."startedAt" IS 'The started date of the ReversiGame.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "reversi_game"."form1" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "reversi_game"."form2" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "reversi_matching"."createdAt" IS 'The created date of the ReversiMatching.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_note_pining"."createdAt" IS 'The created date of the UserNotePinings.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "poll"."noteId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "poll"."noteVisibility" IS '[Denormalized]'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "poll"."userId" IS '[Denormalized]'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "poll"."userHost" IS '[Denormalized]'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_keypair"."userId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_publickey"."userId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "page"."createdAt" IS 'The created date of the Page.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "page"."updatedAt" IS 'The updated date of the Page.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "page"."userId" IS 'The ID of author.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_profile"."userId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_profile"."location" IS 'The location of the User.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_profile"."birthday" IS 'The birthday (YYYY-MM-DD) of the User.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_profile"."description" IS 'The description (bio) of the User.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_profile"."url" IS 'Remote URL of the user.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_profile"."email" IS 'The email address of the User.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_profile"."password" IS 'The password hash of the User. It will be null if the origin of the user is local.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_profile"."clientData" IS 'The client-specific data of the User.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_profile"."room" IS 'The room data of the User.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_profile"."userHost" IS '[Denormalized]'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_security_key"."id" IS 'Variable-length id given to navigator.credentials.get()'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_security_key"."publicKey" IS 'Variable-length public key used to verify attestations (hex-encoded).'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_security_key"."lastUsed" IS 'The date of the last time the UserSecurityKey was successfully validated.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_security_key"."name" IS 'User-defined name for this key'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "attestation_challenge"."challenge" IS 'Hex-encoded sha256 hash of the challenge.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "attestation_challenge"."createdAt" IS 'The date challenge was created for expiry purposes.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "attestation_challenge"."registrationChallenge" IS 'Indicates that the challenge is only for registration purposes if true to prevent the challenge for being used as authentication.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "moderation_log"."createdAt" IS 'The created date of the ModerationLog.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "announcement"."createdAt" IS 'The created date of the Announcement.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "announcement"."updatedAt" IS 'The updated date of the Announcement.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "announcement_read"."createdAt" IS 'The created date of the AnnouncementRead.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "clip"."createdAt" IS 'The created date of the Clip.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "clip"."userId" IS 'The owner ID.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "clip"."name" IS 'The name of the Clip.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "clip"."description" IS 'The description of the Clip.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "clip_note"."noteId" IS 'The note ID.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "clip_note"."clipId" IS 'The clip ID.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "antenna"."createdAt" IS 'The created date of the Antenna.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "antenna"."userId" IS 'The owner ID.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "antenna"."name" IS 'The name of the Antenna.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "antenna_note"."noteId" IS 'The note ID.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "antenna_note"."antennaId" IS 'The antenna ID.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "promo_note"."noteId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "promo_note"."userId" IS '[Denormalized]'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "promo_read"."createdAt" IS 'The created date of the PromoRead.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "muted_note"."noteId" IS 'The note ID.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "muted_note"."userId" IS 'The user ID.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "muted_note"."reason" IS 'The reason of the MutedNote.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "channel_following"."createdAt" IS 'The created date of the ChannelFollowing.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "channel_following"."followeeId" IS 'The followee channel ID.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "channel_following"."followerId" IS 'The follower user ID.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "channel_note_pining"."createdAt" IS 'The created date of the ChannelNotePining.'`); | ||||
|     } | ||||
|  | ||||
|     public async down(queryRunner: QueryRunner): Promise<void> { | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "channel_note_pining"."createdAt" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "channel_following"."followerId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "channel_following"."followeeId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "channel_following"."createdAt" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "muted_note"."reason" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "muted_note"."userId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "muted_note"."noteId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "promo_read"."createdAt" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "promo_note"."userId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "promo_note"."noteId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "antenna_note"."antennaId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "antenna_note"."noteId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "antenna"."name" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "antenna"."userId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "antenna"."createdAt" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "clip_note"."clipId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "clip_note"."noteId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "clip"."description" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "clip"."name" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "clip"."userId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "clip"."createdAt" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "announcement_read"."createdAt" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "announcement"."updatedAt" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "announcement"."createdAt" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "moderation_log"."createdAt" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "attestation_challenge"."registrationChallenge" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "attestation_challenge"."createdAt" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "attestation_challenge"."challenge" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_security_key"."name" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_security_key"."lastUsed" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_security_key"."publicKey" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_security_key"."id" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_profile"."userHost" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_profile"."room" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_profile"."clientData" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_profile"."password" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_profile"."email" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_profile"."url" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_profile"."description" IS 'The description (bio) of the User.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_profile"."birthday" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_profile"."location" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_profile"."userId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "page"."userId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "page"."updatedAt" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "page"."createdAt" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_publickey"."userId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_keypair"."userId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "poll"."userHost" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "poll"."userId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "poll"."noteVisibility" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "poll"."noteId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_note_pining"."createdAt" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "reversi_matching"."createdAt" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "reversi_game"."form2" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "reversi_game"."form1" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "reversi_game"."startedAt" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "reversi_game"."createdAt" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "auth_session"."createdAt" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "signin"."createdAt" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "messaging_message"."groupId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "messaging_message"."userId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "messaging_message"."createdAt" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "abuse_user_report"."reporterHost" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "abuse_user_report"."targetUserHost" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "abuse_user_report"."createdAt" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "note_favorite"."createdAt" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_group_joining"."userGroupId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_group_joining"."userId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_group_joining"."createdAt" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_list_joining"."userListId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_list_joining"."userId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_list_joining"."createdAt" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_list"."name" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_list"."userId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_list"."createdAt" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "blocking"."blockerId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "blocking"."blockeeId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "blocking"."createdAt" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "muting"."muterId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "muting"."muteeId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "muting"."createdAt" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "instance"."themeColor" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "instance"."faviconUrl" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "instance"."iconUrl" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "instance"."maintainerEmail" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "instance"."maintainerName" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "instance"."description" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "instance"."name" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "instance"."openRegistrations" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "instance"."softwareVersion" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "instance"."softwareName" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "instance"."notesCount" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "instance"."usersCount" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "instance"."host" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "instance"."caughtAt" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "following"."followeeSharedInbox" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "following"."followeeInbox" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "following"."followeeHost" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "following"."followerSharedInbox" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "following"."followerInbox" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "following"."followerHost" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "following"."followerId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "following"."followeeId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "following"."createdAt" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "meta"."maxNoteTextLength" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "meta"."remoteDriveCapacityMb" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "meta"."localDriveCapacityMb" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "notification"."isRead" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "notification"."notifieeId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "notification"."createdAt" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_group_invitation"."userGroupId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_group_invitation"."userId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_group_invitation"."createdAt" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_group"."userId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user_group"."createdAt" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "follow_request"."followeeSharedInbox" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "follow_request"."followeeInbox" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "follow_request"."followeeHost" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "follow_request"."followerSharedInbox" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "follow_request"."followerInbox" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "follow_request"."followerHost" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "follow_request"."requestId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "follow_request"."followerId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "follow_request"."followeeId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "follow_request"."createdAt" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "note_unread"."noteChannelId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "note_unread"."noteUserId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "note_watching"."noteUserId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "note_watching"."noteId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "note_watching"."userId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "note_watching"."createdAt" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "note_reaction"."createdAt" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "poll_vote"."createdAt" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "note"."renoteUserHost" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "note"."renoteUserId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "note"."replyUserHost" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "note"."replyUserId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "note"."userHost" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "note"."channelId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "note"."url" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "note"."uri" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "note"."userId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "note"."renoteId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "note"."replyId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "note"."createdAt" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "channel"."usersCount" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "channel"."notesCount" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "channel"."bannerId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "channel"."description" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "channel"."name" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "channel"."userId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "channel"."createdAt" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "access_token"."iconUrl" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "access_token"."description" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "access_token"."name" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "access_token"."appId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "access_token"."session" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "access_token"."lastUsedAt" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "access_token"."createdAt" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "app"."callbackUrl" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "app"."permission" IS 'The permission of the App.'`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "app"."description" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "app"."name" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "app"."secret" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "app"."userId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "app"."createdAt" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user"."token" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user"."uri" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user"."featured" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user"."sharedInbox" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user"."inbox" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user"."host" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user"."isModerator" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user"."isAdmin" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user"."isCat" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user"."isBot" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user"."isLocked" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user"."isSilenced" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user"."isSuspended" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user"."bannerId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user"."avatarId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user"."notesCount" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user"."followingCount" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user"."followersCount" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user"."name" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user"."usernameLower" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user"."username" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user"."updatedAt" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "user"."createdAt" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "drive_file"."isLink" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "drive_file"."isSensitive" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "drive_file"."folderId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "drive_file"."uri" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "drive_file"."webpublicUrl" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "drive_file"."thumbnailUrl" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "drive_file"."url" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "drive_file"."properties" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "drive_file"."blurhash" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "drive_file"."comment" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "drive_file"."size" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "drive_file"."type" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "drive_file"."name" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "drive_file"."md5" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "drive_file"."userHost" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "drive_file"."userId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "drive_file"."createdAt" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "drive_folder"."parentId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "drive_folder"."userId" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "drive_folder"."name" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "drive_folder"."createdAt" IS NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "log"."createdAt" IS NULL`); | ||||
|     } | ||||
|  | ||||
| } | ||||
							
								
								
									
										159
									
								
								package.json
									
									
									
									
									
								
							
							
						
						
									
										159
									
								
								package.json
									
									
									
									
									
								
							| @@ -1,7 +1,7 @@ | ||||
| { | ||||
| 	"name": "misskey", | ||||
| 	"author": "syuilo <syuilotan@yahoo.co.jp>", | ||||
| 	"version": "12.46.0", | ||||
| 	"version": "12.57.0", | ||||
| 	"codename": "indigo", | ||||
| 	"repository": { | ||||
| 		"type": "git", | ||||
| @@ -30,27 +30,26 @@ | ||||
| 	"resolutions": { | ||||
| 		"chokidar": "^3.3.1", | ||||
| 		"constantinople": "^4.0.1", | ||||
| 		"core-js": "^3.6.5", | ||||
| 		"gulp/gulp-cli/yargs/yargs-parser": "5.0.0-security.0", | ||||
| 		"lodash": "^4.17.19", | ||||
| 		"mocha/serialize-javascript": "^3.1.0" | ||||
| 		"jsonld/rdf-canonize/node-forge": "0.10.0", | ||||
| 		"lodash": "^4.17.20" | ||||
| 	}, | ||||
| 	"dependencies": { | ||||
| 		"@babel/plugin-transform-runtime": "7.11.0", | ||||
| 		"@elastic/elasticsearch": "7.8.0", | ||||
| 		"@fortawesome/fontawesome-svg-core": "1.2.30", | ||||
| 		"@fortawesome/free-brands-svg-icons": "5.14.0", | ||||
| 		"@fortawesome/free-regular-svg-icons": "5.14.0", | ||||
| 		"@fortawesome/free-solid-svg-icons": "5.14.0", | ||||
| 		"@fortawesome/vue-fontawesome": "0.1.10", | ||||
| 		"@fortawesome/fontawesome-svg-core": "1.2.32", | ||||
| 		"@fortawesome/free-brands-svg-icons": "5.15.1", | ||||
| 		"@fortawesome/free-regular-svg-icons": "5.15.1", | ||||
| 		"@fortawesome/free-solid-svg-icons": "5.15.1", | ||||
| 		"@fortawesome/vue-fontawesome": "3.0.0-2", | ||||
| 		"@koa/cors": "3.1.0", | ||||
| 		"@koa/multer": "3.0.0", | ||||
| 		"@koa/router": "9.0.1", | ||||
| 		"@sinonjs/fake-timers": "6.0.1", | ||||
| 		"@syuilo/aiscript": "0.11.0", | ||||
| 		"@syuilo/aiscript": "0.11.1", | ||||
| 		"@types/bcryptjs": "2.4.2", | ||||
| 		"@types/bull": "3.14.0", | ||||
| 		"@types/cbor": "5.0.0", | ||||
| 		"@types/cbor": "5.0.1", | ||||
| 		"@types/dateformat": "3.0.1", | ||||
| 		"@types/double-ended-queue": "2.1.1", | ||||
| 		"@types/escape-regexp": "0.0.0", | ||||
| @@ -93,75 +92,74 @@ | ||||
| 		"@types/request-stats": "3.0.0", | ||||
| 		"@types/rimraf": "3.0.0", | ||||
| 		"@types/seedrandom": "2.4.28", | ||||
| 		"@types/sharp": "0.25.0", | ||||
| 		"@types/sharp": "0.26.0", | ||||
| 		"@types/sinonjs__fake-timers": "6.0.1", | ||||
| 		"@types/speakeasy": "2.0.5", | ||||
| 		"@types/tinycolor2": "1.4.2", | ||||
| 		"@types/tmp": "0.2.0", | ||||
| 		"@types/uuid": "8.0.0", | ||||
| 		"@types/uuid": "8.3.0", | ||||
| 		"@types/web-push": "3.3.0", | ||||
| 		"@types/webpack": "4.41.18", | ||||
| 		"@types/webpack": "4.41.24", | ||||
| 		"@types/webpack-stream": "3.2.11", | ||||
| 		"@types/websocket": "1.0.1", | ||||
| 		"@types/ws": "7.2.6", | ||||
| 		"@typescript-eslint/parser": "3.6.0", | ||||
| 		"@types/ws": "7.2.7", | ||||
| 		"@typescript-eslint/parser": "4.6.1", | ||||
| 		"@vue/compiler-sfc": "3.0.2", | ||||
| 		"abort-controller": "3.0.0", | ||||
| 		"apexcharts": "3.20.0", | ||||
| 		"apexcharts": "3.22.1", | ||||
| 		"autobind-decorator": "2.4.0", | ||||
| 		"autosize": "4.0.2", | ||||
| 		"autwh": "0.1.0", | ||||
| 		"aws-sdk": "2.724.0", | ||||
| 		"aws-sdk": "2.787.0", | ||||
| 		"bcryptjs": "2.4.3", | ||||
| 		"blurhash": "1.1.3", | ||||
| 		"bull": "3.16.0", | ||||
| 		"bull": "3.18.1", | ||||
| 		"cafy": "15.2.1", | ||||
| 		"cbor": "5.0.2", | ||||
| 		"cbor": "5.1.0", | ||||
| 		"chalk": "4.1.0", | ||||
| 		"chart.js": "2.9.3", | ||||
| 		"chart.js": "2.9.4", | ||||
| 		"cli-highlight": "2.1.4", | ||||
| 		"commander": "4.1.1", | ||||
| 		"content-disposition": "0.5.3", | ||||
| 		"core-js": "3.6.5", | ||||
| 		"core-js": "3.7.0", | ||||
| 		"crc-32": "1.2.0", | ||||
| 		"css-loader": "4.1.1", | ||||
| 		"css-loader": "5.0.1", | ||||
| 		"cssnano": "4.1.10", | ||||
| 		"dateformat": "3.0.3", | ||||
| 		"deep-entries": "3.1.0", | ||||
| 		"diskusage": "1.1.3", | ||||
| 		"double-ended-queue": "2.1.0-0", | ||||
| 		"escape-regexp": "0.0.1", | ||||
| 		"eslint": "7.4.0", | ||||
| 		"eslint-plugin-vue": "6.2.2", | ||||
| 		"eventemitter3": "4.0.4", | ||||
| 		"eslint": "7.12.1", | ||||
| 		"eslint-plugin-vue": "7.1.0", | ||||
| 		"eventemitter3": "4.0.7", | ||||
| 		"feed": "4.2.1", | ||||
| 		"fibers": "5.0.0", | ||||
| 		"file-type": "14.6.2", | ||||
| 		"file-type": "16.0.1", | ||||
| 		"fluent-ffmpeg": "2.1.2", | ||||
| 		"glob": "7.1.6", | ||||
| 		"got": "11.8.0", | ||||
| 		"gulp": "4.0.2", | ||||
| 		"gulp-clean-css": "4.3.0", | ||||
| 		"gulp-dart-sass": "1.0.2", | ||||
| 		"gulp-rename": "2.0.0", | ||||
| 		"gulp-replace": "1.0.0", | ||||
| 		"gulp-sourcemaps": "2.6.5", | ||||
| 		"gulp-terser": "1.2.1", | ||||
| 		"gulp-tslint": "8.1.4", | ||||
| 		"gulp-typescript": "6.0.0-alpha.1", | ||||
| 		"hard-source-webpack-plugin": "0.13.1", | ||||
| 		"hcaptcha": "0.0.2", | ||||
| 		"html-minifier": "4.0.0", | ||||
| 		"http-proxy-agent": "4.0.1", | ||||
| 		"http-signature": "1.3.4", | ||||
| 		"http-signature": "1.3.5", | ||||
| 		"https-proxy-agent": "5.0.0", | ||||
| 		"idb-keyval": "3.2.0", | ||||
| 		"insert-text-at-cursor": "0.3.0", | ||||
| 		"is-root": "2.1.0", | ||||
| 		"is-svg": "4.2.1", | ||||
| 		"js-yaml": "3.14.0", | ||||
| 		"jsdom": "16.3.0", | ||||
| 		"jsdom": "16.4.0", | ||||
| 		"json5": "2.1.3", | ||||
| 		"json5-loader": "4.0.0", | ||||
| 		"jsonld": "3.1.1", | ||||
| 		"json5-loader": "4.0.1", | ||||
| 		"jsonld": "3.2.0", | ||||
| 		"jsrsasign": "8.0.20", | ||||
| 		"katex": "0.12.0", | ||||
| 		"koa": "2.13.0", | ||||
| @@ -172,39 +170,39 @@ | ||||
| 		"koa-mount": "4.0.0", | ||||
| 		"koa-send": "5.0.1", | ||||
| 		"koa-slow": "2.1.0", | ||||
| 		"koa-views": "6.3.0", | ||||
| 		"koa-views": "6.3.1", | ||||
| 		"langmap": "0.0.16", | ||||
| 		"lookup-dns-cache": "2.1.0", | ||||
| 		"markdown-it": "11.0.0", | ||||
| 		"markdown-it-anchor": "5.3.0", | ||||
| 		"mocha": "8.0.1", | ||||
| 		"markdown-it": "11.0.1", | ||||
| 		"markdown-it-anchor": "6.0.0", | ||||
| 		"mocha": "8.2.1", | ||||
| 		"moji": "0.5.1", | ||||
| 		"ms": "2.1.2", | ||||
| 		"multer": "1.4.2", | ||||
| 		"nested-property": "2.0.1", | ||||
| 		"node-fetch": "2.6.0", | ||||
| 		"nodemailer": "6.4.10", | ||||
| 		"nprogress": "0.2.0", | ||||
| 		"nested-property": "4.0.0", | ||||
| 		"node-fetch": "2.6.1", | ||||
| 		"nodemailer": "6.4.15", | ||||
| 		"object-assign-deep": "0.4.0", | ||||
| 		"os-utils": "0.0.14", | ||||
| 		"p-cancelable": "2.0.0", | ||||
| 		"parse5": "6.0.1", | ||||
| 		"parsimmon": "1.15.0", | ||||
| 		"pg": "8.3.0", | ||||
| 		"portal-vue": "2.1.7", | ||||
| 		"parsimmon": "1.16.0", | ||||
| 		"pg": "8.4.2", | ||||
| 		"portscanner": "2.2.0", | ||||
| 		"postcss-loader": "3.0.0", | ||||
| 		"prismjs": "1.20.0", | ||||
| 		"probe-image-size": "5.0.0", | ||||
| 		"postcss": "8.1.6", | ||||
| 		"postcss-loader": "4.0.4", | ||||
| 		"prismjs": "1.22.0", | ||||
| 		"probe-image-size": "6.0.0", | ||||
| 		"promise-limit": "2.7.0", | ||||
| 		"promise-sequential": "1.1.1", | ||||
| 		"pug": "2.0.4", | ||||
| 		"punycode": "2.1.1", | ||||
| 		"pureimage": "0.2.1", | ||||
| 		"pureimage": "0.2.5", | ||||
| 		"qrcode": "1.4.4", | ||||
| 		"random-seed": "0.3.0", | ||||
| 		"ratelimiter": "3.4.1", | ||||
| 		"re2": "1.15.4", | ||||
| 		"recaptcha-promise": "0.1.3", | ||||
| 		"re2": "1.15.8", | ||||
| 		"recaptcha-promise": "1.0.0", | ||||
| 		"reconnecting-websocket": "4.4.0", | ||||
| 		"redis": "3.0.2", | ||||
| 		"redis-lock": "0.1.4", | ||||
| @@ -216,54 +214,47 @@ | ||||
| 		"rimraf": "3.0.2", | ||||
| 		"rndstr": "1.0.0", | ||||
| 		"s-age": "1.1.2", | ||||
| 		"sass": "1.26.10", | ||||
| 		"sass-loader": "9.0.2", | ||||
| 		"sass": "1.29.0", | ||||
| 		"sass-loader": "10.0.5", | ||||
| 		"seedrandom": "3.0.5", | ||||
| 		"sharp": "0.25.4", | ||||
| 		"sharp": "0.26.2", | ||||
| 		"speakeasy": "2.0.0", | ||||
| 		"stringz": "2.1.0", | ||||
| 		"style-loader": "1.2.1", | ||||
| 		"style-loader": "2.0.0", | ||||
| 		"summaly": "2.4.0", | ||||
| 		"syslog-pro": "1.0.0", | ||||
| 		"systeminformation": "4.26.10", | ||||
| 		"systeminformation": "4.28.1", | ||||
| 		"syuilo-password-strength": "0.0.1", | ||||
| 		"textarea-caret": "3.1.0", | ||||
| 		"three": "0.117.1", | ||||
| 		"tinycolor2": "1.4.1", | ||||
| 		"tinycolor2": "1.4.2", | ||||
| 		"tmp": "0.2.1", | ||||
| 		"ts-loader": "8.0.1", | ||||
| 		"ts-node": "8.10.2", | ||||
| 		"tslint": "6.1.2", | ||||
| 		"ts-loader": "8.0.9", | ||||
| 		"ts-node": "9.0.0", | ||||
| 		"tslint": "6.1.3", | ||||
| 		"tslint-sonarts": "1.9.0", | ||||
| 		"typeorm": "0.2.25", | ||||
| 		"typescript": "3.9.7", | ||||
| 		"typeorm": "0.2.29", | ||||
| 		"typescript": "4.0.5", | ||||
| 		"ulid": "2.3.0", | ||||
| 		"url-loader": "4.1.0", | ||||
| 		"uuid": "8.3.0", | ||||
| 		"v-animate-css": "0.0.3", | ||||
| 		"url-loader": "4.1.1", | ||||
| 		"uuid": "8.3.1", | ||||
| 		"v-debounce": "0.1.2", | ||||
| 		"vue": "2.6.11", | ||||
| 		"vue": "3.0.2", | ||||
| 		"vue-color": "2.7.1", | ||||
| 		"vue-content-loading": "1.6.0", | ||||
| 		"vue-cropperjs": "4.1.0", | ||||
| 		"vue-i18n": "8.20.0", | ||||
| 		"vue-json-pretty": "1.6.5", | ||||
| 		"vue-loader": "15.9.3", | ||||
| 		"vue-marquee-text-component": "1.1.1", | ||||
| 		"vue-meta": "2.4.0", | ||||
| 		"vue-prism-component": "1.2.0", | ||||
| 		"vue-prism-editor": "0.6.1", | ||||
| 		"vue-router": "3.3.4", | ||||
| 		"vue-draggable-next": "1.0.8", | ||||
| 		"vue-i18n": "9.0.0-beta.6", | ||||
| 		"vue-json-pretty": "1.7.1", | ||||
| 		"vue-loader": "16.0.0-beta.8", | ||||
| 		"vue-prism-editor": "1.2.2", | ||||
| 		"vue-router": "4.0.0-rc.2", | ||||
| 		"vue-style-loader": "4.1.2", | ||||
| 		"vue-svg-inline-loader-corejs3": "1.5.0", | ||||
| 		"vue-template-compiler": "2.6.11", | ||||
| 		"vuedraggable": "2.24.0", | ||||
| 		"vuex": "3.5.1", | ||||
| 		"vuex-persistedstate": "3.0.1", | ||||
| 		"vue-template-compiler": "2.6.12", | ||||
| 		"vuex": "4.0.0-rc.1", | ||||
| 		"vuex-persistedstate": "3.1.0", | ||||
| 		"web-push": "3.4.4", | ||||
| 		"webpack": "5.0.0-beta.22", | ||||
| 		"webpack-cli": "3.3.12", | ||||
| 		"websocket": "1.0.31", | ||||
| 		"webpack": "5.4.0", | ||||
| 		"webpack-cli": "4.2.0", | ||||
| 		"websocket": "1.0.32", | ||||
| 		"ws": "7.3.1", | ||||
| 		"xev": "2.0.1" | ||||
| 	}, | ||||
|   | ||||
							
								
								
									
										21
									
								
								src/@types/nested-property.d.ts
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										21
									
								
								src/@types/nested-property.d.ts
									
									
									
									
										vendored
									
									
								
							| @@ -1,21 +0,0 @@ | ||||
| type Obj = { [key: string]: any }; | ||||
|  | ||||
| declare module 'nested-property' { | ||||
| 	interface IHasNestedPropertyOptions { | ||||
| 		own?: boolean; | ||||
| 	} | ||||
|  | ||||
| 	interface IIsInNestedPropertyOptions { | ||||
| 		validPath?: boolean; | ||||
| 	} | ||||
|  | ||||
| 	export function set<T>(object: T, property: string, value: any): T; | ||||
|  | ||||
| 	export function get(object: Obj, property: string): any; | ||||
|  | ||||
| 	export function has(object: Obj, property: string, options?: IHasNestedPropertyOptions): boolean; | ||||
|  | ||||
| 	export function hasOwn(object: Obj, property: string, options?: IHasNestedPropertyOptions): boolean; | ||||
|  | ||||
| 	export function isIn(object: Obj, property: string, objectInPath: Obj, options?: IIsInNestedPropertyOptions): boolean; | ||||
| } | ||||
							
								
								
									
										12
									
								
								src/client/.eslintrc
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								src/client/.eslintrc
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,12 @@ | ||||
| { | ||||
| 	"globals": { | ||||
| 		"_DEV_": false, | ||||
| 		"_LANGS_": false, | ||||
| 		"_VERSION_": false, | ||||
| 		"_ENV_": false, | ||||
| 		"_PERF_PREFIX_": false, | ||||
| 		"_DATA_TRANSFER_DRIVE_FILE_": false, | ||||
| 		"_DATA_TRANSFER_DRIVE_FOLDER_": false, | ||||
| 		"_DATA_TRANSFER_DECK_COLUMN_": false | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										8
									
								
								src/client/@types/global.d.ts
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								src/client/@types/global.d.ts
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,8 @@ | ||||
| declare const _LANGS_: string[]; | ||||
| declare const _VERSION_: string; | ||||
| declare const _ENV_: string; | ||||
| declare const _DEV_: boolean; | ||||
| declare const _PERF_PREFIX_: string; | ||||
| declare const _DATA_TRANSFER_DRIVE_FILE_: string; | ||||
| declare const _DATA_TRANSFER_DRIVE_FOLDER_: string; | ||||
| declare const _DATA_TRANSFER_DECK_COLUMN_: string; | ||||
							
								
								
									
										12
									
								
								src/client/@types/vuex-shim.d.ts
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								src/client/@types/vuex-shim.d.ts
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,12 @@ | ||||
| import { ComponentCustomProperties } from 'vue'; | ||||
| import { Store } from 'vuex'; | ||||
|  | ||||
| declare module '@vue/runtime-core' { | ||||
| 	// tslint:disable-next-line:no-empty-interface | ||||
| 	interface State { | ||||
| 	} | ||||
|  | ||||
| 	interface ComponentCustomProperties { | ||||
| 		$store: Store<State>; | ||||
| 	} | ||||
| } | ||||
| @@ -1,773 +0,0 @@ | ||||
| <template> | ||||
| <div class="mk-app" v-hotkey.global="keymap"> | ||||
| 	<header class="header" ref="header"> | ||||
| 		<div class="title" ref="title"> | ||||
| 			<transition :name="$store.state.device.animation ? 'header' : ''" mode="out-in" appear> | ||||
| 				<button class="_button back" v-if="canBack" @click="back()"><fa :icon="faChevronLeft"/></button> | ||||
| 			</transition> | ||||
| 			<transition :name="$store.state.device.animation ? 'header' : ''" mode="out-in" appear> | ||||
| 				<div class="body" :key="pageKey"> | ||||
| 					<div class="default"> | ||||
| 						<portal-target name="avatar" slim/> | ||||
| 						<h1 class="title"><portal-target name="icon" slim/><portal-target name="title" slim/></h1> | ||||
| 					</div> | ||||
| 					<div class="custom"> | ||||
| 						<portal-target name="header" slim/> | ||||
| 					</div> | ||||
| 				</div> | ||||
| 			</transition> | ||||
| 		</div> | ||||
| 		<div class="sub"> | ||||
| 			<template v-if="$store.getters.isSignedIn"> | ||||
| 				<button v-if="widgetsEditMode" class="_button edit active" @click="widgetsEditMode = false"><fa :icon="faGripVertical"/></button> | ||||
| 				<button v-else class="_button edit" @click="widgetsEditMode = true"><fa :icon="faGripVertical"/></button> | ||||
| 			</template> | ||||
| 			<div class="search"> | ||||
| 				<fa :icon="faSearch"/> | ||||
| 				<input type="search" :placeholder="$t('search')" v-model="searchQuery" v-autocomplete="{ model: 'searchQuery' }" :disabled="searchWait" @keypress="searchKeypress"/> | ||||
| 			</div> | ||||
| 			<button v-if="$store.getters.isSignedIn" class="post _buttonPrimary" @click="post()"><fa :icon="faPencilAlt"/></button> | ||||
| 			<x-clock v-if="isDesktop" class="clock"/> | ||||
| 		</div> | ||||
| 	</header> | ||||
|  | ||||
| 	<x-sidebar ref="nav" @change-view-mode="calcHeaderWidth"/> | ||||
|  | ||||
| 	<div class="contents" ref="contents" :class="{ wallpaper }"> | ||||
| 		<main ref="main"> | ||||
| 			<div class="content"> | ||||
| 				<transition :name="$store.state.device.animation ? 'page' : ''" mode="out-in" @enter="onTransition"> | ||||
| 					<keep-alive :include="['index']"> | ||||
| 						<router-view></router-view> | ||||
| 					</keep-alive> | ||||
| 				</transition> | ||||
| 			</div> | ||||
| 			<div class="powerd-by" :class="{ visible: !$store.getters.isSignedIn }"> | ||||
| 				<b><router-link to="/">{{ host }}</router-link></b> | ||||
| 				<small>Powered by <a href="https://github.com/syuilo/misskey" target="_blank">Misskey</a></small> | ||||
| 			</div> | ||||
| 		</main> | ||||
|  | ||||
| 		<template v-if="isDesktop"> | ||||
| 			<div v-for="place in ['left', 'right']" ref="widgets" class="widgets" :class="{ edit: widgetsEditMode, fixed: $store.state.device.fixedWidgetsPosition, empty: widgets[place].length === 0 && !widgetsEditMode }" :key="place"> | ||||
| 				<div class="spacer"></div> | ||||
| 				<div class="container" v-if="widgetsEditMode"> | ||||
| 					<mk-button primary @click="addWidget(place)" class="add"><fa :icon="faPlus"/></mk-button> | ||||
| 					<x-draggable | ||||
| 						:list="widgets[place]" | ||||
| 						handle=".handle" | ||||
| 						animation="150" | ||||
| 						class="sortable" | ||||
| 						@sort="onWidgetSort" | ||||
| 					> | ||||
| 						<div v-for="widget in widgets[place]" class="customize-container _panel" :key="widget.id"> | ||||
| 							<header> | ||||
| 								<span class="handle"><fa :icon="faBars"/></span>{{ $t('_widgets.' + widget.name) }}<button class="remove _button" @click="removeWidget(widget)"><fa :icon="faTimes"/></button> | ||||
| 							</header> | ||||
| 							<div @click="widgetFunc(widget.id)"> | ||||
| 								<component class="_close_ _forceContainerFull_" :is="`mkw-${widget.name}`" :widget="widget" :ref="widget.id" :is-customize-mode="true"/> | ||||
| 							</div> | ||||
| 						</div> | ||||
| 					</x-draggable> | ||||
| 				</div> | ||||
| 				<div class="container" v-else> | ||||
| 					<component v-for="widget in widgets[place]" class="_close_ _forceContainerFull_" :is="`mkw-${widget.name}`" :key="widget.id" :ref="widget.id" :widget="widget"/> | ||||
| 				</div> | ||||
| 			</div> | ||||
| 		</template> | ||||
| 	</div> | ||||
|  | ||||
| 	<div class="buttons" :class="{ navHidden }"> | ||||
| 		<button class="button nav _button" @click="showNav" ref="navButton"><fa :icon="faBars"/><i v-if="navIndicated"><fa :icon="faCircle"/></i></button> | ||||
| 		<button v-if="$route.name === 'index'" class="button home _button" @click="top()"><fa :icon="faHome"/></button> | ||||
| 		<button v-else class="button home _button" @click="$router.push('/')"><fa :icon="faHome"/></button> | ||||
| 		<button v-if="$store.getters.isSignedIn" class="button notifications _button" @click="$router.push('/my/notifications')"><fa :icon="faBell"/><i v-if="$store.state.i.hasUnreadNotification"><fa :icon="faCircle"/></i></button> | ||||
| 		<button v-if="$store.getters.isSignedIn" class="button post _buttonPrimary" @click="post()"><fa :icon="faPencilAlt"/></button> | ||||
| 	</div> | ||||
|  | ||||
| 	<button v-if="$store.getters.isSignedIn" class="post _buttonPrimary" :class="{ navHidden }" @click="post()"><fa :icon="faPencilAlt"/></button> | ||||
|  | ||||
| 	<stream-indicator v-if="$store.getters.isSignedIn"/> | ||||
| </div> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import Vue from 'vue'; | ||||
| import { faGripVertical, faChevronLeft, faHashtag, faBroadcastTower, faFireAlt, faEllipsisH, faPencilAlt, faBars, faTimes, faSearch, faUserCog, faCog, faUser, faHome, faStar, faCircle, faAt, faListUl, faPlus, faUserClock, faUsers, faTachometerAlt, faExchangeAlt, faGlobe, faChartBar, faCloud, faServer, faInfoCircle, faQuestionCircle, faProjectDiagram } from '@fortawesome/free-solid-svg-icons'; | ||||
| import { faBell, faEnvelope, faLaugh, faComments } from '@fortawesome/free-regular-svg-icons'; | ||||
| import { v4 as uuid } from 'uuid'; | ||||
| import { host } from './config'; | ||||
| import { search } from './scripts/search'; | ||||
| import { StickySidebar } from './scripts/sticky-sidebar'; | ||||
| import { widgets } from './widgets'; | ||||
| import XSidebar from './components/sidebar.vue'; | ||||
|  | ||||
| const DESKTOP_THRESHOLD = 1100; | ||||
|  | ||||
| export default Vue.extend({ | ||||
| 	components: { | ||||
| 		XSidebar, | ||||
| 		XClock: () => import('./components/header-clock.vue').then(m => m.default), | ||||
| 		MkButton: () => import('./components/ui/button.vue').then(m => m.default), | ||||
| 		XDraggable: () => import('vuedraggable'), | ||||
| 	}, | ||||
|  | ||||
| 	data() { | ||||
| 		return { | ||||
| 			host: host, | ||||
| 			pageKey: 0, | ||||
| 			searching: false, | ||||
| 			connection: null, | ||||
| 			searchQuery: '', | ||||
| 			searchWait: false, | ||||
| 			widgetsEditMode: false, | ||||
| 			isDesktop: window.innerWidth >= DESKTOP_THRESHOLD, | ||||
| 			canBack: false, | ||||
| 			menuDef: this.$store.getters.nav({}), | ||||
| 			navHidden: false, | ||||
| 			wallpaper: localStorage.getItem('wallpaper') != null, | ||||
| 			faGripVertical, faChevronLeft, faComments, faHashtag, faBroadcastTower, faFireAlt, faEllipsisH, faPencilAlt, faBars, faTimes, faBell, faSearch, faUserCog, faCog, faUser, faHome, faStar, faCircle, faAt, faEnvelope, faListUl, faPlus, faUserClock, faLaugh, faUsers, faTachometerAlt, faExchangeAlt, faGlobe, faChartBar, faCloud, faServer, faProjectDiagram | ||||
| 		}; | ||||
| 	}, | ||||
|  | ||||
| 	computed: { | ||||
| 		keymap(): any { | ||||
| 			return { | ||||
| 				'd': () => { | ||||
| 					if (this.$store.state.device.syncDeviceDarkMode) return; | ||||
| 					this.$store.commit('device/set', { key: 'darkMode', value: !this.$store.state.device.darkMode }); | ||||
| 				}, | ||||
| 				'p': this.post, | ||||
| 				'n': this.post, | ||||
| 				's': this.search, | ||||
| 				'h|/': this.help | ||||
| 			}; | ||||
| 		}, | ||||
|  | ||||
| 		widgets(): any { | ||||
| 			if (this.$store.getters.isSignedIn) { | ||||
| 				const widgets = this.$store.state.deviceUser.widgets; | ||||
| 				return { | ||||
| 					left: widgets.filter(x => x.place === 'left'), | ||||
| 					right: widgets.filter(x => x.place == null || x.place === 'right'), | ||||
| 					mobile: widgets.filter(x => x.place === 'mobile'), | ||||
| 				}; | ||||
| 			} else { | ||||
| 				const right = [{ | ||||
| 					name: 'calendar', | ||||
| 					id: 'b', place: 'right', data: {} | ||||
| 				}, { | ||||
| 					name: 'trends', | ||||
| 					id: 'c', place: 'right', data: {} | ||||
| 				}]; | ||||
|  | ||||
| 				if (this.$route.name !== 'index') { | ||||
| 					right.unshift({ | ||||
| 						name: 'welcome', | ||||
| 						id: 'a', place: 'right', data: {} | ||||
| 					}); | ||||
| 				} | ||||
|  | ||||
| 				return { | ||||
| 					left: [], | ||||
| 					right, | ||||
| 					mobile: [], | ||||
| 				}; | ||||
| 			} | ||||
| 		}, | ||||
|  | ||||
| 		menu(): string[] { | ||||
| 			return this.$store.state.deviceUser.menu; | ||||
| 		}, | ||||
|  | ||||
| 		navIndicated(): boolean { | ||||
| 			if (!this.$store.getters.isSignedIn) return false; | ||||
| 			for (const def in this.menuDef) { | ||||
| 				if (def === 'notifications') continue; // 通知は下にボタンとして表示されてるから | ||||
| 				if (this.menuDef[def].indicated) return true; | ||||
| 			} | ||||
| 			return false; | ||||
| 		} | ||||
| 	}, | ||||
|  | ||||
| 	watch: { | ||||
| 		$route(to, from) { | ||||
| 			this.pageKey++; | ||||
| 			this.canBack = (window.history.length > 0 && !['index'].includes(to.name)); | ||||
| 		}, | ||||
|  | ||||
| 		isDesktop() { | ||||
| 			this.$nextTick(() => { | ||||
| 				this.attachSticky(); | ||||
| 			}); | ||||
| 		} | ||||
| 	}, | ||||
|  | ||||
| 	created() { | ||||
| 		document.documentElement.style.overflowY = 'scroll'; | ||||
|  | ||||
| 		if (this.$store.getters.isSignedIn) { | ||||
| 			this.connection = this.$root.stream.useSharedConnection('main'); | ||||
| 			this.connection.on('notification', this.onNotification); | ||||
|  | ||||
| 			if (this.$store.state.deviceUser.widgets.length === 0) { | ||||
| 				this.$store.commit('deviceUser/setWidgets', [{ | ||||
| 					name: 'calendar', | ||||
| 					id: 'a', place: 'right', data: {} | ||||
| 				}, { | ||||
| 					name: 'notifications', | ||||
| 					id: 'b', place: 'right', data: {} | ||||
| 				}, { | ||||
| 					name: 'trends', | ||||
| 					id: 'c', place: 'right', data: {} | ||||
| 				}]); | ||||
| 			} | ||||
| 		} | ||||
| 	}, | ||||
|  | ||||
| 	mounted() { | ||||
| 		this.adjustTitlePosition(); | ||||
|  | ||||
| 		const ro = new ResizeObserver((entries, observer) => { | ||||
| 			this.adjustTitlePosition(); | ||||
| 		}); | ||||
|  | ||||
| 		ro.observe(this.$refs.contents); | ||||
|  | ||||
| 		window.addEventListener('resize', this.adjustTitlePosition, { passive: true }); | ||||
|  | ||||
| 		if (!this.isDesktop) { | ||||
| 			window.addEventListener('resize', () => { | ||||
| 				if (window.innerWidth >= DESKTOP_THRESHOLD) this.isDesktop = true; | ||||
| 			}, { passive: true }); | ||||
| 		} | ||||
|  | ||||
| 		// widget follow | ||||
| 		this.attachSticky(); | ||||
|  | ||||
| 		this.$nextTick(() => { | ||||
| 			this.calcHeaderWidth(); | ||||
| 		}); | ||||
| 	}, | ||||
|  | ||||
| 	methods: { | ||||
| 		adjustTitlePosition() { | ||||
| 			const left = this.$refs.main.getBoundingClientRect().left - this.$refs.nav.$el.offsetWidth; | ||||
| 			if (left >= 0) { | ||||
| 				this.$refs.title.style.left = left + 'px'; | ||||
| 			} | ||||
| 		}, | ||||
|  | ||||
| 		calcHeaderWidth() { | ||||
| 			const navWidth = this.$refs.nav.$el.offsetWidth; | ||||
| 			this.navHidden = navWidth === 0; | ||||
| 			this.$refs.header.style.width = `calc(100% - ${navWidth}px)`; | ||||
| 			this.adjustTitlePosition(); | ||||
| 		}, | ||||
|  | ||||
| 		showNav() { | ||||
| 			this.$refs.nav.show(); | ||||
| 		}, | ||||
|  | ||||
| 		attachSticky() { | ||||
| 			if (!this.isDesktop) return; | ||||
| 			if (this.$store.state.device.fixedWidgetsPosition) return; | ||||
|  | ||||
| 			const stickyWidgetColumns = this.$refs.widgets.map(w => new StickySidebar(w.children[1], w.children[0], w.offsetTop)); | ||||
| 			window.addEventListener('scroll', () => { | ||||
| 				for (const stickyWidgetColumn of stickyWidgetColumns) { | ||||
| 					stickyWidgetColumn.calc(window.scrollY); | ||||
| 				} | ||||
| 			}, { passive: true }); | ||||
| 		}, | ||||
|  | ||||
| 		top() { | ||||
| 			window.scroll({ top: 0, behavior: 'smooth' }); | ||||
| 		}, | ||||
|  | ||||
| 		help() { | ||||
| 			this.$router.push('/docs/keyboard-shortcut'); | ||||
| 		}, | ||||
|  | ||||
| 		back() { | ||||
| 			if (this.canBack) window.history.back(); | ||||
| 		}, | ||||
|  | ||||
| 		onTransition() { | ||||
| 			if (window._scroll) window._scroll(); | ||||
| 		}, | ||||
|  | ||||
| 		post() { | ||||
| 			this.$root.post(); | ||||
| 		}, | ||||
|  | ||||
| 		search() { | ||||
| 			if (this.searching) return; | ||||
|  | ||||
| 			this.$root.dialog({ | ||||
| 				title: this.$t('search'), | ||||
| 				input: true | ||||
| 			}).then(async ({ canceled, result: query }) => { | ||||
| 				if (canceled || query == null || query === '') return; | ||||
|  | ||||
| 				this.searching = true; | ||||
| 				search(this, query).finally(() => { | ||||
| 					this.searching = false; | ||||
| 				}); | ||||
| 			}); | ||||
| 		}, | ||||
|  | ||||
| 		searchKeypress(e) { | ||||
| 			if (e.keyCode === 13) { | ||||
| 				this.searchWait = true; | ||||
| 				search(this, this.searchQuery).finally(() => { | ||||
| 					this.searchWait = false; | ||||
| 					this.searchQuery = ''; | ||||
| 				}); | ||||
| 			} | ||||
| 		}, | ||||
|  | ||||
| 		async onNotification(notification) { | ||||
| 			if (document.visibilityState === 'visible') { | ||||
| 				this.$root.stream.send('readNotification', { | ||||
| 					id: notification.id | ||||
| 				}); | ||||
|  | ||||
| 				this.$root.new(await import('./components/toast.vue').then(m => m.default), { | ||||
| 					notification | ||||
| 				}); | ||||
| 			} | ||||
|  | ||||
| 			this.$root.sound('notification'); | ||||
| 		}, | ||||
|  | ||||
| 		widgetFunc(id) { | ||||
| 			this.$refs[id][0].setting(); | ||||
| 		}, | ||||
|  | ||||
| 		onWidgetSort() { | ||||
| 			this.saveHome(); | ||||
| 		}, | ||||
|  | ||||
| 		async addWidget(place) { | ||||
| 			const { canceled, result: widget } = await this.$root.dialog({ | ||||
| 				type: null, | ||||
| 				title: this.$t('chooseWidget'), | ||||
| 				select: { | ||||
| 					items: widgets.map(widget => ({ | ||||
| 						value: widget, | ||||
| 						text: this.$t('_widgets.' + widget), | ||||
| 					})) | ||||
| 				}, | ||||
| 				showCancelButton: true | ||||
| 			}); | ||||
| 			if (canceled) return; | ||||
|  | ||||
| 			this.$store.commit('deviceUser/addWidget', { | ||||
| 				name: widget, | ||||
| 				id: uuid(), | ||||
| 				place: place, | ||||
| 				data: {} | ||||
| 			}); | ||||
| 		}, | ||||
|  | ||||
| 		removeWidget(widget) { | ||||
| 			this.$store.commit('deviceUser/removeWidget', widget); | ||||
| 		}, | ||||
|  | ||||
| 		saveHome() { | ||||
| 			this.$store.commit('deviceUser/setWidgets', [...this.widgets.left, ...this.widgets.right, ...this.widgets.mobile]); | ||||
| 		} | ||||
| 	} | ||||
| }); | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" scoped> | ||||
| .mk-app { | ||||
| 	$header-height: 60px; | ||||
| 	$main-width: 670px; | ||||
| 	$ui-font-size: 1em; // TODO: どこかに集約したい | ||||
| 	$header-sub-hide-threshold: 1090px; | ||||
| 	$left-widgets-hide-threshold: 1600px; | ||||
| 	$right-widgets-hide-threshold: 1090px; | ||||
|  | ||||
| 	// ほんとは単に 100vh と書きたいところだが... https://css-tricks.com/the-trick-to-viewport-units-on-mobile/ | ||||
| 	min-height: calc(var(--vh, 1vh) * 100); | ||||
| 	box-sizing: border-box; | ||||
| 	padding-top: $header-height; | ||||
|  | ||||
| 	&, > .header > .body { | ||||
| 		display: flex; | ||||
| 		margin: 0 auto; | ||||
| 	} | ||||
|  | ||||
| 	> .header { | ||||
| 		position: fixed; | ||||
| 		z-index: 1000; | ||||
| 		top: 0; | ||||
| 		right: 0; | ||||
| 		height: $header-height; | ||||
| 		width: 100%; | ||||
| 		//background-color: var(--panel); | ||||
| 		-webkit-backdrop-filter: blur(32px); | ||||
| 		backdrop-filter: blur(32px); | ||||
| 		background-color: var(--header); | ||||
| 		border-bottom: solid 1px var(--divider); | ||||
|  | ||||
| 		> .title { | ||||
| 			position: relative; | ||||
| 			line-height: $header-height; | ||||
| 			height: $header-height; | ||||
| 			max-width: $main-width; | ||||
| 			text-align: center; | ||||
|  | ||||
| 			> .back { | ||||
| 				position: absolute; | ||||
| 				z-index: 1; | ||||
| 				top: 0; | ||||
| 				left: 0; | ||||
| 				height: $header-height; | ||||
| 				width: $header-height; | ||||
| 			} | ||||
|  | ||||
| 			> .body { | ||||
| 				white-space: nowrap; | ||||
| 				overflow: hidden; | ||||
| 				text-overflow: ellipsis; | ||||
| 				height: $header-height; | ||||
|  | ||||
| 				> .default { | ||||
| 					padding: 0 $header-height; | ||||
|  | ||||
| 					> .avatar { | ||||
| 						$size: 32px; | ||||
| 						display: inline-block; | ||||
| 						width: $size; | ||||
| 						height: $size; | ||||
| 						vertical-align: bottom; | ||||
| 						margin: (($header-height - $size) / 2) 8px (($header-height - $size) / 2) 0; | ||||
| 					} | ||||
|  | ||||
| 					> .title { | ||||
| 						display: inline-block; | ||||
| 						font-size: $ui-font-size; | ||||
| 						margin: 0; | ||||
| 						line-height: $header-height; | ||||
|  | ||||
| 						> [data-icon] { | ||||
| 							margin-right: 8px; | ||||
| 						} | ||||
| 					} | ||||
| 				} | ||||
|  | ||||
| 				> .custom { | ||||
| 					position: absolute; | ||||
| 					top: 0; | ||||
| 					left: 0; | ||||
| 					height: 100%; | ||||
| 					width: 100%; | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		> .sub { | ||||
| 			$post-button-size: 42px; | ||||
| 			$post-button-margin: (($header-height - $post-button-size) / 2); | ||||
| 			display: flex; | ||||
| 			align-items: center; | ||||
| 			position: absolute; | ||||
| 			top: 0; | ||||
| 			right: 16px; | ||||
| 			height: $header-height; | ||||
|  | ||||
| 			@media (max-width: $header-sub-hide-threshold) { | ||||
| 				display: none; | ||||
| 			} | ||||
|  | ||||
| 			> .edit { | ||||
| 				padding: 16px; | ||||
|  | ||||
| 				&.active { | ||||
| 					color: var(--accent); | ||||
| 				} | ||||
| 			} | ||||
|  | ||||
| 			> .search { | ||||
| 				position: relative; | ||||
|  | ||||
| 				> input { | ||||
| 					width: 220px; | ||||
| 					box-sizing: border-box; | ||||
| 					margin-right: 8px; | ||||
| 					padding: 0 12px 0 42px; | ||||
| 					font-size: 1rem; | ||||
| 					line-height: 38px; | ||||
| 					border: none; | ||||
| 					border-radius: 38px; | ||||
| 					color: var(--fg); | ||||
| 					background: var(--bg); | ||||
| 					-webkit-appearance: textfield; | ||||
|  | ||||
| 					&:focus { | ||||
| 						outline: none; | ||||
| 					} | ||||
| 				} | ||||
|  | ||||
| 				> [data-icon] { | ||||
| 					position: absolute; | ||||
| 					top: 0; | ||||
| 					left: 16px; | ||||
| 					height: 100%; | ||||
| 					pointer-events: none; | ||||
| 					font-size: 16px; | ||||
| 				} | ||||
| 			} | ||||
|  | ||||
| 			> .post { | ||||
| 				width: $post-button-size; | ||||
| 				height: $post-button-size; | ||||
| 				margin-left: $post-button-margin; | ||||
| 				border-radius: 100%; | ||||
| 				font-size: 16px; | ||||
| 			} | ||||
|  | ||||
| 			> .clock { | ||||
| 				margin-left: 8px; | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	> .contents { | ||||
| 		display: flex; | ||||
| 		margin: 0 auto; | ||||
| 		min-width: 0; | ||||
|  | ||||
| 		&.wallpaper { | ||||
| 			background: var(--wallpaperOverlay); | ||||
| 			backdrop-filter: blur(4px); | ||||
| 		} | ||||
|  | ||||
| 		> main { | ||||
| 			width: $main-width; | ||||
| 			min-width: 0; | ||||
|  | ||||
| 			> .content { | ||||
| 				> * { | ||||
| 					// ほんとは単に calc(100vh - #{$header-height}) と書きたいところだが... https://css-tricks.com/the-trick-to-viewport-units-on-mobile/ | ||||
| 					min-height: calc((var(--vh, 1vh) * 100) - #{$header-height}); | ||||
| 					box-sizing: border-box; | ||||
| 					padding: var(--margin); | ||||
|  | ||||
| 					&.full { | ||||
| 						padding: 0 var(--margin); | ||||
| 					} | ||||
| 				} | ||||
| 			} | ||||
|  | ||||
| 			> .powerd-by { | ||||
| 				font-size: 14px; | ||||
| 				text-align: center; | ||||
| 				margin: 32px 0; | ||||
| 				visibility: hidden; | ||||
|  | ||||
| 				&.visible { | ||||
| 					visibility: visible; | ||||
| 				} | ||||
|  | ||||
| 				&:not(.visible) { | ||||
| 					@media (min-width: 850px) { | ||||
| 						display: none; | ||||
| 					} | ||||
| 				} | ||||
|  | ||||
| 				@media (max-width: 500px) { | ||||
| 					margin-top: 16px; | ||||
| 				} | ||||
|  | ||||
| 				> small { | ||||
| 					display: block; | ||||
| 					margin-top: 8px; | ||||
| 					opacity: 0.5; | ||||
|  | ||||
| 					@media (max-width: 500px) { | ||||
| 						margin-top: 4px; | ||||
| 					} | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		> .widgets { | ||||
| 			padding: 0 var(--margin); | ||||
| 			box-shadow: 1px 0 0 0 var(--divider), -1px 0 0 0 var(--divider); | ||||
|  | ||||
| 			&.fixed { | ||||
| 				position: sticky; | ||||
| 				overflow: auto; | ||||
| 				// ほんとは単に calc(100vh - #{$header-height}) と書きたいところだが... https://css-tricks.com/the-trick-to-viewport-units-on-mobile/ | ||||
| 				height: calc((var(--vh, 1vh) * 100) - #{$header-height}); | ||||
| 				top: $header-height; | ||||
| 			} | ||||
|  | ||||
| 			&:first-of-type { | ||||
| 				order: -1; | ||||
|  | ||||
| 				@media (max-width: $left-widgets-hide-threshold) { | ||||
| 					display: none; | ||||
| 				} | ||||
| 			} | ||||
|  | ||||
| 			&.empty { | ||||
| 				display: none; | ||||
| 			} | ||||
|  | ||||
| 			@media (max-width: $right-widgets-hide-threshold) { | ||||
| 				display: none; | ||||
| 			} | ||||
|  | ||||
| 			> .container { | ||||
| 				position: sticky; | ||||
| 				height: min-content; | ||||
| 				// ほんとは単に calc(100vh - #{$header-height}) と書きたいところだが... https://css-tricks.com/the-trick-to-viewport-units-on-mobile/ | ||||
| 				min-height: calc((var(--vh, 1vh) * 100) - #{$header-height}); | ||||
| 				padding: var(--margin) 0; | ||||
| 				box-sizing: border-box; | ||||
|  | ||||
| 				> * { | ||||
| 					margin: var(--margin) 0; | ||||
| 					width: 300px; | ||||
|  | ||||
| 					&:first-child { | ||||
| 						margin-top: 0; | ||||
| 					} | ||||
|  | ||||
| 					&:last-child { | ||||
| 						margin-bottom: 0; | ||||
| 					} | ||||
| 				} | ||||
| 			} | ||||
|  | ||||
| 			> .add { | ||||
| 				margin: 0 auto; | ||||
| 			} | ||||
|  | ||||
| 			.customize-container { | ||||
| 				margin: 8px 0; | ||||
|  | ||||
| 				> header { | ||||
| 					position: relative; | ||||
| 					line-height: 32px; | ||||
|  | ||||
| 					> .handle { | ||||
| 						padding: 0 8px; | ||||
| 						cursor: move; | ||||
| 					} | ||||
|  | ||||
| 					> .remove { | ||||
| 						position: absolute; | ||||
| 						top: 0; | ||||
| 						right: 0; | ||||
| 						padding: 0 8px; | ||||
| 						line-height: 32px; | ||||
| 					} | ||||
| 				} | ||||
|  | ||||
| 				> div { | ||||
| 					padding: 8px; | ||||
|  | ||||
| 					> * { | ||||
| 						pointer-events: none; | ||||
| 					} | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	> .post { | ||||
| 		display: block; | ||||
| 		position: fixed; | ||||
| 		z-index: 1000; | ||||
| 		bottom: 32px; | ||||
| 		right: 32px; | ||||
| 		width: 64px; | ||||
| 		height: 64px; | ||||
| 		border-radius: 100%; | ||||
| 		box-shadow: 0 3px 5px -1px rgba(0, 0, 0, 0.2), 0 6px 10px 0 rgba(0, 0, 0, 0.14), 0 1px 18px 0 rgba(0, 0, 0, 0.12); | ||||
| 		font-size: 22px; | ||||
|  | ||||
| 		&.navHidden { | ||||
| 			display: none; | ||||
| 		} | ||||
|  | ||||
| 		@media (min-width: ($header-sub-hide-threshold + 1px)) { | ||||
| 			display: none; | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	> .buttons { | ||||
| 		position: fixed; | ||||
| 		z-index: 1000; | ||||
| 		bottom: 0; | ||||
| 		padding: 0 32px 32px 32px; | ||||
| 		display: flex; | ||||
| 		width: 100%; | ||||
| 		box-sizing: border-box; | ||||
| 		background: linear-gradient(0deg, var(--bg), var(--X1)); | ||||
|  | ||||
| 		@media (max-width: 500px) { | ||||
| 			padding: 0 16px 16px 16px; | ||||
| 		} | ||||
|  | ||||
| 		&:not(.navHidden) { | ||||
| 			display: none; | ||||
| 		} | ||||
|  | ||||
| 		> .button { | ||||
| 			position: relative; | ||||
| 			padding: 0; | ||||
| 			margin: auto; | ||||
| 			width: 64px; | ||||
| 			height: 64px; | ||||
| 			border-radius: 100%; | ||||
| 			box-shadow: 0 3px 5px -1px rgba(0, 0, 0, 0.2), 0 6px 10px 0 rgba(0, 0, 0, 0.14), 0 1px 18px 0 rgba(0, 0, 0, 0.12); | ||||
|  | ||||
| 			&:first-child { | ||||
| 				margin-left: 0; | ||||
| 			} | ||||
|  | ||||
| 			&:last-child { | ||||
| 				margin-right: 0; | ||||
| 			} | ||||
|  | ||||
| 			> * { | ||||
| 				font-size: 22px; | ||||
| 			} | ||||
|  | ||||
| 			&:disabled { | ||||
| 				cursor: default; | ||||
|  | ||||
| 				> * { | ||||
| 					opacity: 0.5; | ||||
| 				} | ||||
| 			} | ||||
|  | ||||
| 			&:not(.post) { | ||||
| 				background: var(--panel); | ||||
| 				color: var(--fg); | ||||
|  | ||||
| 				&:hover { | ||||
| 					background: var(--X2); | ||||
| 				} | ||||
|  | ||||
| 				> i { | ||||
| 					position: absolute; | ||||
| 					top: 0; | ||||
| 					left: 0; | ||||
| 					color: var(--indicator); | ||||
| 					font-size: 16px; | ||||
| 					animation: blink 1s infinite; | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| </style> | ||||
							
								
								
									
										
											BIN
										
									
								
								src/client/assets/sounds/noizenecio/kick_gaba2.mp3
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								src/client/assets/sounds/noizenecio/kick_gaba2.mp3
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								src/client/assets/sounds/syuilo/reverved.mp3
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								src/client/assets/sounds/syuilo/reverved.mp3
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								src/client/assets/sounds/syuilo/ryukyu.mp3
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								src/client/assets/sounds/syuilo/ryukyu.mp3
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								src/client/assets/sounds/syuilo/square-pico.mp3
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								src/client/assets/sounds/syuilo/square-pico.mp3
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										85
									
								
								src/client/components/abuse-report-window.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										85
									
								
								src/client/components/abuse-report-window.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,85 @@ | ||||
| <template> | ||||
| <XWindow ref="window" :initial-width="400" :initial-height="500" :can-resize="true" @closed="$emit('closed')"> | ||||
| 	<template #header> | ||||
| 		<Fa :icon="faExclamationCircle" style="margin-right: 0.5em;"/> | ||||
| 		<i18n-t keypath="reportAbuseOf" tag="span"> | ||||
| 			<template #name> | ||||
| 				<b><MkAcct :user="user"/></b> | ||||
| 			</template> | ||||
| 		</i18n-t> | ||||
| 	</template> | ||||
| 	<div class="dpvffvvy"> | ||||
| 		<div class="_section"> | ||||
| 			<div class="_content"> | ||||
| 				<MkTextarea v-model:value="comment"> | ||||
| 					<span>{{ $t('details') }}</span> | ||||
| 					<template #desc>{{ $t('fillAbuseReportDescription') }}</template> | ||||
| 				</MkTextarea> | ||||
| 			</div> | ||||
| 		</div> | ||||
| 		<div class="_section"> | ||||
| 			<div class="_content"> | ||||
| 				<MkButton @click="send" primary full :disabled="comment.length === 0">{{ $t('send') }}</MkButton> | ||||
| 			</div> | ||||
| 		</div> | ||||
| 	</div> | ||||
| </XWindow> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import { defineComponent, markRaw } from 'vue'; | ||||
| import { faExclamationCircle } from '@fortawesome/free-solid-svg-icons'; | ||||
| import XWindow from '@/components/ui/window.vue'; | ||||
| import MkTextarea from '@/components/ui/textarea.vue'; | ||||
| import MkButton from '@/components/ui/button.vue'; | ||||
| import * as os from '@/os'; | ||||
|  | ||||
| export default defineComponent({ | ||||
| 	components: { | ||||
| 		XWindow, | ||||
| 		MkTextarea, | ||||
| 		MkButton, | ||||
| 	}, | ||||
|  | ||||
| 	props: { | ||||
| 		user: { | ||||
| 			type: Object, | ||||
| 			required: true, | ||||
| 		}, | ||||
| 		initialComment: { | ||||
| 			type: String, | ||||
| 			required: false, | ||||
| 		}, | ||||
| 	}, | ||||
|  | ||||
| 	emits: ['closed'], | ||||
|  | ||||
| 	data() { | ||||
| 		return { | ||||
| 			comment: this.initialComment || '', | ||||
| 			faExclamationCircle, | ||||
| 		}; | ||||
| 	}, | ||||
|  | ||||
| 	methods: { | ||||
| 		send() { | ||||
| 			os.apiWithDialog('users/report-abuse', { | ||||
| 				userId: this.user.id, | ||||
| 				comment: this.comment, | ||||
| 			}, undefined, res => { | ||||
| 				os.dialog({ | ||||
| 					type: 'success', | ||||
| 					text: this.$t('abuseReported') | ||||
| 				}); | ||||
| 				this.$refs.window.close(); | ||||
| 			}); | ||||
| 		} | ||||
| 	}, | ||||
| }); | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" scoped> | ||||
| .dpvffvvy { | ||||
| 	--section-padding: 16px; | ||||
| } | ||||
| </style> | ||||
| @@ -6,11 +6,11 @@ | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import Vue from 'vue'; | ||||
| import { defineComponent } from 'vue'; | ||||
| import { toUnicode } from 'punycode'; | ||||
| import { host } from '../config'; | ||||
| import { host } from '@/config'; | ||||
|  | ||||
| export default Vue.extend({ | ||||
| export default defineComponent({ | ||||
| 	props: ['user', 'detail'], | ||||
| 	data() { | ||||
| 		return { | ||||
|   | ||||
| @@ -34,10 +34,11 @@ | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import Vue from 'vue'; | ||||
| import { defineComponent } from 'vue'; | ||||
| import * as tinycolor from 'tinycolor2'; | ||||
| import * as os from '@/os'; | ||||
|  | ||||
| export default Vue.extend({ | ||||
| export default defineComponent({ | ||||
| 	data() { | ||||
| 		return { | ||||
| 			now: new Date(), | ||||
| @@ -127,7 +128,7 @@ export default Vue.extend({ | ||||
| 		}); | ||||
| 	}, | ||||
|  | ||||
| 	beforeDestroy() { | ||||
| 	beforeUnmount() { | ||||
| 		this.enabled = false; | ||||
| 	}, | ||||
|  | ||||
|   | ||||
| @@ -1,12 +1,12 @@ | ||||
| <template> | ||||
| <div class="swhvrteh" @contextmenu.prevent="() => {}"> | ||||
| <div class="swhvrteh _popup _shadow" @contextmenu.prevent="() => {}"> | ||||
| 	<ol class="users" ref="suggests" v-if="type === 'user'"> | ||||
| 		<li v-for="user in users" @click="complete(type, user)" @keydown="onKeydown" tabindex="-1" class="user"> | ||||
| 			<img class="avatar" :src="user.avatarUrl"/> | ||||
| 			<span class="name"> | ||||
| 				<mk-user-name :user="user" :key="user.id"/> | ||||
| 				<MkUserName :user="user" :key="user.id"/> | ||||
| 			</span> | ||||
| 			<span class="username">@{{ user | acct }}</span> | ||||
| 			<span class="username">@{{ acct(user) }}</span> | ||||
| 		</li> | ||||
| 		<li @click="chooseUser()" @keydown="onKeydown" tabindex="-1" class="choose">{{ $t('selectUser') }}</li> | ||||
| 	</ol> | ||||
| @@ -28,12 +28,13 @@ | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import Vue from 'vue'; | ||||
| import { defineComponent, markRaw } from 'vue'; | ||||
| import { emojilist } from '../../misc/emojilist'; | ||||
| import contains from '../scripts/contains'; | ||||
| import contains from '@/scripts/contains'; | ||||
| import { twemojiSvgBase } from '../../misc/twemoji-base'; | ||||
| import { getStaticImageUrl } from '../scripts/get-static-image-url'; | ||||
| import MkUserSelect from './user-select.vue'; | ||||
| import { getStaticImageUrl } from '@/scripts/get-static-image-url'; | ||||
| import { acct } from '@/filters/user'; | ||||
| import * as os from '@/os'; | ||||
|  | ||||
| type EmojiDef = { | ||||
| 	emoji: string; | ||||
| @@ -74,7 +75,7 @@ for (const x of lib) { | ||||
|  | ||||
| emjdb.sort((a, b) => a.name.length - b.name.length); | ||||
|  | ||||
| export default Vue.extend({ | ||||
| export default defineComponent({ | ||||
| 	props: { | ||||
| 		type: { | ||||
| 			type: String, | ||||
| @@ -91,11 +92,6 @@ export default Vue.extend({ | ||||
| 			required: true, | ||||
| 		}, | ||||
|  | ||||
| 		complete: { | ||||
| 			type: Function, | ||||
| 			required: true, | ||||
| 		}, | ||||
|  | ||||
| 		close: { | ||||
| 			type: Function, | ||||
| 			required: true, | ||||
| @@ -110,8 +106,15 @@ export default Vue.extend({ | ||||
| 			type: Number, | ||||
| 			required: true, | ||||
| 		}, | ||||
|  | ||||
| 		showing: { | ||||
| 			type: Boolean, | ||||
| 			required: true | ||||
| 		}, | ||||
| 	}, | ||||
|  | ||||
| 	emits: ['done', 'closed'], | ||||
|  | ||||
| 	data() { | ||||
| 		return { | ||||
| 			getStaticImageUrl, | ||||
| @@ -119,24 +122,29 @@ export default Vue.extend({ | ||||
| 			users: [], | ||||
| 			hashtags: [], | ||||
| 			emojis: [], | ||||
| 			items: [], | ||||
| 			select: -1, | ||||
| 			emojilist, | ||||
| 			emojiDb: [] as EmojiDef[] | ||||
| 		} | ||||
| 	}, | ||||
|  | ||||
| 	computed: { | ||||
| 		items(): HTMLCollection { | ||||
| 			return (this.$refs.suggests as Element).children; | ||||
| 		}, | ||||
|  | ||||
| 		useOsNativeEmojis(): boolean { | ||||
| 			return this.$store.state.device.useOsNativeEmojis; | ||||
| 		} | ||||
| 	}, | ||||
|  | ||||
| 	watch: { | ||||
| 		showing() { | ||||
| 			if (!this.showing) { | ||||
| 				this.$emit('closed'); | ||||
| 			} | ||||
| 		} | ||||
| 	}, | ||||
|  | ||||
| 	updated() { | ||||
| 		this.setPosition(); | ||||
| 		this.items = (this.$refs.suggests as Element | undefined)?.children || []; | ||||
| 	}, | ||||
|  | ||||
| 	mounted() { | ||||
| @@ -169,7 +177,7 @@ export default Vue.extend({ | ||||
|  | ||||
| 		emojiDefinitions.sort((a, b) => a.name.length - b.name.length); | ||||
|  | ||||
| 		this.emojiDb = emojiDefinitions.concat(emjdb); | ||||
| 		this.emojiDb = markRaw(emojiDefinitions.concat(emjdb)); | ||||
| 		//#endregion | ||||
|  | ||||
| 		this.textarea.addEventListener('keydown', this.onKeydown); | ||||
| @@ -189,7 +197,7 @@ export default Vue.extend({ | ||||
| 		}); | ||||
| 	}, | ||||
|  | ||||
| 	beforeDestroy() { | ||||
| 	beforeUnmount() { | ||||
| 		this.textarea.removeEventListener('keydown', this.onKeydown); | ||||
|  | ||||
| 		for (const el of Array.from(document.querySelectorAll('body *'))) { | ||||
| @@ -198,6 +206,11 @@ export default Vue.extend({ | ||||
| 	}, | ||||
|  | ||||
| 	methods: { | ||||
| 		complete(type, value) { | ||||
| 			this.$emit('done', { type, value }); | ||||
| 			this.$emit('closed'); | ||||
| 		}, | ||||
|  | ||||
| 		setPosition() { | ||||
| 			if (this.x + this.$el.offsetWidth > window.innerWidth) { | ||||
| 				this.$el.style.left = (window.innerWidth - this.$el.offsetWidth) + 'px'; | ||||
| @@ -236,8 +249,8 @@ export default Vue.extend({ | ||||
| 					this.users = users; | ||||
| 					this.fetching = false; | ||||
| 				} else { | ||||
| 					this.$root.api('users/search', { | ||||
| 						query: this.q, | ||||
| 					os.api('users/search-by-username-and-host', { | ||||
| 						username: this.q, | ||||
| 						limit: 10, | ||||
| 						detail: false | ||||
| 					}).then(users => { | ||||
| @@ -260,7 +273,7 @@ export default Vue.extend({ | ||||
| 						this.hashtags = hashtags; | ||||
| 						this.fetching = false; | ||||
| 					} else { | ||||
| 						this.$root.api('hashtags/search', { | ||||
| 						os.api('hashtags/search', { | ||||
| 							query: this.q, | ||||
| 							limit: 30 | ||||
| 						}).then(hashtags => { | ||||
| @@ -355,6 +368,7 @@ export default Vue.extend({ | ||||
|  | ||||
| 		selectNext() { | ||||
| 			if (++this.select >= this.items.length) this.select = 0; | ||||
| 			if (this.items.length === 0) this.select = -1; | ||||
| 			this.applySelect(); | ||||
| 		}, | ||||
|  | ||||
| @@ -368,20 +382,21 @@ export default Vue.extend({ | ||||
| 				el.removeAttribute('data-selected'); | ||||
| 			} | ||||
|  | ||||
| 			this.items[this.select].setAttribute('data-selected', 'true'); | ||||
| 			(this.items[this.select] as any).focus(); | ||||
| 			if (this.select !== -1) { | ||||
| 				this.items[this.select].setAttribute('data-selected', 'true'); | ||||
| 				(this.items[this.select] as any).focus(); | ||||
| 			} | ||||
| 		}, | ||||
|  | ||||
| 		chooseUser() { | ||||
| 			this.close(); | ||||
| 			const vm = this.$root.new(MkUserSelect, {}); | ||||
| 			vm.$once('selected', user => { | ||||
| 			os.selectUser().then(user => { | ||||
| 				this.complete('user', user); | ||||
| 			}); | ||||
| 			vm.$once('closed', () => { | ||||
| 				this.textarea.focus(); | ||||
| 			}); | ||||
| 		} | ||||
| 		}, | ||||
|  | ||||
| 		acct | ||||
| 	} | ||||
| }); | ||||
| </script> | ||||
| @@ -393,9 +408,6 @@ export default Vue.extend({ | ||||
| 	max-width: 100%; | ||||
| 	margin-top: calc(1em + 8px); | ||||
| 	overflow: hidden; | ||||
| 	background: var(--panel); | ||||
| 	border: solid 1px rgba(#000, 0.1); | ||||
| 	border-radius: 4px; | ||||
| 	transition: top 0.1s ease, left 0.1s ease; | ||||
|  | ||||
| 	> ol { | ||||
|   | ||||
| @@ -1,17 +1,19 @@ | ||||
| <template> | ||||
| <span class="eiwwqkts" :class="{ cat }" :title="user | acct" v-if="disableLink" v-user-preview="disablePreview ? undefined : user.id" @click="onClick"> | ||||
| <span class="eiwwqkts" :class="{ cat }" :title="acct(user)" v-if="disableLink" v-user-preview="disablePreview ? undefined : user.id" @click="onClick"> | ||||
| 	<img class="inner" :src="url"/> | ||||
| </span> | ||||
| <router-link class="eiwwqkts" :class="{ cat }" :to="user | userPage" :title="user | acct" :target="target" v-else v-user-preview="disablePreview ? undefined : user.id"> | ||||
| <MkA class="eiwwqkts" :class="{ cat }" :to="userPage(user)" :title="acct(user)" :target="target" v-else v-user-preview="disablePreview ? undefined : user.id"> | ||||
| 	<img class="inner" :src="url"/> | ||||
| </router-link> | ||||
| </MkA> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import Vue from 'vue'; | ||||
| import { getStaticImageUrl } from '../scripts/get-static-image-url'; | ||||
| import { defineComponent } from 'vue'; | ||||
| import { getStaticImageUrl } from '@/scripts/get-static-image-url'; | ||||
| import { extractAvgColorFromBlurhash } from '@/scripts/extract-avg-color-from-blurhash'; | ||||
| import { acct, userPage } from '../filters/user'; | ||||
|  | ||||
| export default Vue.extend({ | ||||
| export default defineComponent({ | ||||
| 	props: { | ||||
| 		user: { | ||||
| 			type: Object, | ||||
| @@ -30,6 +32,7 @@ export default Vue.extend({ | ||||
| 			default: false | ||||
| 		} | ||||
| 	}, | ||||
| 	emits: ['click'], | ||||
| 	computed: { | ||||
| 		cat(): boolean { | ||||
| 			return this.user.isCat; | ||||
| @@ -42,25 +45,19 @@ export default Vue.extend({ | ||||
| 	}, | ||||
| 	watch: { | ||||
| 		'user.avatarBlurhash'() { | ||||
| 			this.$el.style.color = this.getBlurhashAvgColor(this.user.avatarBlurhash); | ||||
| 			if (this.$el == null) return; | ||||
| 			this.$el.style.color = extractAvgColorFromBlurhash(this.user.avatarBlurhash); | ||||
| 		} | ||||
| 	}, | ||||
| 	mounted() { | ||||
| 		this.$el.style.color = this.getBlurhashAvgColor(this.user.avatarBlurhash); | ||||
| 		this.$el.style.color = extractAvgColorFromBlurhash(this.user.avatarBlurhash); | ||||
| 	}, | ||||
| 	methods: { | ||||
| 		getBlurhashAvgColor(s) { | ||||
| 			return typeof s == 'string' | ||||
| 				? '#' + [...s.slice(2, 6)] | ||||
| 						.map(x => '0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz#$%*+,-.:;=?@[]^_{|}~'.indexOf(x)) | ||||
| 						.reduce((a, c) => a * 83 + c, 0) | ||||
| 						.toString(16) | ||||
| 						.padStart(6, '0') | ||||
| 				: undefined; | ||||
| 		}, | ||||
| 		onClick(e) { | ||||
| 			this.$emit('click', e); | ||||
| 		} | ||||
| 		}, | ||||
| 		acct, | ||||
| 		userPage | ||||
| 	} | ||||
| }); | ||||
| </script> | ||||
|   | ||||
| @@ -1,15 +1,16 @@ | ||||
| <template> | ||||
| <div> | ||||
| 	<div v-for="user in us" :key="user.id" style="display:inline-block;width:32px;height:32px;margin-right:8px;"> | ||||
| 		<mk-avatar :user="user" style="width:32px;height:32px;"/> | ||||
| 		<MkAvatar :user="user" style="width:32px;height:32px;"/> | ||||
| 	</div> | ||||
| </div> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import Vue from 'vue'; | ||||
| import { defineComponent } from 'vue'; | ||||
| import * as os from '@/os'; | ||||
|  | ||||
| export default Vue.extend({ | ||||
| export default defineComponent({ | ||||
| 	props: { | ||||
| 		userIds: { | ||||
| 			required: true | ||||
| @@ -21,7 +22,7 @@ export default Vue.extend({ | ||||
| 		}; | ||||
| 	}, | ||||
| 	async created() { | ||||
| 		this.us = await this.$root.api('users/show', { | ||||
| 		this.us = await os.api('users/show', { | ||||
| 			userIds: this.userIds | ||||
| 		}); | ||||
| 	} | ||||
|   | ||||
| @@ -1,12 +1,12 @@ | ||||
| <template> | ||||
| <div> | ||||
| 	<span v-if="!available">{{ $t('waiting') }}<mk-ellipsis/></span> | ||||
| 	<span v-if="!available">{{ $t('waiting') }}<MkEllipsis/></span> | ||||
| 	<div ref="captcha"></div> | ||||
| </div> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import Vue from 'vue'; | ||||
| import { defineComponent } from 'vue'; | ||||
|  | ||||
| type Captcha = { | ||||
| 	render(container: string | Node, options: { | ||||
| @@ -28,8 +28,9 @@ declare global { | ||||
| 	interface Window extends CaptchaContainer { | ||||
| 	} | ||||
| } | ||||
| import * as os from '@/os'; | ||||
|  | ||||
| export default Vue.extend({ | ||||
| export default defineComponent({ | ||||
| 	props: { | ||||
| 		provider: { | ||||
| 			type: String, | ||||
| @@ -88,7 +89,7 @@ export default Vue.extend({ | ||||
| 		} | ||||
| 	}, | ||||
|  | ||||
| 	beforeDestroy() { | ||||
| 	beforeUnmount() { | ||||
| 		this.reset(); | ||||
| 	}, | ||||
|  | ||||
| @@ -110,7 +111,7 @@ export default Vue.extend({ | ||||
| 			} | ||||
| 		}, | ||||
| 		callback(response?: string) { | ||||
| 			this.$emit('input', typeof response == 'string' ? response : null); | ||||
| 			this.$emit('update:value', typeof response == 'string' ? response : null); | ||||
| 		}, | ||||
| 	}, | ||||
| }); | ||||
|   | ||||
							
								
								
									
										142
									
								
								src/client/components/channel-follow-button.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										142
									
								
								src/client/components/channel-follow-button.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,142 @@ | ||||
| <template> | ||||
| <button class="hdcaacmi _button" | ||||
| 	:class="{ wait, active: isFollowing, full }" | ||||
| 	@click="onClick" | ||||
| 	:disabled="wait" | ||||
| > | ||||
| 	<template v-if="!wait"> | ||||
| 		<template v-if="isFollowing"> | ||||
| 			<span v-if="full">{{ $t('unfollow') }}</span><Fa :icon="faMinus"/> | ||||
| 		</template> | ||||
| 		<template v-else> | ||||
| 			<span v-if="full">{{ $t('follow') }}</span><Fa :icon="faPlus"/> | ||||
| 		</template> | ||||
| 	</template> | ||||
| 	<template v-else> | ||||
| 		<span v-if="full">{{ $t('processing') }}</span><Fa :icon="faSpinner" pulse fixed-width/> | ||||
| 	</template> | ||||
| </button> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import { defineComponent } from 'vue'; | ||||
| import { faSpinner, faPlus, faMinus, } from '@fortawesome/free-solid-svg-icons'; | ||||
| import * as os from '@/os'; | ||||
|  | ||||
| export default defineComponent({ | ||||
| 	props: { | ||||
| 		channel: { | ||||
| 			type: Object, | ||||
| 			required: true | ||||
| 		}, | ||||
| 		full: { | ||||
| 			type: Boolean, | ||||
| 			required: false, | ||||
| 			default: false, | ||||
| 		}, | ||||
| 	}, | ||||
|  | ||||
| 	data() { | ||||
| 		return { | ||||
| 			isFollowing: this.channel.isFollowing, | ||||
| 			wait: false, | ||||
| 			faSpinner, faPlus, faMinus, | ||||
| 		}; | ||||
| 	}, | ||||
|  | ||||
| 	methods: { | ||||
| 		async onClick() { | ||||
| 			this.wait = true; | ||||
|  | ||||
| 			try { | ||||
| 				if (this.isFollowing) { | ||||
| 					await os.api('channels/unfollow', { | ||||
| 						channelId: this.channel.id | ||||
| 					}); | ||||
| 					this.isFollowing = false; | ||||
| 				} else { | ||||
| 					await os.api('channels/follow', { | ||||
| 						channelId: this.channel.id | ||||
| 					}); | ||||
| 					this.isFollowing = true; | ||||
| 				} | ||||
| 			} catch (e) { | ||||
| 				console.error(e); | ||||
| 			} finally { | ||||
| 				this.wait = false; | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| }); | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" scoped> | ||||
| .hdcaacmi { | ||||
| 	position: relative; | ||||
| 	display: inline-block; | ||||
| 	font-weight: bold; | ||||
| 	color: var(--accent); | ||||
| 	background: transparent; | ||||
| 	border: solid 1px var(--accent); | ||||
| 	padding: 0; | ||||
| 	height: 31px; | ||||
| 	font-size: 16px; | ||||
| 	border-radius: 32px; | ||||
| 	background: #fff; | ||||
|  | ||||
| 	&.full { | ||||
| 		padding: 0 8px 0 12px; | ||||
| 		font-size: 14px; | ||||
| 	} | ||||
|  | ||||
| 	&:not(.full) { | ||||
| 		width: 31px; | ||||
| 	} | ||||
|  | ||||
| 	&:focus { | ||||
| 		&:after { | ||||
| 			content: ""; | ||||
| 			pointer-events: none; | ||||
| 			position: absolute; | ||||
| 			top: -5px; | ||||
| 			right: -5px; | ||||
| 			bottom: -5px; | ||||
| 			left: -5px; | ||||
| 			border: 2px solid var(--focus); | ||||
| 			border-radius: 32px; | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	&:hover { | ||||
| 		//background: mix($primary, #fff, 20); | ||||
| 	} | ||||
|  | ||||
| 	&:active { | ||||
| 		//background: mix($primary, #fff, 40); | ||||
| 	} | ||||
|  | ||||
| 	&.active { | ||||
| 		color: #fff; | ||||
| 		background: var(--accent); | ||||
|  | ||||
| 		&:hover { | ||||
| 			background: var(--accentLighten); | ||||
| 			border-color: var(--accentLighten); | ||||
| 		} | ||||
|  | ||||
| 		&:active { | ||||
| 			background: var(--accentDarken); | ||||
| 			border-color: var(--accentDarken); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	&.wait { | ||||
| 		cursor: wait !important; | ||||
| 		opacity: 0.7; | ||||
| 	} | ||||
|  | ||||
| 	> span { | ||||
| 		margin-right: 6px; | ||||
| 	} | ||||
| } | ||||
| </style> | ||||
							
								
								
									
										167
									
								
								src/client/components/channel-preview.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										167
									
								
								src/client/components/channel-preview.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,167 @@ | ||||
| <template> | ||||
| <MkA :to="`/channels/${channel.id}`" class="eftoefju _panel" tabindex="-1"> | ||||
| 	<div class="banner" :style="bannerStyle"> | ||||
| 		<div class="fade"></div> | ||||
| 		<div class="name"><Fa :icon="faSatelliteDish"/> {{ channel.name }}</div> | ||||
| 		<div class="status"> | ||||
| 			<div> | ||||
| 				<Fa :icon="faUsers" fixed-width/> | ||||
| 				<i18n-t keypath="_channel.usersCount" tag="span" style="margin-left: 4px;"> | ||||
| 					<template #n> | ||||
| 						<b>{{ channel.usersCount }}</b> | ||||
| 					</template> | ||||
| 				</i18n-t> | ||||
| 			</div> | ||||
| 			<div> | ||||
| 				<Fa :icon="faPencilAlt" fixed-width/> | ||||
| 				<i18n-t keypath="_channel.notesCount" tag="span" style="margin-left: 4px;"> | ||||
| 					<template #n> | ||||
| 						<b>{{ channel.notesCount }}</b> | ||||
| 					</template> | ||||
| 				</i18n-t> | ||||
| 			</div> | ||||
| 		</div> | ||||
| 	</div> | ||||
| 	<article v-if="channel.description"> | ||||
| 		<p :title="channel.description">{{ channel.description.length > 85 ? channel.description.slice(0, 85) + '…' : channel.description }}</p> | ||||
| 	</article> | ||||
| 	<footer> | ||||
| 		<span v-if="channel.lastNotedAt"> | ||||
| 			{{ $t('updatedAt') }}: <MkTime :time="channel.lastNotedAt"/> | ||||
| 		</span> | ||||
| 	</footer> | ||||
| </MkA> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import { defineComponent } from 'vue'; | ||||
| import { faSatelliteDish, faUsers, faPencilAlt } from '@fortawesome/free-solid-svg-icons'; | ||||
|  | ||||
| export default defineComponent({ | ||||
| 	props: { | ||||
| 		channel: { | ||||
| 			type: Object, | ||||
| 			required: true | ||||
| 		}, | ||||
| 	}, | ||||
|  | ||||
| 	computed: { | ||||
| 		bannerStyle() { | ||||
| 			if (this.channel.bannerUrl) { | ||||
| 				return { backgroundImage: `url(${this.channel.bannerUrl})` }; | ||||
| 			} else { | ||||
| 				return { backgroundColor: '#4c5e6d' }; | ||||
| 			} | ||||
| 		} | ||||
| 	}, | ||||
|  | ||||
| 	data() { | ||||
| 		return { | ||||
| 			faSatelliteDish, faUsers, faPencilAlt, | ||||
| 		}; | ||||
| 	}, | ||||
| }); | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" scoped> | ||||
| .eftoefju { | ||||
| 	display: block; | ||||
| 	overflow: hidden; | ||||
| 	width: 100%; | ||||
|  | ||||
| 	&:hover { | ||||
| 		text-decoration: none; | ||||
| 	} | ||||
|  | ||||
| 	> .banner { | ||||
| 		position: relative; | ||||
| 		width: 100%; | ||||
| 		height: 200px; | ||||
| 		background-position: center; | ||||
| 		background-size: cover; | ||||
|  | ||||
| 		> .fade { | ||||
| 			position: absolute; | ||||
| 			bottom: 0; | ||||
| 			left: 0; | ||||
| 			width: 100%; | ||||
| 			height: 64px; | ||||
| 			background: linear-gradient(0deg, var(--panel), var(--X15)); | ||||
| 		} | ||||
|  | ||||
| 		> .name { | ||||
| 			position: absolute; | ||||
| 			top: 16px; | ||||
| 			left: 16px; | ||||
| 			padding: 12px 16px; | ||||
| 			background: rgba(0, 0, 0, 0.7); | ||||
| 			color: #fff; | ||||
| 			font-size: 1.2em; | ||||
| 		} | ||||
|  | ||||
| 		> .status { | ||||
| 			position: absolute; | ||||
| 			z-index: 1; | ||||
| 			bottom: 16px; | ||||
| 			right: 16px; | ||||
| 			padding: 8px 12px; | ||||
| 			font-size: 80%; | ||||
| 			background: rgba(0, 0, 0, 0.7); | ||||
| 			border-radius: 6px; | ||||
| 			color: #fff; | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	> article { | ||||
| 		padding: 16px; | ||||
|  | ||||
| 		> p { | ||||
| 			margin: 0; | ||||
| 			font-size: 1em; | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	> footer { | ||||
| 		padding: 12px 16px; | ||||
| 		border-top: solid 1px var(--divider); | ||||
|  | ||||
| 		> span { | ||||
| 			opacity: 0.7; | ||||
| 			font-size: 0.9em; | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	@media (max-width: 550px) { | ||||
| 		font-size: 0.9em; | ||||
|  | ||||
| 		> .banner { | ||||
| 			height: 80px; | ||||
|  | ||||
| 			> .status { | ||||
| 				display: none; | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		> article { | ||||
| 			padding: 12px; | ||||
| 		} | ||||
|  | ||||
| 		> footer { | ||||
| 			display: none; | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	@media (max-width: 500px) { | ||||
| 		font-size: 0.8em; | ||||
|  | ||||
| 		> .banner { | ||||
| 			height: 70px; | ||||
| 		} | ||||
|  | ||||
| 		> article { | ||||
| 			padding: 8px; | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| </style> | ||||
| @@ -1,16 +1,14 @@ | ||||
| <template> | ||||
| <x-prism :inline="inline" :language="prismLang">{{ code }}</x-prism> | ||||
| <code v-if="inline" v-html="html" :class="`language-${prismLang}`"></code> | ||||
| <pre v-else :class="`language-${prismLang}`"><code v-html="html" :class="`language-${prismLang}`"></code></pre> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import Vue from 'vue'; | ||||
| import { defineComponent } from 'vue'; | ||||
| import 'prismjs'; | ||||
| import 'prismjs/themes/prism-okaidia.css'; | ||||
| import XPrism from 'vue-prism-component'; | ||||
| export default Vue.extend({ | ||||
| 	components: { | ||||
| 		XPrism | ||||
| 	}, | ||||
|  | ||||
| export default defineComponent({ | ||||
| 	props: { | ||||
| 		code: { | ||||
| 			type: String, | ||||
| @@ -28,6 +26,9 @@ export default Vue.extend({ | ||||
| 	computed: { | ||||
| 		prismLang() { | ||||
| 			return Prism.languages[this.lang] ? this.lang : 'js'; | ||||
| 		}, | ||||
| 		html() { | ||||
| 			return Prism.highlight(this.code, Prism.languages[this.prismLang], this.prismLang); | ||||
| 		} | ||||
| 	} | ||||
| }); | ||||
|   | ||||
| @@ -1,12 +1,13 @@ | ||||
| <template> | ||||
| <x-code :code="code" :lang="lang" :inline="inline"/> | ||||
| <XCode :code="code" :lang="lang" :inline="inline"/> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import Vue from 'vue'; | ||||
| export default Vue.extend({ | ||||
| import { defineComponent, defineAsyncComponent } from 'vue'; | ||||
|  | ||||
| export default defineComponent({ | ||||
| 	components: { | ||||
| 		XCode: () => import('./code-core.vue').then(m => m.default) | ||||
| 		XCode: defineAsyncComponent(() => import('./code-core.vue')) | ||||
| 	}, | ||||
| 	props: { | ||||
| 		code: { | ||||
|   | ||||
| @@ -1,16 +1,16 @@ | ||||
| <template> | ||||
| <button class="nrvgflfuaxwgkxoynpnumyookecqrrvh _button" @click="toggle"> | ||||
| 	<b>{{ value ? this.$t('_cw.hide') : this.$t('_cw.show') }}</b> | ||||
| 	<span v-if="!value">{{ this.label }}</span> | ||||
| <button class="nrvgflfu _button" @click="toggle"> | ||||
| 	<b>{{ value ? $t('_cw.hide') : $t('_cw.show') }}</b> | ||||
| 	<span v-if="!value">{{ label }}</span> | ||||
| </button> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import Vue from 'vue'; | ||||
| import { defineComponent } from 'vue'; | ||||
| import { length } from 'stringz'; | ||||
| import { concat } from '../../prelude/array'; | ||||
|  | ||||
| export default Vue.extend({ | ||||
| export default defineComponent({ | ||||
| 	props: { | ||||
| 		value: { | ||||
| 			type: Boolean, | ||||
| @@ -36,14 +36,14 @@ export default Vue.extend({ | ||||
| 		length, | ||||
|  | ||||
| 		toggle() { | ||||
| 			this.$emit('input', !this.value); | ||||
| 			this.$emit('update:value', !this.value); | ||||
| 		} | ||||
| 	} | ||||
| }); | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" scoped> | ||||
| .nrvgflfuaxwgkxoynpnumyookecqrrvh { | ||||
| .nrvgflfu { | ||||
| 	display: inline-block; | ||||
| 	padding: 4px 8px; | ||||
| 	font-size: 0.7em; | ||||
|   | ||||
| @@ -1,22 +1,22 @@ | ||||
| <template> | ||||
| <component :is="$store.state.device.animation ? 'transition-group' : 'div'" class="sqadhkmv _list_" name="list" tag="div" :data-direction="direction" :data-reversed="reversed ? 'true' : 'false'"> | ||||
| <transition-group class="sqadhkmv _list_" name="list" tag="div" :data-direction="direction" :data-reversed="reversed ? 'true' : 'false'"> | ||||
| 	<template v-for="(item, i) in items"> | ||||
| 		<slot :item="item"></slot> | ||||
| 		<div class="separator" v-if="showDate(i, item)" :key="item.id + '_date'"> | ||||
| 			<p class="date"> | ||||
| 				<span><fa class="icon" :icon="faAngleUp"/>{{ getDateText(item.createdAt) }}</span> | ||||
| 				<span>{{ getDateText(items[i + 1].createdAt) }}<fa class="icon" :icon="faAngleDown"/></span> | ||||
| 				<span><Fa class="icon" :icon="faAngleUp"/>{{ getDateText(item.createdAt) }}</span> | ||||
| 				<span>{{ getDateText(items[i + 1].createdAt) }}<Fa class="icon" :icon="faAngleDown"/></span> | ||||
| 			</p> | ||||
| 		</div> | ||||
| 	</template> | ||||
| </component> | ||||
| </transition-group> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import Vue from 'vue'; | ||||
| import { defineComponent } from 'vue'; | ||||
| import { faAngleUp, faAngleDown } from '@fortawesome/free-solid-svg-icons'; | ||||
|  | ||||
| export default Vue.extend({ | ||||
| export default defineComponent({ | ||||
| 	props: { | ||||
| 		items: { | ||||
| 			type: Array, | ||||
| @@ -82,14 +82,14 @@ export default Vue.extend({ | ||||
| 	} | ||||
|  | ||||
| 	&[data-direction="up"] { | ||||
| 		> .list-enter { | ||||
| 		> .list-enter-from { | ||||
| 			opacity: 0; | ||||
| 			transform: translateY(64px); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	&[data-direction="down"] { | ||||
| 		> .list-enter { | ||||
| 		> .list-enter-from { | ||||
| 			opacity: 0; | ||||
| 			transform: translateY(-64px); | ||||
| 		} | ||||
|   | ||||
| @@ -1,69 +0,0 @@ | ||||
| <template> | ||||
| <x-column :column="column" :is-stacked="isStacked" :menu="menu"> | ||||
| 	<template #header><fa :icon="faBell" style="margin-right: 8px;"/>{{ column.name }}</template> | ||||
|  | ||||
| 	<x-notifications/> | ||||
| </x-column> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import Vue from 'vue'; | ||||
| import { faCog } from '@fortawesome/free-solid-svg-icons'; | ||||
| import { faBell } from '@fortawesome/free-regular-svg-icons'; | ||||
| import XColumn from './column.vue'; | ||||
| import XNotifications from '../notifications.vue'; | ||||
|  | ||||
| export default Vue.extend({ | ||||
| 	components: { | ||||
| 		XColumn, | ||||
| 		XNotifications | ||||
| 	}, | ||||
|  | ||||
| 	props: { | ||||
| 		column: { | ||||
| 			type: Object, | ||||
| 			required: true | ||||
| 		}, | ||||
| 		isStacked: { | ||||
| 			type: Boolean, | ||||
| 			required: true | ||||
| 		} | ||||
| 	}, | ||||
|  | ||||
| 	data() { | ||||
| 		return { | ||||
| 			menu: null, | ||||
| 			faBell | ||||
| 		} | ||||
| 	}, | ||||
|  | ||||
| 	created() { | ||||
| 		if (this.column.notificationType == null) { | ||||
| 			this.column.notificationType = 'all'; | ||||
| 			this.$store.commit('deviceUser/updateDeckColumn', this.column); | ||||
| 		} | ||||
|  | ||||
| 		this.menu = [{ | ||||
| 			icon: faCog, | ||||
| 			text: this.$t('notificationType'), | ||||
| 			action: () => { | ||||
| 				this.$root.dialog({ | ||||
| 					title: this.$t('notificationType'), | ||||
| 					type: null, | ||||
| 					select: { | ||||
| 						items: ['all', 'follow', 'mention', 'reply', 'renote', 'quote', 'reaction', 'pollVote', 'receiveFollowRequest'].map(x => ({ | ||||
| 							value: x, text: this.$t(`_notification._types.${x}`) | ||||
| 						})) | ||||
| 						default: this.column.notificationType, | ||||
| 					}, | ||||
| 					showCancelButton: true | ||||
| 				}).then(({ canceled, result: type }) => { | ||||
| 					if (canceled) return; | ||||
| 					this.column.notificationType = type; | ||||
| 					this.$store.commit('deviceUser/updateDeckColumn', this.column); | ||||
| 				}); | ||||
| 			} | ||||
| 		}]; | ||||
| 	}, | ||||
| }); | ||||
| </script> | ||||
| @@ -1,69 +1,56 @@ | ||||
| <template> | ||||
| <div class="mk-dialog" :class="{ iconOnly }"> | ||||
| 	<transition :name="$store.state.device.animation ? 'bg-fade' : ''" appear> | ||||
| 		<div class="bg _modalBg" ref="bg" @click="onBgClick" v-if="show"></div> | ||||
| 	</transition> | ||||
| 	<transition :name="$store.state.device.animation ? 'dialog' : ''" appear @after-leave="() => { destroyDom(); }"> | ||||
| 		<div class="main" ref="main" v-if="show"> | ||||
| 			<template v-if="type == 'signin'"> | ||||
| 				<mk-signin/> | ||||
| <MkModal ref="modal" @click="done(true)" @closed="$emit('closed')"> | ||||
| 	<div class="mk-dialog"> | ||||
| 		<div class="icon" v-if="icon"> | ||||
| 			<Fa :icon="icon"/> | ||||
| 		</div> | ||||
| 		<div class="icon" v-else-if="!input && !select" :class="type"> | ||||
| 			<Fa :icon="faCheck" v-if="type === 'success'"/> | ||||
| 			<Fa :icon="faTimesCircle" v-if="type === 'error'"/> | ||||
| 			<Fa :icon="faExclamationTriangle" v-if="type === 'warning'"/> | ||||
| 			<Fa :icon="faInfoCircle" v-if="type === 'info'"/> | ||||
| 			<Fa :icon="faQuestionCircle" v-if="type === 'question'"/> | ||||
| 			<Fa :icon="faSpinner" pulse v-if="type === 'waiting'"/> | ||||
| 		</div> | ||||
| 		<header v-if="title"><Mfm :text="title"/></header> | ||||
| 		<div class="body" v-if="text"><Mfm :text="text"/></div> | ||||
| 		<MkInput v-if="input" v-model:value="inputValue" autofocus :type="input.type || 'text'" :placeholder="input.placeholder" @keydown="onInputKeydown"></MkInput> | ||||
| 		<MkSelect v-if="select" v-model:value="selectedValue" autofocus> | ||||
| 			<template v-if="select.items"> | ||||
| 				<option v-for="item in select.items" :value="item.value">{{ item.text }}</option> | ||||
| 			</template> | ||||
| 			<template v-else> | ||||
| 				<div class="icon" v-if="icon"> | ||||
| 					<fa :icon="icon"/> | ||||
| 				</div> | ||||
| 				<div class="icon" v-else-if="!input && !select && !user" :class="type"> | ||||
| 					<fa :icon="faCheck" v-if="type === 'success'"/> | ||||
| 					<fa :icon="faTimesCircle" v-if="type === 'error'"/> | ||||
| 					<fa :icon="faExclamationTriangle" v-if="type === 'warning'"/> | ||||
| 					<fa :icon="faInfoCircle" v-if="type === 'info'"/> | ||||
| 					<fa :icon="faQuestionCircle" v-if="type === 'question'"/> | ||||
| 					<fa :icon="faSpinner" pulse v-if="type === 'waiting'"/> | ||||
| 				</div> | ||||
| 				<header v-if="title" v-html="title"></header> | ||||
| 				<header v-if="title == null && user">{{ $t('enterUsername') }}</header> | ||||
| 				<div class="body" v-if="text" v-html="text"></div> | ||||
| 				<mk-input v-if="input" v-model="inputValue" autofocus :type="input.type || 'text'" :placeholder="input.placeholder" @keydown="onInputKeydown"></mk-input> | ||||
| 				<mk-input v-if="user" v-model="userInputValue" autofocus @keydown="onInputKeydown"><template #prefix>@</template></mk-input> | ||||
| 				<mk-select v-if="select" v-model="selectedValue" autofocus> | ||||
| 					<template v-if="select.items"> | ||||
| 						<option v-for="item in select.items" :value="item.value">{{ item.text }}</option> | ||||
| 					</template> | ||||
| 					<template v-else> | ||||
| 						<optgroup v-for="groupedItem in select.groupedItems" :label="groupedItem.label"> | ||||
| 							<option v-for="item in groupedItem.items" :value="item.value">{{ item.text }}</option> | ||||
| 						</optgroup> | ||||
| 					</template> | ||||
| 				</mk-select> | ||||
| 				<div class="buttons" v-if="!iconOnly && (showOkButton || showCancelButton) && !actions"> | ||||
| 					<mk-button inline @click="ok" v-if="showOkButton" primary :autofocus="!input && !select && !user" :disabled="!canOk">{{ (showCancelButton || input || select || user) ? $t('ok') : $t('gotIt') }}</mk-button> | ||||
| 					<mk-button inline @click="cancel" v-if="showCancelButton || input || select || user">{{ $t('cancel') }}</mk-button> | ||||
| 				</div> | ||||
| 				<div class="buttons" v-if="actions"> | ||||
| 					<mk-button v-for="action in actions" inline @click="() => { action.callback(); close(); }" :primary="action.primary" :key="action.text">{{ action.text }}</mk-button> | ||||
| 				</div> | ||||
| 				<optgroup v-for="groupedItem in select.groupedItems" :label="groupedItem.label"> | ||||
| 					<option v-for="item in groupedItem.items" :value="item.value">{{ item.text }}</option> | ||||
| 				</optgroup> | ||||
| 			</template> | ||||
| 		</MkSelect> | ||||
| 		<div class="buttons" v-if="(showOkButton || showCancelButton) && !actions"> | ||||
| 			<MkButton inline @click="ok" v-if="showOkButton" primary :autofocus="!input && !select">{{ (showCancelButton || input || select) ? $t('ok') : $t('gotIt') }}</MkButton> | ||||
| 			<MkButton inline @click="cancel" v-if="showCancelButton || input || select">{{ $t('cancel') }}</MkButton> | ||||
| 		</div> | ||||
| 	</transition> | ||||
| </div> | ||||
| 		<div class="buttons" v-if="actions"> | ||||
| 			<MkButton v-for="action in actions" inline @click="() => { action.callback(); close(); }" :primary="action.primary" :key="action.text">{{ action.text }}</MkButton> | ||||
| 		</div> | ||||
| 	</div> | ||||
| </MkModal> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import Vue from 'vue'; | ||||
| import { defineComponent } from 'vue'; | ||||
| import { faSpinner, faInfoCircle, faExclamationTriangle, faCheck } from '@fortawesome/free-solid-svg-icons'; | ||||
| import { faTimesCircle, faQuestionCircle } from '@fortawesome/free-regular-svg-icons'; | ||||
| import MkButton from './ui/button.vue'; | ||||
| import MkInput from './ui/input.vue'; | ||||
| import MkSelect from './ui/select.vue'; | ||||
| import MkSignin from './signin.vue'; | ||||
| import parseAcct from '../../misc/acct/parse'; | ||||
| import MkModal from '@/components/ui/modal.vue'; | ||||
| import MkButton from '@/components/ui/button.vue'; | ||||
| import MkInput from '@/components/ui/input.vue'; | ||||
| import MkSelect from '@/components/ui/select.vue'; | ||||
|  | ||||
| export default Vue.extend({ | ||||
| export default defineComponent({ | ||||
| 	components: { | ||||
| 		MkModal, | ||||
| 		MkButton, | ||||
| 		MkInput, | ||||
| 		MkSelect, | ||||
| 		MkSignin, | ||||
| 	}, | ||||
|  | ||||
| 	props: { | ||||
| @@ -86,9 +73,6 @@ export default Vue.extend({ | ||||
| 		select: { | ||||
| 			required: false | ||||
| 		}, | ||||
| 		user: { | ||||
| 			required: false | ||||
| 		}, | ||||
| 		icon: { | ||||
| 			required: false | ||||
| 		}, | ||||
| @@ -107,87 +91,44 @@ export default Vue.extend({ | ||||
| 			type: Boolean, | ||||
| 			default: true | ||||
| 		}, | ||||
| 		iconOnly: { | ||||
| 			type: Boolean, | ||||
| 			default: false | ||||
| 		}, | ||||
| 		autoClose: { | ||||
| 			type: Boolean, | ||||
| 			default: false | ||||
| 		} | ||||
| 	}, | ||||
|  | ||||
| 	emits: ['done', 'closed'], | ||||
|  | ||||
| 	data() { | ||||
| 		return { | ||||
| 			show: true, | ||||
| 			inputValue: this.input && this.input.default ? this.input.default : null, | ||||
| 			userInputValue: null, | ||||
| 			selectedValue: this.select ? this.select.default ? this.select.default : this.select.items ? this.select.items[0].value : this.select.groupedItems[0].items[0].value : null, | ||||
| 			canOk: true, | ||||
| 			faTimesCircle, faQuestionCircle, faSpinner, faInfoCircle, faExclamationTriangle, faCheck | ||||
| 		}; | ||||
| 	}, | ||||
|  | ||||
| 	watch: { | ||||
| 		userInputValue() { | ||||
| 			if (this.user) { | ||||
| 				this.$root.api('users/show', parseAcct(this.userInputValue)).then(u => { | ||||
| 					this.canOk = u != null; | ||||
| 				}).catch(() => { | ||||
| 					this.canOk = false; | ||||
| 				}); | ||||
| 			} | ||||
| 		} | ||||
| 	}, | ||||
|  | ||||
| 	mounted() { | ||||
| 		if (this.user) this.canOk = false; | ||||
|  | ||||
| 		if (this.autoClose) { | ||||
| 			setTimeout(() => { | ||||
| 				this.close(); | ||||
| 			}, 1000); | ||||
| 		} | ||||
|  | ||||
| 		document.addEventListener('keydown', this.onKeydown); | ||||
| 	}, | ||||
|  | ||||
| 	beforeDestroy() { | ||||
| 	beforeUnmount() { | ||||
| 		document.removeEventListener('keydown', this.onKeydown); | ||||
| 	}, | ||||
|  | ||||
| 	methods: { | ||||
| 		done(canceled, result?) { | ||||
| 			this.$emit('done', { canceled, result }); | ||||
| 			this.$refs.modal.close(); | ||||
| 		}, | ||||
|  | ||||
| 		async ok() { | ||||
| 			if (!this.canOk) return; | ||||
| 			if (!this.showOkButton) return; | ||||
|  | ||||
| 			if (this.user) { | ||||
| 				const user = await this.$root.api('users/show', parseAcct(this.userInputValue)); | ||||
| 				if (user) { | ||||
| 					this.$emit('ok', user); | ||||
| 					this.close(); | ||||
| 				} | ||||
| 			} else { | ||||
| 				const result = | ||||
| 					this.input ? this.inputValue : | ||||
| 					this.select ? this.selectedValue : | ||||
| 					true; | ||||
| 				this.$emit('ok', result); | ||||
| 				this.close(); | ||||
| 			} | ||||
| 			const result = | ||||
| 				this.input ? this.inputValue : | ||||
| 				this.select ? this.selectedValue : | ||||
| 				true; | ||||
| 			this.done(false, result); | ||||
| 		}, | ||||
|  | ||||
| 		cancel() { | ||||
| 			this.$emit('cancel'); | ||||
| 			this.close(); | ||||
| 		}, | ||||
|  | ||||
| 		close() { | ||||
| 			if (!this.show) return; | ||||
| 			this.show = false; | ||||
| 			this.$el.style.pointerEvents = 'none'; | ||||
| 			(this.$refs.bg as any).style.pointerEvents = 'none'; | ||||
| 			(this.$refs.main as any).style.pointerEvents = 'none'; | ||||
| 			this.done(true); | ||||
| 		}, | ||||
|  | ||||
| 		onBgClick() { | ||||
| @@ -214,95 +155,60 @@ export default Vue.extend({ | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" scoped> | ||||
| .dialog-enter-active, .dialog-leave-active { | ||||
| 	transition: opacity 0.3s, transform 0.3s !important; | ||||
| } | ||||
| .dialog-enter, .dialog-leave-to { | ||||
| 	opacity: 0; | ||||
| 	transform: scale(0.9); | ||||
| } | ||||
|  | ||||
| .bg-fade-enter-active, .bg-fade-leave-active { | ||||
| 	transition: opacity 0.3s !important; | ||||
| } | ||||
| .bg-fade-enter, .bg-fade-leave-to { | ||||
| 	opacity: 0; | ||||
| } | ||||
|  | ||||
| .mk-dialog { | ||||
| 	display: flex; | ||||
| 	align-items: center; | ||||
| 	justify-content: center; | ||||
| 	position: fixed; | ||||
| 	z-index: 30000; | ||||
| 	top: 0; | ||||
| 	left: 0; | ||||
| 	width: 100%; | ||||
| 	height: 100%; | ||||
| 	position: relative; | ||||
| 	padding: 32px; | ||||
| 	min-width: 320px; | ||||
| 	max-width: 480px; | ||||
| 	box-sizing: border-box; | ||||
| 	text-align: center; | ||||
| 	background: var(--panel); | ||||
| 	border-radius: var(--radius); | ||||
|  | ||||
| 	&.iconOnly > .main { | ||||
| 		min-width: 0; | ||||
| 		width: initial; | ||||
| 	> .icon { | ||||
| 		font-size: 32px; | ||||
|  | ||||
| 		&.success { | ||||
| 			color: var(--success); | ||||
| 		} | ||||
|  | ||||
| 		&.error { | ||||
| 			color: var(--error); | ||||
| 		} | ||||
|  | ||||
| 		&.warning { | ||||
| 			color: var(--warn); | ||||
| 		} | ||||
|  | ||||
| 		> * { | ||||
| 			display: block; | ||||
| 			margin: 0 auto; | ||||
| 		} | ||||
|  | ||||
| 		& + header { | ||||
| 			margin-top: 16px; | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	> .main { | ||||
| 		display: block; | ||||
| 		position: fixed; | ||||
| 		margin: auto; | ||||
| 		padding: 32px; | ||||
| 		min-width: 320px; | ||||
| 		max-width: 480px; | ||||
| 		box-sizing: border-box; | ||||
| 		width: calc(100% - 32px); | ||||
| 		text-align: center; | ||||
| 		background: var(--panel); | ||||
| 		border-radius: var(--radius); | ||||
| 	> header { | ||||
| 		margin: 0 0 8px 0; | ||||
| 		font-weight: bold; | ||||
| 		font-size: 20px; | ||||
|  | ||||
| 		> .icon { | ||||
| 			font-size: 32px; | ||||
|  | ||||
| 			&.success { | ||||
| 				color: var(--accent); | ||||
| 			} | ||||
|  | ||||
| 			&.error { | ||||
| 				color: #ec4137; | ||||
| 			} | ||||
|  | ||||
| 			&.warning { | ||||
| 				color: #ecb637; | ||||
| 			} | ||||
|  | ||||
| 			> * { | ||||
| 				display: block; | ||||
| 				margin: 0 auto; | ||||
| 			} | ||||
|  | ||||
| 			& + header { | ||||
| 				margin-top: 16px; | ||||
| 			} | ||||
| 		& + .body { | ||||
| 			margin-top: 8px; | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 		> header { | ||||
| 			margin: 0 0 8px 0; | ||||
| 			font-weight: bold; | ||||
| 			font-size: 20px; | ||||
| 	> .body { | ||||
| 		margin: 16px 0 0 0; | ||||
| 	} | ||||
|  | ||||
| 			& + .body { | ||||
| 				margin-top: 8px; | ||||
| 			} | ||||
| 		} | ||||
| 	> .buttons { | ||||
| 		margin-top: 16px; | ||||
|  | ||||
| 		> .body { | ||||
| 			margin: 16px 0 0 0; | ||||
| 		} | ||||
|  | ||||
| 		> .buttons { | ||||
| 			margin-top: 16px; | ||||
|  | ||||
| 			> * { | ||||
| 				margin: 0 8px; | ||||
| 			} | ||||
| 		> * { | ||||
| 			margin: 0 8px; | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|   | ||||
| @@ -1,20 +1,20 @@ | ||||
| <template> | ||||
| <div class="zdjebgpv" ref="thumbnail"> | ||||
| 	<img-with-blurhash v-if="isThumbnailAvailable" :hash="file.blurhash" :src="file.thumbnailUrl" :alt="file.name" :title="file.name" :style="`object-fit: ${ fit }`"/> | ||||
| 	<fa :icon="faFileImage" class="icon" v-else-if="is === 'image'"/> | ||||
| 	<fa :icon="faFileVideo" class="icon" v-else-if="is === 'video'"/> | ||||
| 	<fa :icon="faMusic" class="icon" v-else-if="is === 'audio' || is === 'midi'"/> | ||||
| 	<fa :icon="faFileCsv" class="icon" v-else-if="is === 'csv'"/> | ||||
| 	<fa :icon="faFilePdf" class="icon" v-else-if="is === 'pdf'"/> | ||||
| 	<fa :icon="faFileAlt" class="icon" v-else-if="is === 'textfile'"/> | ||||
| 	<fa :icon="faFileArchive" class="icon" v-else-if="is === 'archive'"/> | ||||
| 	<fa :icon="faFile" class="icon" v-else/> | ||||
| 	<fa :icon="faFilm" class="icon-sub" v-if="isThumbnailAvailable && is === 'video'"/> | ||||
| 	<ImgWithBlurhash v-if="isThumbnailAvailable" :hash="file.blurhash" :src="file.thumbnailUrl" :alt="file.name" :title="file.name" :style="`object-fit: ${ fit }`"/> | ||||
| 	<Fa :icon="faFileImage" class="icon" v-else-if="is === 'image'"/> | ||||
| 	<Fa :icon="faFileVideo" class="icon" v-else-if="is === 'video'"/> | ||||
| 	<Fa :icon="faMusic" class="icon" v-else-if="is === 'audio' || is === 'midi'"/> | ||||
| 	<Fa :icon="faFileCsv" class="icon" v-else-if="is === 'csv'"/> | ||||
| 	<Fa :icon="faFilePdf" class="icon" v-else-if="is === 'pdf'"/> | ||||
| 	<Fa :icon="faFileAlt" class="icon" v-else-if="is === 'textfile'"/> | ||||
| 	<Fa :icon="faFileArchive" class="icon" v-else-if="is === 'archive'"/> | ||||
| 	<Fa :icon="faFile" class="icon" v-else/> | ||||
| 	<Fa :icon="faFilm" class="icon-sub" v-if="isThumbnailAvailable && is === 'video'"/> | ||||
| </div> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import Vue from 'vue'; | ||||
| import { defineComponent } from 'vue'; | ||||
| import { | ||||
| 	faFile, | ||||
| 	faFileAlt, | ||||
| @@ -28,7 +28,7 @@ import { | ||||
| 	} from '@fortawesome/free-solid-svg-icons'; | ||||
| import ImgWithBlurhash from './img-with-blurhash.vue'; | ||||
|  | ||||
| export default Vue.extend({ | ||||
| export default defineComponent({ | ||||
| 	components: { | ||||
| 		ImgWithBlurhash | ||||
| 	}, | ||||
|   | ||||
							
								
								
									
										70
									
								
								src/client/components/drive-select-dialog.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										70
									
								
								src/client/components/drive-select-dialog.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,70 @@ | ||||
| <template> | ||||
| <XModalWindow ref="dialog" | ||||
| 	:width="800" | ||||
| 	:height="500" | ||||
| 	:with-ok-button="true" | ||||
| 	:ok-button-disabled="(type === 'file') && (selected.length === 0)" | ||||
| 	@click="cancel()" | ||||
| 	@close="cancel()" | ||||
| 	@ok="ok()" | ||||
| 	@closed="$emit('closed')" | ||||
| > | ||||
| 	<template #header> | ||||
| 		{{ multiple ? ((type === 'file') ? $t('selectFiles') : $t('selectFolders')) : ((type === 'file') ? $t('selectFile') : $t('selectFolder')) }} | ||||
| 		<span v-if="selected.length > 0" style="margin-left: 8px; opacity: 0.5;">({{ number(selected.length) }})</span> | ||||
| 	</template> | ||||
| 	<XDrive :multiple="multiple" @changeSelection="onChangeSelection" @selected="ok()" :select="type"/> | ||||
| </XModalWindow> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import { defineComponent } from 'vue'; | ||||
| import XDrive from './drive.vue'; | ||||
| import XModalWindow from '@/components/ui/modal-window.vue'; | ||||
| import number from '@/filters/number'; | ||||
|  | ||||
| export default defineComponent({ | ||||
| 	components: { | ||||
| 		XDrive, | ||||
| 		XModalWindow, | ||||
| 	}, | ||||
|  | ||||
| 	props: { | ||||
| 		type: { | ||||
| 			type: String, | ||||
| 			required: false, | ||||
| 			default: 'file' | ||||
| 		}, | ||||
| 		multiple: { | ||||
| 			type: Boolean, | ||||
| 			default: false | ||||
| 		} | ||||
| 	}, | ||||
|  | ||||
| 	emits: ['done', 'closed'], | ||||
|  | ||||
| 	data() { | ||||
| 		return { | ||||
| 			selected: [] | ||||
| 		}; | ||||
| 	}, | ||||
|  | ||||
| 	methods: { | ||||
| 		ok() { | ||||
| 			this.$emit('done', this.selected); | ||||
| 			this.$refs.dialog.close(); | ||||
| 		}, | ||||
|  | ||||
| 		cancel() { | ||||
| 			this.$emit('done'); | ||||
| 			this.$refs.dialog.close(); | ||||
| 		}, | ||||
|  | ||||
| 		onChangeSelection(xs) { | ||||
| 			this.selected = xs; | ||||
| 		}, | ||||
|  | ||||
| 		number | ||||
| 	} | ||||
| }); | ||||
| </script> | ||||
| @@ -1,53 +1,44 @@ | ||||
| <template> | ||||
| <x-window ref="window" :width="800" :height="500" @closed="() => { $emit('closed'); destroyDom(); }" :with-ok-button="true" :ok-button-disabled="(type === 'file') && (selected.length === 0)" @ok="ok()"> | ||||
| <XWindow ref="window" | ||||
| 	:initial-width="800" | ||||
| 	:initial-height="500" | ||||
| 	:can-resize="true" | ||||
| 	@closed="$emit('closed')" | ||||
| > | ||||
| 	<template #header> | ||||
| 		{{ multiple ? ((type === 'file') ? $t('selectFiles') : $t('selectFolders')) : ((type === 'file') ? $t('selectFile') : $t('selectFolder')) }} | ||||
| 		<span v-if="selected.length > 0" style="margin-left: 8px; opacity: 0.5;">({{ selected.length | number }})</span> | ||||
| 		{{ $t('drive') }} | ||||
| 	</template> | ||||
| 	<div> | ||||
| 		<x-drive :multiple="multiple" @change-selection="onChangeSelection" :select="type"/> | ||||
| 	</div> | ||||
| </x-window> | ||||
| 	<XDrive :initial-folder="initialFolder"/> | ||||
| </XWindow> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import Vue from 'vue'; | ||||
| import { defineComponent } from 'vue'; | ||||
| import XDrive from './drive.vue'; | ||||
| import XWindow from './window.vue'; | ||||
| import XWindow from '@/components/ui/window.vue'; | ||||
|  | ||||
| export default Vue.extend({ | ||||
| export default defineComponent({ | ||||
| 	components: { | ||||
| 		XDrive, | ||||
| 		XWindow, | ||||
| 	}, | ||||
|  | ||||
| 	props: { | ||||
| 		type: { | ||||
| 			type: String, | ||||
| 			required: false, | ||||
| 			default: 'file'  | ||||
| 		initialFolder: { | ||||
| 			type: Object, | ||||
| 			required: false | ||||
| 		}, | ||||
| 		multiple: { | ||||
| 			type: Boolean, | ||||
| 			default: false | ||||
| 		} | ||||
| 	}, | ||||
|  | ||||
| 	emits: ['closed'], | ||||
|  | ||||
| 	data() { | ||||
| 		return { | ||||
| 			selected: [] | ||||
| 		}; | ||||
| 	}, | ||||
|  | ||||
| 	methods: { | ||||
| 		ok() { | ||||
| 			this.$emit('selected', this.selected); | ||||
| 			this.$refs.window.close(); | ||||
| 		}, | ||||
|  | ||||
| 		onChangeSelection(xs) { | ||||
| 			this.selected = xs; | ||||
| 		} | ||||
| 	} | ||||
| }); | ||||
| </script> | ||||
|   | ||||
| @@ -1,7 +1,8 @@ | ||||
| <template> | ||||
| <div class="ncvczrfv" | ||||
| 	:data-is-selected="isSelected" | ||||
| 	:class="{ isSelected }" | ||||
| 	@click="onClick" | ||||
| 	@contextmenu.stop="onContextmenu" | ||||
| 	draggable="true" | ||||
| 	@dragstart="onDragstart" | ||||
| 	@dragend="onDragend" | ||||
| @@ -20,7 +21,7 @@ | ||||
| 		<p>{{ $t('nsfw') }}</p> | ||||
| 	</div> | ||||
|  | ||||
| 	<x-file-thumbnail class="thumbnail" :file="file" fit="contain"/> | ||||
| 	<MkDriveFileThumbnail class="thumbnail" :file="file" fit="contain"/> | ||||
|  | ||||
| 	<p class="name"> | ||||
| 		<span>{{ file.name.lastIndexOf('.') != -1 ? file.name.substr(0, file.name.lastIndexOf('.')) : file.name }}</span> | ||||
| @@ -30,17 +31,17 @@ | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import Vue from 'vue'; | ||||
| import { defineComponent } from 'vue'; | ||||
| import { faEye, faEyeSlash } from '@fortawesome/free-regular-svg-icons'; | ||||
| import copyToClipboard from '../scripts/copy-to-clipboard'; | ||||
| //import updateAvatar from '../api/update-avatar'; | ||||
| //import updateBanner from '../api/update-banner'; | ||||
| import XFileThumbnail from './drive-file-thumbnail.vue'; | ||||
| import { faDownload, faLink, faICursor, faTrashAlt } from '@fortawesome/free-solid-svg-icons'; | ||||
| import copyToClipboard from '@/scripts/copy-to-clipboard'; | ||||
| import MkDriveFileThumbnail from './drive-file-thumbnail.vue'; | ||||
| import bytes from '../filters/bytes'; | ||||
| import * as os from '@/os'; | ||||
|  | ||||
| export default Vue.extend({ | ||||
| export default defineComponent({ | ||||
| 	components: { | ||||
| 		XFileThumbnail | ||||
| 		MkDriveFileThumbnail | ||||
| 	}, | ||||
|  | ||||
| 	props: { | ||||
| @@ -60,6 +61,8 @@ export default Vue.extend({ | ||||
| 		} | ||||
| 	}, | ||||
|  | ||||
| 	emits: ['chosen'], | ||||
|  | ||||
| 	data() { | ||||
| 		return { | ||||
| 			isDragging: false | ||||
| @@ -72,48 +75,54 @@ export default Vue.extend({ | ||||
| 			return this.$parent; | ||||
| 		}, | ||||
| 		title(): string { | ||||
| 			return `${this.file.name}\n${this.file.type} ${Vue.filter('bytes')(this.file.size)}`; | ||||
| 			return `${this.file.name}\n${this.file.type} ${bytes(this.file.size)}`; | ||||
| 		} | ||||
| 	}, | ||||
|  | ||||
| 	methods: { | ||||
| 		getMenu() { | ||||
| 			return [{ | ||||
| 				text: this.$t('rename'), | ||||
| 				icon: faICursor, | ||||
| 				action: this.rename | ||||
| 			}, { | ||||
| 				text: this.file.isSensitive ? this.$t('unmarkAsSensitive') : this.$t('markAsSensitive'), | ||||
| 				icon: this.file.isSensitive ? faEye : faEyeSlash, | ||||
| 				action: this.toggleSensitive | ||||
| 			}, null, { | ||||
| 				text: this.$t('copyUrl'), | ||||
| 				icon: faLink, | ||||
| 				action: this.copyUrl | ||||
| 			}, { | ||||
| 				type: 'a', | ||||
| 				href: this.file.url, | ||||
| 				target: '_blank', | ||||
| 				text: this.$t('download'), | ||||
| 				icon: faDownload, | ||||
| 				download: this.file.name | ||||
| 			}, null, { | ||||
| 				text: this.$t('delete'), | ||||
| 				icon: faTrashAlt, | ||||
| 				danger: true, | ||||
| 				action: this.deleteFile | ||||
| 			}]; | ||||
| 		}, | ||||
|  | ||||
| 		onClick(ev) { | ||||
| 			if (this.selectMode) { | ||||
| 				this.$emit('chosen', this.file); | ||||
| 			} else { | ||||
| 				this.$root.menu({ | ||||
| 					items: [{ | ||||
| 						text: this.$t('rename'), | ||||
| 						icon: faICursor, | ||||
| 						action: this.rename | ||||
| 					}, { | ||||
| 						text: this.file.isSensitive ? this.$t('unmarkAsSensitive') : this.$t('markAsSensitive'), | ||||
| 						icon: this.file.isSensitive ? faEye : faEyeSlash, | ||||
| 						action: this.toggleSensitive | ||||
| 					}, null, { | ||||
| 						text: this.$t('copyUrl'), | ||||
| 						icon: faLink, | ||||
| 						action: this.copyUrl | ||||
| 					}, { | ||||
| 						type: 'a', | ||||
| 						href: this.file.url, | ||||
| 						target: '_blank', | ||||
| 						text: this.$t('download'), | ||||
| 						icon: faDownload, | ||||
| 						download: this.file.name | ||||
| 					}, null, { | ||||
| 						text: this.$t('delete'), | ||||
| 						icon: faTrashAlt, | ||||
| 						action: this.deleteFile | ||||
| 					}], | ||||
| 					source: ev.currentTarget || ev.target, | ||||
| 				}); | ||||
| 				os.modalMenu(this.getMenu(), ev.currentTarget || ev.target); | ||||
| 			} | ||||
| 		}, | ||||
|  | ||||
| 		onContextmenu(e) { | ||||
| 			os.contextMenu(this.getMenu(), e); | ||||
| 		}, | ||||
|  | ||||
| 		onDragstart(e) { | ||||
| 			e.dataTransfer.effectAllowed = 'move'; | ||||
| 			e.dataTransfer.setData('mk_drive_file', JSON.stringify(this.file)); | ||||
| 			e.dataTransfer.setData(_DATA_TRANSFER_DRIVE_FILE_, JSON.stringify(this.file)); | ||||
| 			this.isDragging = true; | ||||
|  | ||||
| 			// 親ブラウザに対して、ドラッグが開始されたフラグを立てる | ||||
| @@ -127,7 +136,7 @@ export default Vue.extend({ | ||||
| 		}, | ||||
|  | ||||
| 		rename() { | ||||
| 			this.$root.dialog({ | ||||
| 			os.dialog({ | ||||
| 				title: this.$t('renameFile'), | ||||
| 				input: { | ||||
| 					placeholder: this.$t('inputNewFileName'), | ||||
| @@ -136,7 +145,7 @@ export default Vue.extend({ | ||||
| 				} | ||||
| 			}).then(({ canceled, result: name }) => { | ||||
| 				if (canceled) return; | ||||
| 				this.$root.api('drive/files/update', { | ||||
| 				os.api('drive/files/update', { | ||||
| 					fileId: this.file.id, | ||||
| 					name: name | ||||
| 				}); | ||||
| @@ -144,7 +153,7 @@ export default Vue.extend({ | ||||
| 		}, | ||||
|  | ||||
| 		toggleSensitive() { | ||||
| 			this.$root.api('drive/files/update', { | ||||
| 			os.api('drive/files/update', { | ||||
| 				fileId: this.file.id, | ||||
| 				isSensitive: !this.file.isSensitive | ||||
| 			}); | ||||
| @@ -152,18 +161,15 @@ export default Vue.extend({ | ||||
|  | ||||
| 		copyUrl() { | ||||
| 			copyToClipboard(this.file.url); | ||||
| 			this.$root.dialog({ | ||||
| 				type: 'success', | ||||
| 				iconOnly: true, autoClose: true | ||||
| 			}); | ||||
| 			os.success(); | ||||
| 		}, | ||||
|  | ||||
| 		setAsAvatar() { | ||||
| 			updateAvatar(this.$root)(this.file); | ||||
| 			os.updateAvatar(this.file); | ||||
| 		}, | ||||
|  | ||||
| 		setAsBanner() { | ||||
| 			updateBanner(this.$root)(this.file); | ||||
| 			os.updateBanner(this.file); | ||||
| 		}, | ||||
|  | ||||
| 		addApp() { | ||||
| @@ -171,17 +177,19 @@ export default Vue.extend({ | ||||
| 		}, | ||||
|  | ||||
| 		async deleteFile() { | ||||
| 			const { canceled } = await this.$root.dialog({ | ||||
| 			const { canceled } = await os.dialog({ | ||||
| 				type: 'warning', | ||||
| 				text: this.$t('driveFileDeleteConfirm', { name: this.file.name }), | ||||
| 				showCancelButton: true | ||||
| 			}); | ||||
| 			if (canceled) return; | ||||
|  | ||||
| 			this.$root.api('drive/files/delete', { | ||||
| 			os.api('drive/files/delete', { | ||||
| 				fileId: this.file.id | ||||
| 			}); | ||||
| 		} | ||||
| 		}, | ||||
|  | ||||
| 		bytes | ||||
| 	} | ||||
| }); | ||||
| </script> | ||||
| @@ -197,6 +205,10 @@ export default Vue.extend({ | ||||
| 		cursor: pointer; | ||||
| 	} | ||||
|  | ||||
| 	> * { | ||||
| 		pointer-events: none; | ||||
| 	} | ||||
|  | ||||
| 	&:hover { | ||||
| 		background: rgba(#000, 0.05); | ||||
|  | ||||
| @@ -233,7 +245,7 @@ export default Vue.extend({ | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	&[data-is-selected] { | ||||
| 	&.isSelected { | ||||
| 		background: var(--accent); | ||||
|  | ||||
| 		&:hover { | ||||
|   | ||||
| @@ -1,7 +1,8 @@ | ||||
| <template> | ||||
| <div class="rghtznwe" | ||||
| 	:data-draghover="draghover" | ||||
| 	:class="{ draghover }" | ||||
| 	@click="onClick" | ||||
| 	@contextmenu.stop="onContextmenu" | ||||
| 	@mouseover="onMouseover" | ||||
| 	@mouseout="onMouseout" | ||||
| 	@dragover.prevent.stop="onDragover" | ||||
| @@ -14,8 +15,8 @@ | ||||
| 	:title="title" | ||||
| > | ||||
| 	<p class="name"> | ||||
| 		<template v-if="hover"><fa :icon="faFolderOpen" fixed-width/></template> | ||||
| 		<template v-if="!hover"><fa :icon="faFolder" fixed-width/></template> | ||||
| 		<template v-if="hover"><Fa :icon="faFolderOpen" fixed-width/></template> | ||||
| 		<template v-if="!hover"><Fa :icon="faFolder" fixed-width/></template> | ||||
| 		{{ folder.name }} | ||||
| 	</p> | ||||
| 	<p class="upload" v-if="$store.state.settings.uploadFolder == folder.id"> | ||||
| @@ -26,10 +27,12 @@ | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import Vue from 'vue'; | ||||
| import { faFolder, faFolderOpen } from '@fortawesome/free-regular-svg-icons'; | ||||
| import { defineComponent } from 'vue'; | ||||
| import { faFolder, faFolderOpen, faTrashAlt, faWindowRestore } from '@fortawesome/free-regular-svg-icons'; | ||||
| import * as os from '@/os'; | ||||
| import { faICursor } from '@fortawesome/free-solid-svg-icons'; | ||||
|  | ||||
| export default Vue.extend({ | ||||
| export default defineComponent({ | ||||
| 	props: { | ||||
| 		folder: { | ||||
| 			type: Object, | ||||
| @@ -47,6 +50,8 @@ export default Vue.extend({ | ||||
| 		} | ||||
| 	}, | ||||
|  | ||||
| 	emits: ['chosen'], | ||||
|  | ||||
| 	data() { | ||||
| 		return { | ||||
| 			hover: false, | ||||
| @@ -91,8 +96,8 @@ export default Vue.extend({ | ||||
| 			} | ||||
|  | ||||
| 			const isFile = e.dataTransfer.items[0].kind == 'file'; | ||||
| 			const isDriveFile = e.dataTransfer.types[0] == 'mk_drive_file'; | ||||
| 			const isDriveFolder = e.dataTransfer.types[0] == 'mk_drive_folder'; | ||||
| 			const isDriveFile = e.dataTransfer.types[0] == _DATA_TRANSFER_DRIVE_FILE_; | ||||
| 			const isDriveFolder = e.dataTransfer.types[0] == _DATA_TRANSFER_DRIVE_FOLDER_; | ||||
|  | ||||
| 			if (isFile || isDriveFile || isDriveFolder) { | ||||
| 				e.dataTransfer.dropEffect = e.dataTransfer.effectAllowed == 'all' ? 'copy' : 'move'; | ||||
| @@ -121,11 +126,11 @@ export default Vue.extend({ | ||||
| 			} | ||||
|  | ||||
| 			//#region ドライブのファイル | ||||
| 			const driveFile = e.dataTransfer.getData('mk_drive_file'); | ||||
| 			const driveFile = e.dataTransfer.getData(_DATA_TRANSFER_DRIVE_FILE_); | ||||
| 			if (driveFile != null && driveFile != '') { | ||||
| 				const file = JSON.parse(driveFile); | ||||
| 				this.browser.removeFile(file.id); | ||||
| 				this.$root.api('drive/files/update', { | ||||
| 				os.api('drive/files/update', { | ||||
| 					fileId: file.id, | ||||
| 					folderId: this.folder.id | ||||
| 				}); | ||||
| @@ -133,7 +138,7 @@ export default Vue.extend({ | ||||
| 			//#endregion | ||||
|  | ||||
| 			//#region ドライブのフォルダ | ||||
| 			const driveFolder = e.dataTransfer.getData('mk_drive_folder'); | ||||
| 			const driveFolder = e.dataTransfer.getData(_DATA_TRANSFER_DRIVE_FOLDER_); | ||||
| 			if (driveFolder != null && driveFolder != '') { | ||||
| 				const folder = JSON.parse(driveFolder); | ||||
|  | ||||
| @@ -141,7 +146,7 @@ export default Vue.extend({ | ||||
| 				if (folder.id == this.folder.id) return; | ||||
|  | ||||
| 				this.browser.removeFolder(folder.id); | ||||
| 				this.$root.api('drive/folders/update', { | ||||
| 				os.api('drive/folders/update', { | ||||
| 					folderId: folder.id, | ||||
| 					parentId: this.folder.id | ||||
| 				}).then(() => { | ||||
| @@ -149,15 +154,15 @@ export default Vue.extend({ | ||||
| 				}).catch(err => { | ||||
| 					switch (err) { | ||||
| 						case 'detected-circular-definition': | ||||
| 							this.$root.dialog({ | ||||
| 							os.dialog({ | ||||
| 								title: this.$t('unableToProcess'), | ||||
| 								text: this.$t('circularReferenceFolder') | ||||
| 							}); | ||||
| 							break; | ||||
| 						default: | ||||
| 							this.$root.dialog({ | ||||
| 							os.dialog({ | ||||
| 								type: 'error', | ||||
| 								text: this.$t('error') | ||||
| 								text: this.$t('somethingHappened') | ||||
| 							}); | ||||
| 					} | ||||
| 				}); | ||||
| @@ -167,7 +172,7 @@ export default Vue.extend({ | ||||
|  | ||||
| 		onDragstart(e) { | ||||
| 			e.dataTransfer.effectAllowed = 'move'; | ||||
| 			e.dataTransfer.setData('mk_drive_folder', JSON.stringify(this.folder)); | ||||
| 			e.dataTransfer.setData(_DATA_TRANSFER_DRIVE_FOLDER_, JSON.stringify(this.folder)); | ||||
| 			this.isDragging = true; | ||||
|  | ||||
| 			// 親ブラウザに対して、ドラッグが開始されたフラグを立てる | ||||
| @@ -189,7 +194,7 @@ export default Vue.extend({ | ||||
| 		}, | ||||
|  | ||||
| 		rename() { | ||||
| 			this.$root.dialog({ | ||||
| 			os.dialog({ | ||||
| 				title: this.$t('renameFolder'), | ||||
| 				input: { | ||||
| 					placeholder: this.$t('inputNewFolderName'), | ||||
| @@ -197,7 +202,7 @@ export default Vue.extend({ | ||||
| 				} | ||||
| 			}).then(({ canceled, result: name }) => { | ||||
| 				if (canceled) return; | ||||
| 				this.$root.api('drive/folders/update', { | ||||
| 				os.api('drive/folders/update', { | ||||
| 					folderId: this.folder.id, | ||||
| 					name: name | ||||
| 				}); | ||||
| @@ -205,7 +210,7 @@ export default Vue.extend({ | ||||
| 		}, | ||||
|  | ||||
| 		deleteFolder() { | ||||
| 			this.$root.api('drive/folders/delete', { | ||||
| 			os.api('drive/folders/delete', { | ||||
| 				folderId: this.folder.id | ||||
| 			}).then(() => { | ||||
| 				if (this.$store.state.settings.uploadFolder === this.folder.id) { | ||||
| @@ -217,14 +222,14 @@ export default Vue.extend({ | ||||
| 			}).catch(err => { | ||||
| 				switch(err.id) { | ||||
| 					case 'b0fc8a17-963c-405d-bfbc-859a487295e1': | ||||
| 						this.$root.dialog({ | ||||
| 						os.dialog({ | ||||
| 							type: 'error', | ||||
| 							title: this.$t('unableToDelete'), | ||||
| 							text: this.$t('hasChildFilesOrFolders') | ||||
| 						}); | ||||
| 						break; | ||||
| 					default: | ||||
| 						this.$root.dialog({ | ||||
| 						os.dialog({ | ||||
| 							type: 'error', | ||||
| 							text: this.$t('unableToDelete') | ||||
| 						}); | ||||
| @@ -238,6 +243,28 @@ export default Vue.extend({ | ||||
| 				value: this.folder.id | ||||
| 			}); | ||||
| 		}, | ||||
|  | ||||
| 		onContextmenu(e) { | ||||
| 			os.contextMenu([{ | ||||
| 				text: this.$t('openInWindow'), | ||||
| 				icon: faWindowRestore, | ||||
| 				action: () => { | ||||
| 					os.popup(import('./drive-window.vue'), { | ||||
| 						initialFolder: this.folder | ||||
| 					}, { | ||||
| 					}, 'closed'); | ||||
| 				} | ||||
| 			}, null, { | ||||
| 				text: this.$t('rename'), | ||||
| 				icon: faICursor, | ||||
| 				action: this.rename | ||||
| 			}, null, { | ||||
| 				text: this.$t('delete'), | ||||
| 				icon: faTrashAlt, | ||||
| 				danger: true, | ||||
| 				action: this.deleteFolder | ||||
| 			}], e); | ||||
| 		}, | ||||
| 	} | ||||
| }); | ||||
| </script> | ||||
| @@ -272,7 +299,7 @@ export default Vue.extend({ | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	&[data-draghover] { | ||||
| 	&.draghover { | ||||
| 		&:after { | ||||
| 			content: ""; | ||||
| 			pointer-events: none; | ||||
|   | ||||
| @@ -1,22 +1,23 @@ | ||||
| <template> | ||||
| <div class="drylbebk" | ||||
| 	:data-draghover="draghover" | ||||
| 	:class="{ draghover }" | ||||
| 	@click="onClick" | ||||
| 	@dragover.prevent.stop="onDragover" | ||||
| 	@dragenter="onDragenter" | ||||
| 	@dragleave="onDragleave" | ||||
| 	@drop.stop="onDrop" | ||||
| > | ||||
| 	<i v-if="folder == null"><fa :icon="faCloud"/></i> | ||||
| 	<i v-if="folder == null"><Fa :icon="faCloud"/></i> | ||||
| 	<span>{{ folder == null ? $t('drive') : folder.name }}</span> | ||||
| </div> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import Vue from 'vue'; | ||||
| import { defineComponent } from 'vue'; | ||||
| import { faCloud } from '@fortawesome/free-solid-svg-icons'; | ||||
| import * as os from '@/os'; | ||||
|  | ||||
| export default Vue.extend({ | ||||
| export default defineComponent({ | ||||
| 	props: { | ||||
| 		folder: { | ||||
| 			type: Object, | ||||
| @@ -58,8 +59,8 @@ export default Vue.extend({ | ||||
| 			} | ||||
|  | ||||
| 			const isFile = e.dataTransfer.items[0].kind == 'file'; | ||||
| 			const isDriveFile = e.dataTransfer.types[0] == 'mk_drive_file'; | ||||
| 			const isDriveFolder = e.dataTransfer.types[0] == 'mk_drive_folder'; | ||||
| 			const isDriveFile = e.dataTransfer.types[0] == _DATA_TRANSFER_DRIVE_FILE_; | ||||
| 			const isDriveFolder = e.dataTransfer.types[0] == _DATA_TRANSFER_DRIVE_FOLDER_; | ||||
|  | ||||
| 			if (isFile || isDriveFile || isDriveFolder) { | ||||
| 				e.dataTransfer.dropEffect = e.dataTransfer.effectAllowed == 'all' ? 'copy' : 'move'; | ||||
| @@ -90,11 +91,11 @@ export default Vue.extend({ | ||||
| 			} | ||||
|  | ||||
| 			//#region ドライブのファイル | ||||
| 			const driveFile = e.dataTransfer.getData('mk_drive_file'); | ||||
| 			const driveFile = e.dataTransfer.getData(_DATA_TRANSFER_DRIVE_FILE_); | ||||
| 			if (driveFile != null && driveFile != '') { | ||||
| 				const file = JSON.parse(driveFile); | ||||
| 				this.browser.removeFile(file.id); | ||||
| 				this.$root.api('drive/files/update', { | ||||
| 				os.api('drive/files/update', { | ||||
| 					fileId: file.id, | ||||
| 					folderId: this.folder ? this.folder.id : null | ||||
| 				}); | ||||
| @@ -102,13 +103,13 @@ export default Vue.extend({ | ||||
| 			//#endregion | ||||
|  | ||||
| 			//#region ドライブのフォルダ | ||||
| 			const driveFolder = e.dataTransfer.getData('mk_drive_folder'); | ||||
| 			const driveFolder = e.dataTransfer.getData(_DATA_TRANSFER_DRIVE_FOLDER_); | ||||
| 			if (driveFolder != null && driveFolder != '') { | ||||
| 				const folder = JSON.parse(driveFolder); | ||||
| 				// 移動先が自分自身ならreject | ||||
| 				if (this.folder && folder.id == this.folder.id) return; | ||||
| 				this.browser.removeFolder(folder.id); | ||||
| 				this.$root.api('drive/folders/update', { | ||||
| 				os.api('drive/folders/update', { | ||||
| 					folderId: folder.id, | ||||
| 					parentId: this.folder ? this.folder.id : null | ||||
| 				}); | ||||
| @@ -125,7 +126,7 @@ export default Vue.extend({ | ||||
| 		pointer-events: none; | ||||
| 	} | ||||
|  | ||||
| 	&[data-draghover] { | ||||
| 	&.draghover { | ||||
| 		background: #eee; | ||||
| 	} | ||||
|  | ||||
|   | ||||
| @@ -2,34 +2,35 @@ | ||||
| <div class="yfudmmck"> | ||||
| 	<nav> | ||||
| 		<div class="path" @contextmenu.prevent.stop="() => {}"> | ||||
| 			<x-nav-folder :class="{ current: folder == null }"/> | ||||
| 			<XNavFolder :class="{ current: folder == null }"/> | ||||
| 			<template v-for="f in hierarchyFolders"> | ||||
| 				<span class="separator" :key="f.id + ':separator'"><fa :icon="faAngleRight"/></span> | ||||
| 				<x-nav-folder :folder="f" :key="f.id"/> | ||||
| 				<span class="separator"><Fa :icon="faAngleRight"/></span> | ||||
| 				<XNavFolder :folder="f"/> | ||||
| 			</template> | ||||
| 			<span class="separator" v-if="folder != null"><fa :icon="faAngleRight"/></span> | ||||
| 			<span class="separator" v-if="folder != null"><Fa :icon="faAngleRight"/></span> | ||||
| 			<span class="folder current" v-if="folder != null">{{ folder.name }}</span> | ||||
| 		</div> | ||||
| 	</nav> | ||||
| 	<div class="main" :class="{ uploading: uploadings.length > 0, fetching }" | ||||
| 	<div class="main _section" :class="{ uploading: uploadings.length > 0, fetching }" | ||||
| 		ref="main" | ||||
| 		@dragover.prevent.stop="onDragover" | ||||
| 		@dragenter="onDragenter" | ||||
| 		@dragleave="onDragleave" | ||||
| 		@drop.prevent.stop="onDrop" | ||||
| 		@contextmenu="onContextmenu" | ||||
| 	> | ||||
| 		<div class="contents" ref="contents"> | ||||
| 			<div class="folders" ref="foldersContainer" v-show="folders.length > 0"> | ||||
| 				<x-folder v-for="f in folders" :key="f.id" class="folder" :folder="f" :select-mode="select === 'folder'" :is-selected="selectedFolders.some(x => x.id === f.id)" @chosen="chooseFolder"/> | ||||
| 				<XFolder v-for="f in folders" :key="f.id" class="folder" :folder="f" :select-mode="select === 'folder'" :is-selected="selectedFolders.some(x => x.id === f.id)" @chosen="chooseFolder"/> | ||||
| 				<!-- SEE: https://stackoverflow.com/questions/18744164/flex-box-align-last-row-to-grid --> | ||||
| 				<div class="padding" v-for="(n, i) in 16" :key="i"></div> | ||||
| 				<mk-button ref="moreFolders" v-if="moreFolders">{{ $t('loadMore') }}</mk-button> | ||||
| 				<MkButton ref="moreFolders" v-if="moreFolders">{{ $t('loadMore') }}</MkButton> | ||||
| 			</div> | ||||
| 			<div class="files" ref="filesContainer" v-show="files.length > 0"> | ||||
| 				<x-file v-for="file in files" :key="file.id" class="file" :file="file" :select-mode="select === 'file'" :is-selected="selectedFiles.some(x => x.id === file.id)" @chosen="chooseFile"/> | ||||
| 				<XFile v-for="file in files" :key="file.id" class="file" :file="file" :select-mode="select === 'file'" :is-selected="selectedFiles.some(x => x.id === file.id)" @chosen="chooseFile"/> | ||||
| 				<!-- SEE: https://stackoverflow.com/questions/18744164/flex-box-align-last-row-to-grid --> | ||||
| 				<div class="padding" v-for="(n, i) in 16" :key="i"></div> | ||||
| 				<mk-button ref="loadMoreFiles" @click="fetchMoreFiles" v-show="moreFiles">{{ $t('loadMore') }}</mk-button> | ||||
| 				<MkButton ref="loadMoreFiles" @click="fetchMoreFiles" v-show="moreFiles">{{ $t('loadMore') }}</MkButton> | ||||
| 			</div> | ||||
| 			<div class="empty" v-if="files.length == 0 && folders.length == 0 && !fetching"> | ||||
| 				<p v-if="draghover">{{ $t('empty-draghover') }}</p> | ||||
| @@ -37,34 +38,33 @@ | ||||
| 				<p v-if="!draghover && folder != null">{{ $t('emptyFolder') }}</p> | ||||
| 			</div> | ||||
| 		</div> | ||||
| 		<mk-loading v-if="fetching"/> | ||||
| 		<MkLoading v-if="fetching"/> | ||||
| 	</div> | ||||
| 	<div class="dropzone" v-if="draghover"></div> | ||||
| 	<x-uploader ref="uploader" @change="onChangeUploaderUploads" @uploaded="onUploaderUploaded"/> | ||||
| 	<input ref="fileInput" type="file" accept="*/*" multiple="multiple" tabindex="-1" @change="onChangeFileInput"/> | ||||
| </div> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import Vue from 'vue'; | ||||
| import { faAngleRight } from '@fortawesome/free-solid-svg-icons'; | ||||
| import { defineComponent } from 'vue'; | ||||
| import { faAngleRight, faFolderPlus, faICursor, faLink, faUpload } from '@fortawesome/free-solid-svg-icons'; | ||||
| import XNavFolder from './drive.nav-folder.vue'; | ||||
| import XFolder from './drive.folder.vue'; | ||||
| import XFile from './drive.file.vue'; | ||||
| import XUploader from './uploader.vue'; | ||||
| import MkButton from './ui/button.vue'; | ||||
| import * as os from '@/os'; | ||||
| import { faTrashAlt } from '@fortawesome/free-regular-svg-icons'; | ||||
|  | ||||
| export default Vue.extend({ | ||||
| export default defineComponent({ | ||||
| 	components: { | ||||
| 		XNavFolder, | ||||
| 		XFolder, | ||||
| 		XFile, | ||||
| 		XUploader, | ||||
| 		MkButton, | ||||
| 	}, | ||||
|  | ||||
| 	props: { | ||||
| 		initFolder: { | ||||
| 		initialFolder: { | ||||
| 			type: Object, | ||||
| 			required: false | ||||
| 		}, | ||||
| @@ -85,6 +85,8 @@ export default Vue.extend({ | ||||
| 		} | ||||
| 	}, | ||||
|  | ||||
| 	emits: ['selected', 'change-selection', 'move-root', 'cd', 'open-folder'], | ||||
|  | ||||
| 	data() { | ||||
| 		return { | ||||
| 			/** | ||||
| @@ -100,7 +102,7 @@ export default Vue.extend({ | ||||
| 			hierarchyFolders: [], | ||||
| 			selectedFiles: [], | ||||
| 			selectedFolders: [], | ||||
| 			uploadings: [], | ||||
| 			uploadings: os.uploads, | ||||
| 			connection: null, | ||||
|  | ||||
| 			/** | ||||
| @@ -140,7 +142,7 @@ export default Vue.extend({ | ||||
| 			}); | ||||
| 		} | ||||
|  | ||||
| 		this.connection = this.$root.stream.useSharedConnection('drive'); | ||||
| 		this.connection = os.stream.useSharedConnection('drive'); | ||||
|  | ||||
| 		this.connection.on('fileCreated', this.onStreamDriveFileCreated); | ||||
| 		this.connection.on('fileUpdated', this.onStreamDriveFileUpdated); | ||||
| @@ -149,8 +151,8 @@ export default Vue.extend({ | ||||
| 		this.connection.on('folderUpdated', this.onStreamDriveFolderUpdated); | ||||
| 		this.connection.on('folderDeleted', this.onStreamDriveFolderDeleted); | ||||
|  | ||||
| 		if (this.initFolder) { | ||||
| 			this.move(this.initFolder); | ||||
| 		if (this.initialFolder) { | ||||
| 			this.move(this.initialFolder); | ||||
| 		} else { | ||||
| 			this.fetch(); | ||||
| 		} | ||||
| @@ -164,7 +166,7 @@ export default Vue.extend({ | ||||
| 		} | ||||
| 	}, | ||||
|  | ||||
| 	beforeDestroy() { | ||||
| 	beforeUnmount() { | ||||
| 		this.connection.dispose(); | ||||
| 		this.ilFilesObserver.disconnect(); | ||||
| 	}, | ||||
| @@ -204,14 +206,6 @@ export default Vue.extend({ | ||||
| 			this.removeFolder(folderId); | ||||
| 		}, | ||||
|  | ||||
| 		onChangeUploaderUploads(uploads) { | ||||
| 			this.uploadings = uploads; | ||||
| 		}, | ||||
|  | ||||
| 		onUploaderUploaded(file) { | ||||
| 			this.addFile(file, true); | ||||
| 		}, | ||||
|  | ||||
| 		onDragover(e): any { | ||||
| 			// ドラッグ元が自分自身の所有するアイテムだったら | ||||
| 			if (this.isDragSource) { | ||||
| @@ -221,8 +215,8 @@ export default Vue.extend({ | ||||
| 			} | ||||
|  | ||||
| 			const isFile = e.dataTransfer.items[0].kind == 'file'; | ||||
| 			const isDriveFile = e.dataTransfer.types[0] == 'mk_drive_file'; | ||||
| 			const isDriveFolder = e.dataTransfer.types[0] == 'mk_drive_folder'; | ||||
| 			const isDriveFile = e.dataTransfer.types[0] == _DATA_TRANSFER_DRIVE_FILE_; | ||||
| 			const isDriveFolder = e.dataTransfer.types[0] == _DATA_TRANSFER_DRIVE_FOLDER_; | ||||
|  | ||||
| 			if (isFile || isDriveFile || isDriveFolder) { | ||||
| 				e.dataTransfer.dropEffect = e.dataTransfer.effectAllowed == 'all' ? 'copy' : 'move'; | ||||
| @@ -253,12 +247,12 @@ export default Vue.extend({ | ||||
| 			} | ||||
|  | ||||
| 			//#region ドライブのファイル | ||||
| 			const driveFile = e.dataTransfer.getData('mk_drive_file'); | ||||
| 			const driveFile = e.dataTransfer.getData(_DATA_TRANSFER_DRIVE_FILE_); | ||||
| 			if (driveFile != null && driveFile != '') { | ||||
| 				const file = JSON.parse(driveFile); | ||||
| 				if (this.files.some(f => f.id == file.id)) return; | ||||
| 				this.removeFile(file.id); | ||||
| 				this.$root.api('drive/files/update', { | ||||
| 				os.api('drive/files/update', { | ||||
| 					fileId: file.id, | ||||
| 					folderId: this.folder ? this.folder.id : null | ||||
| 				}); | ||||
| @@ -266,7 +260,7 @@ export default Vue.extend({ | ||||
| 			//#endregion | ||||
|  | ||||
| 			//#region ドライブのフォルダ | ||||
| 			const driveFolder = e.dataTransfer.getData('mk_drive_folder'); | ||||
| 			const driveFolder = e.dataTransfer.getData(_DATA_TRANSFER_DRIVE_FOLDER_); | ||||
| 			if (driveFolder != null && driveFolder != '') { | ||||
| 				const folder = JSON.parse(driveFolder); | ||||
|  | ||||
| @@ -274,7 +268,7 @@ export default Vue.extend({ | ||||
| 				if (this.folder && folder.id == this.folder.id) return false; | ||||
| 				if (this.folders.some(f => f.id == folder.id)) return false; | ||||
| 				this.removeFolder(folder.id); | ||||
| 				this.$root.api('drive/folders/update', { | ||||
| 				os.api('drive/folders/update', { | ||||
| 					folderId: folder.id, | ||||
| 					parentId: this.folder ? this.folder.id : null | ||||
| 				}).then(() => { | ||||
| @@ -282,15 +276,15 @@ export default Vue.extend({ | ||||
| 				}).catch(err => { | ||||
| 					switch (err) { | ||||
| 						case 'detected-circular-definition': | ||||
| 							this.$root.dialog({ | ||||
| 							os.dialog({ | ||||
| 								title: this.$t('unableToProcess'), | ||||
| 								text: this.$t('circularReferenceFolder') | ||||
| 							}); | ||||
| 							break; | ||||
| 						default: | ||||
| 							this.$root.dialog({ | ||||
| 							os.dialog({ | ||||
| 								type: 'error', | ||||
| 								text: this.$t('error') | ||||
| 								text: this.$t('somethingHappened') | ||||
| 							}); | ||||
| 					} | ||||
| 				}); | ||||
| @@ -303,19 +297,19 @@ export default Vue.extend({ | ||||
| 		}, | ||||
|  | ||||
| 		urlUpload() { | ||||
| 			this.$root.dialog({ | ||||
| 			os.dialog({ | ||||
| 				title: this.$t('uploadFromUrl'), | ||||
| 				input: { | ||||
| 					placeholder: this.$t('uploadFromUrlDescription') | ||||
| 				} | ||||
| 			}).then(({ canceled, result: url }) => { | ||||
| 				if (canceled) return; | ||||
| 				this.$root.api('drive/files/upload_from_url', { | ||||
| 				os.api('drive/files/upload_from_url', { | ||||
| 					url: url, | ||||
| 					folderId: this.folder ? this.folder.id : undefined | ||||
| 				}); | ||||
|  | ||||
| 				this.$root.dialog({ | ||||
| 				os.dialog({ | ||||
| 					title: this.$t('uploadFromUrlRequested'), | ||||
| 					text: this.$t('uploadFromUrlMayTakeTime') | ||||
| 				}); | ||||
| @@ -323,14 +317,14 @@ export default Vue.extend({ | ||||
| 		}, | ||||
|  | ||||
| 		createFolder() { | ||||
| 			this.$root.dialog({ | ||||
| 			os.dialog({ | ||||
| 				title: this.$t('createFolder'), | ||||
| 				input: { | ||||
| 					placeholder: this.$t('folderName') | ||||
| 				} | ||||
| 			}).then(({ canceled, result: name }) => { | ||||
| 				if (canceled) return; | ||||
| 				this.$root.api('drive/folders/create', { | ||||
| 				os.api('drive/folders/create', { | ||||
| 					name: name, | ||||
| 					parentId: this.folder ? this.folder.id : undefined | ||||
| 				}).then(folder => { | ||||
| @@ -340,7 +334,7 @@ export default Vue.extend({ | ||||
| 		}, | ||||
|  | ||||
| 		renameFolder(folder) { | ||||
| 			this.$root.dialog({ | ||||
| 			os.dialog({ | ||||
| 				title: this.$t('renameFolder'), | ||||
| 				input: { | ||||
| 					placeholder: this.$t('inputNewFolderName'), | ||||
| @@ -348,7 +342,7 @@ export default Vue.extend({ | ||||
| 				} | ||||
| 			}).then(({ canceled, result: name }) => { | ||||
| 				if (canceled) return; | ||||
| 				this.$root.api('drive/folders/update', { | ||||
| 				os.api('drive/folders/update', { | ||||
| 					folderId: folder.id, | ||||
| 					name: name | ||||
| 				}).then(folder => { | ||||
| @@ -359,7 +353,7 @@ export default Vue.extend({ | ||||
| 		}, | ||||
|  | ||||
| 		deleteFolder(folder) { | ||||
| 			this.$root.api('drive/folders/delete', { | ||||
| 			os.api('drive/folders/delete', { | ||||
| 				folderId: folder.id | ||||
| 			}).then(() => { | ||||
| 				// 削除時に親フォルダに移動 | ||||
| @@ -367,14 +361,14 @@ export default Vue.extend({ | ||||
| 			}).catch(err => { | ||||
| 				switch(err.id) { | ||||
| 					case 'b0fc8a17-963c-405d-bfbc-859a487295e1': | ||||
| 						this.$root.dialog({ | ||||
| 						os.dialog({ | ||||
| 							type: 'error', | ||||
| 							title: this.$t('unableToDelete'), | ||||
| 							text: this.$t('hasChildFilesOrFolders') | ||||
| 						}); | ||||
| 						break; | ||||
| 					default: | ||||
| 						this.$root.dialog({ | ||||
| 						os.dialog({ | ||||
| 							type: 'error', | ||||
| 							text: this.$t('unableToDelete') | ||||
| 						}); | ||||
| @@ -390,7 +384,9 @@ export default Vue.extend({ | ||||
|  | ||||
| 		upload(file, folder) { | ||||
| 			if (folder && typeof folder == 'object') folder = folder.id; | ||||
| 			(this.$refs.uploader as any).upload(file, folder); | ||||
| 			os.upload(file, folder).then(res => { | ||||
| 				this.addFile(res, true); | ||||
| 			}); | ||||
| 		}, | ||||
|  | ||||
| 		chooseFile(file) { | ||||
| @@ -441,7 +437,7 @@ export default Vue.extend({ | ||||
|  | ||||
| 			this.fetching = true; | ||||
|  | ||||
| 			this.$root.api('drive/folders/show', { | ||||
| 			os.api('drive/folders/show', { | ||||
| 				folderId: target | ||||
| 			}).then(folder => { | ||||
| 				this.folder = folder; | ||||
| @@ -465,7 +461,7 @@ export default Vue.extend({ | ||||
|  | ||||
| 			if (this.folders.some(f => f.id == folder.id)) { | ||||
| 				const exist = this.folders.map(f => f.id).indexOf(folder.id); | ||||
| 				Vue.set(this.folders, exist, folder); | ||||
| 				this.folders[exist] = folder; | ||||
| 				return; | ||||
| 			} | ||||
|  | ||||
| @@ -482,7 +478,7 @@ export default Vue.extend({ | ||||
|  | ||||
| 			if (this.files.some(f => f.id == file.id)) { | ||||
| 				const exist = this.files.map(f => f.id).indexOf(file.id); | ||||
| 				Vue.set(this.files, exist, file); | ||||
| 				this.files[exist] = file; | ||||
| 				return; | ||||
| 			} | ||||
|  | ||||
| @@ -543,7 +539,7 @@ export default Vue.extend({ | ||||
| 			const filesMax = 30; | ||||
|  | ||||
| 			// フォルダ一覧取得 | ||||
| 			this.$root.api('drive/folders', { | ||||
| 			os.api('drive/folders', { | ||||
| 				folderId: this.folder ? this.folder.id : null, | ||||
| 				limit: foldersMax + 1 | ||||
| 			}).then(folders => { | ||||
| @@ -556,7 +552,7 @@ export default Vue.extend({ | ||||
| 			}); | ||||
|  | ||||
| 			// ファイル一覧取得 | ||||
| 			this.$root.api('drive/files', { | ||||
| 			os.api('drive/files', { | ||||
| 				folderId: this.folder ? this.folder.id : null, | ||||
| 				type: this.type, | ||||
| 				limit: filesMax + 1 | ||||
| @@ -587,7 +583,7 @@ export default Vue.extend({ | ||||
| 			const max = 30; | ||||
|  | ||||
| 			// ファイル一覧取得 | ||||
| 			this.$root.api('drive/files', { | ||||
| 			os.api('drive/files', { | ||||
| 				folderId: this.folder ? this.folder.id : null, | ||||
| 				type: this.type, | ||||
| 				untilId: this.files[this.files.length - 1].id, | ||||
| @@ -602,17 +598,57 @@ export default Vue.extend({ | ||||
| 				for (const x of files) this.appendFile(x); | ||||
| 				this.fetching = false; | ||||
| 			}); | ||||
| 		} | ||||
| 		}, | ||||
|  | ||||
| 		getMenu() { | ||||
| 			return [{ | ||||
| 				text: this.$t('addFile'), | ||||
| 				type: 'label' | ||||
| 			}, { | ||||
| 				text: this.$t('upload'), | ||||
| 				icon: faUpload, | ||||
| 				action: () => { this.selectLocalFile(); } | ||||
| 			}, { | ||||
| 				text: this.$t('fromUrl'), | ||||
| 				icon: faLink, | ||||
| 				action: () => { this.urlUpload(); } | ||||
| 			}, null, { | ||||
| 				text: this.folder ? this.folder.name : this.$t('drive'), | ||||
| 				type: 'label' | ||||
| 			}, this.folder ? { | ||||
| 				text: this.$t('renameFolder'), | ||||
| 				icon: faICursor, | ||||
| 				action: () => { this.renameFolder(this.folder); } | ||||
| 			} : undefined, this.folder ? { | ||||
| 				text: this.$t('deleteFolder'), | ||||
| 				icon: faTrashAlt, | ||||
| 				action: () => { this.deleteFolder(this.folder); } | ||||
| 			} : undefined, { | ||||
| 				text: this.$t('createFolder'), | ||||
| 				icon: faFolderPlus, | ||||
| 				action: () => { this.createFolder(); } | ||||
| 			}]; | ||||
| 		}, | ||||
|  | ||||
| 		onContextmenu(e) { | ||||
| 			os.contextMenu(this.getMenu(), e); | ||||
| 		}, | ||||
| 	} | ||||
| }); | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" scoped> | ||||
| .yfudmmck { | ||||
| 	display: flex; | ||||
| 	flex-direction: column; | ||||
| 	height: 100%; | ||||
|  | ||||
| 	> nav { | ||||
| 		display: block; | ||||
| 		z-index: 2; | ||||
| 		width: 100%; | ||||
| 		padding: 0 8px; | ||||
| 		box-sizing: border-box; | ||||
| 		overflow: auto; | ||||
| 		font-size: 0.9em; | ||||
| 		box-shadow: 0 1px 0 var(--divider); | ||||
| @@ -666,7 +702,7 @@ export default Vue.extend({ | ||||
| 	} | ||||
|  | ||||
| 	> .main { | ||||
| 		padding: 8px 0; | ||||
| 		flex: 1; | ||||
| 		overflow: auto; | ||||
|  | ||||
| 		&, * { | ||||
| @@ -734,11 +770,6 @@ export default Vue.extend({ | ||||
| 		pointer-events: none; | ||||
| 	} | ||||
|  | ||||
| 	> .mk-uploader { | ||||
| 		height: 100px; | ||||
| 		padding: 16px; | ||||
| 	} | ||||
|  | ||||
| 	> input { | ||||
| 		display: none; | ||||
| 	} | ||||
|   | ||||
| @@ -1,97 +1,138 @@ | ||||
| <template> | ||||
| <x-popup :source="source" ref="popup" @closed="() => { $emit('closed'); destroyDom(); }"> | ||||
| 	<div class="omfetrab"> | ||||
| 		<header> | ||||
| 			<button v-for="(category, i) in categories" | ||||
| 				class="_button" | ||||
| 				@click="go(category)" | ||||
| 				:class="{ active: category.isActive }" | ||||
| 				:key="i" | ||||
| 			> | ||||
| 				<fa :icon="category.icon" fixed-width/> | ||||
| 			</button> | ||||
| 		</header> | ||||
|  | ||||
| <MkModal ref="modal" :src="src" @click="$refs.modal.close()" @closed="$emit('closed')"> | ||||
| 	<div class="omfetrab _popup" :class="{ compact }"> | ||||
| 		<input ref="search" class="search" :class="{ filled: q != null && q != '' }" v-model.trim="q" :placeholder="$t('search')" @paste.stop="paste" @keyup.enter="done()"> | ||||
| 		<div class="emojis"> | ||||
| 			<template v-if="categories[0].isActive"> | ||||
| 				<header class="category"><fa :icon="faHistory" fixed-width/> {{ $t('recentUsed') }}</header> | ||||
| 				<div class="list"> | ||||
| 					<button v-for="(emoji, i) in ($store.state.device.recentEmojis || [])" | ||||
| 			<section class="result"> | ||||
| 				<div v-if="searchResultCustom.length > 0"> | ||||
| 					<button v-for="emoji in searchResultCustom" | ||||
| 						class="_button" | ||||
| 						:title="emoji.name" | ||||
| 						@click="chosen(emoji)" | ||||
| 						:key="i" | ||||
| 						@click="chosen(emoji, $event)" | ||||
| 						:key="emoji" | ||||
| 						tabindex="0" | ||||
| 					> | ||||
| 						<mk-emoji v-if="emoji.char != null" :emoji="emoji.char"/> | ||||
| 						<MkEmoji v-if="emoji.char != null" :emoji="emoji.char"/> | ||||
| 						<img v-else :src="$store.state.device.disableShowingAnimatedImages ? getStaticImageUrl(emoji.url) : emoji.url"/> | ||||
| 					</button> | ||||
| 				</div> | ||||
|  | ||||
| 				<header class="category"><fa :icon="faAsterisk" fixed-width/> {{ $t('customEmojis') }}</header> | ||||
| 			</template> | ||||
|  | ||||
| 			<template v-if="categories.find(x => x.isActive).name"> | ||||
| 				<div class="list"> | ||||
| 					<button v-for="emoji in emojilist.filter(e => e.category === categories.find(x => x.isActive).name)" | ||||
| 				<div v-if="searchResultUnicode.length > 0"> | ||||
| 					<button v-for="emoji in searchResultUnicode" | ||||
| 						class="_button" | ||||
| 						:title="emoji.name" | ||||
| 						@click="chosen(emoji)" | ||||
| 						@click="chosen(emoji, $event)" | ||||
| 						:key="emoji.name" | ||||
| 						tabindex="0" | ||||
| 					> | ||||
| 						<mk-emoji :emoji="emoji.char"/> | ||||
| 						<MkEmoji :emoji="emoji.char"/> | ||||
| 					</button> | ||||
| 				</div> | ||||
| 			</template> | ||||
| 			<template v-else> | ||||
| 				<div v-for="(key, i) in Object.keys(customEmojis)" :key="i"> | ||||
| 					<header class="sub" v-if="key">{{ key }}</header> | ||||
| 					<div class="list"> | ||||
| 						<button v-for="emoji in customEmojis[key]" | ||||
| 			</section> | ||||
|  | ||||
| 			<div class="index"> | ||||
| 				<section v-if="showPinned"> | ||||
| 					<div> | ||||
| 						<button v-for="emoji in pinned" | ||||
| 							class="_button" | ||||
| 							:title="emoji.name" | ||||
| 							@click="chosen(emoji)" | ||||
| 							:key="emoji.name" | ||||
| 							@click="chosen(emoji, $event)" | ||||
| 							tabindex="0" | ||||
| 						> | ||||
| 							<img :src="$store.state.device.disableShowingAnimatedImages ? getStaticImageUrl(emoji.url) : emoji.url"/> | ||||
| 							<MkEmoji :emoji="emoji" :normal="true"/> | ||||
| 						</button> | ||||
| 					</div> | ||||
| 				</section> | ||||
|  | ||||
| 				<section> | ||||
| 					<header class="_acrylic"><Fa :icon="faClock" fixed-width/> {{ $t('recentUsed') }}</header> | ||||
| 					<div> | ||||
| 						<button v-for="emoji in $store.state.device.recentlyUsedEmojis" | ||||
| 							class="_button" | ||||
| 							@click="chosen(emoji, $event)" | ||||
| 							:key="emoji" | ||||
| 						> | ||||
| 							<MkEmoji :emoji="emoji" :normal="true"/> | ||||
| 						</button> | ||||
| 					</div> | ||||
| 				</section> | ||||
|  | ||||
| 				<div class="arrow"><Fa :icon="faChevronDown"/></div> | ||||
| 			</div> | ||||
|  | ||||
| 			<section v-for="category in customEmojiCategories" :key="'custom:' + category" class="custom"> | ||||
| 				<header class="_acrylic" v-appear="() => visibleCategories[category] = true">{{ category || $t('other') }}</header> | ||||
| 				<div v-if="visibleCategories[category]"> | ||||
| 					<button v-for="emoji in customEmojis.filter(e => e.category === category)" | ||||
| 						class="_button" | ||||
| 						:title="emoji.name" | ||||
| 						@click="chosen(emoji, $event)" | ||||
| 						:key="emoji.name" | ||||
| 					> | ||||
| 						<img :src="$store.state.device.disableShowingAnimatedImages ? getStaticImageUrl(emoji.url) : emoji.url"/> | ||||
| 					</button> | ||||
| 				</div> | ||||
| 			</template> | ||||
| 			</section> | ||||
|  | ||||
| 			<section v-for="category in categories" :key="category.name" class="unicode"> | ||||
| 				<header class="_acrylic" v-appear="() => category.isActive = true"><Fa :icon="category.icon" fixed-width/> {{ category.name }}</header> | ||||
| 				<div v-if="category.isActive"> | ||||
| 					<button v-for="emoji in emojilist.filter(e => e.category === category.name)" | ||||
| 						class="_button" | ||||
| 						:title="emoji.name" | ||||
| 						@click="chosen(emoji, $event)" | ||||
| 						:key="emoji.name" | ||||
| 					> | ||||
| 						<MkEmoji :emoji="emoji.char"/> | ||||
| 					</button> | ||||
| 				</div> | ||||
| 			</section> | ||||
| 		</div> | ||||
| 	</div> | ||||
| </x-popup> | ||||
| </MkModal> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import Vue from 'vue'; | ||||
| import { defineComponent, markRaw } from 'vue'; | ||||
| import { emojilist } from '../../misc/emojilist'; | ||||
| import { getStaticImageUrl } from '../scripts/get-static-image-url'; | ||||
| import { faAsterisk, faLeaf, faUtensils, faFutbol, faCity, faDice, faGlobe, faHistory, faUser } from '@fortawesome/free-solid-svg-icons'; | ||||
| import { getStaticImageUrl } from '@/scripts/get-static-image-url'; | ||||
| import { faAsterisk, faLeaf, faUtensils, faFutbol, faCity, faDice, faGlobe, faClock, faUser, faChevronDown } from '@fortawesome/free-solid-svg-icons'; | ||||
| import { faHeart, faFlag, faLaugh } from '@fortawesome/free-regular-svg-icons'; | ||||
| import { groupByX } from '../../prelude/array'; | ||||
| import XPopup from './popup.vue'; | ||||
| import MkModal from '@/components/ui/modal.vue'; | ||||
| import Particle from '@/components/particle.vue'; | ||||
| import * as os from '@/os'; | ||||
|  | ||||
| export default Vue.extend({ | ||||
| export default defineComponent({ | ||||
| 	components: { | ||||
| 		XPopup, | ||||
| 		MkModal, | ||||
| 	}, | ||||
|  | ||||
| 	props: { | ||||
| 		source: { | ||||
| 			required: true | ||||
| 		src: { | ||||
| 			required: false | ||||
| 		}, | ||||
| 		showPinned: { | ||||
| 			required: false, | ||||
| 			default: true | ||||
| 		}, | ||||
| 		compact: { | ||||
| 			required: false | ||||
| 		}, | ||||
| 	}, | ||||
|  | ||||
| 	emits: ['done', 'closed'], | ||||
|  | ||||
| 	data() { | ||||
| 		return { | ||||
| 			emojilist, | ||||
| 			emojilist: markRaw(emojilist), | ||||
| 			getStaticImageUrl, | ||||
| 			customEmojis: {}, | ||||
| 			faGlobe, faHistory, | ||||
| 			pinned: this.$store.state.settings.reactions, | ||||
| 			customEmojiCategories: this.$store.getters['instance/emojiCategories'], | ||||
| 			customEmojis: this.$store.state.instance.meta.emojis, | ||||
| 			visibleCategories: {}, | ||||
| 			q: null, | ||||
| 			searchResultCustom: [], | ||||
| 			searchResultUnicode: [], | ||||
| 			faGlobe, faClock, faChevronDown, | ||||
| 			categories: [{ | ||||
| 				icon: faAsterisk, | ||||
| 				isActive: true | ||||
| 			}, { | ||||
| 				name: 'face', | ||||
| 				icon: faLaugh, | ||||
| 				isActive: false | ||||
| @@ -132,129 +173,336 @@ export default Vue.extend({ | ||||
| 		}; | ||||
| 	}, | ||||
|  | ||||
| 	created() { | ||||
| 		let local = this.$store.state.instance.meta.emojis; | ||||
| 		local = groupByX(local, (x: any) => x.category || ''); | ||||
| 		this.customEmojis = local; | ||||
| 	watch: { | ||||
| 		q() { | ||||
| 			if (this.q == null || this.q === '') { | ||||
| 				this.searchResultCustom = []; | ||||
| 				this.searchResultUnicode = []; | ||||
| 				return; | ||||
| 			} | ||||
|  | ||||
| 			const q = this.q.replace(/:/g, ''); | ||||
|  | ||||
| 			const searchCustom = () => { | ||||
| 				const max = 8; | ||||
| 				const emojis = this.customEmojis; | ||||
| 				const matches = new Set(); | ||||
|  | ||||
| 				const exactMatch = emojis.find(e => e.name === q); | ||||
| 				if (exactMatch) matches.add(exactMatch); | ||||
|  | ||||
| 				if (q.includes(' ')) { // AND検索 | ||||
| 					const keywords = q.split(' '); | ||||
|  | ||||
| 					// 名前にキーワードが含まれている | ||||
| 					for (const emoji of emojis) { | ||||
| 						if (keywords.every(keyword => emoji.name.includes(keyword))) { | ||||
| 							matches.add(emoji); | ||||
| 							if (matches.size >= max) break; | ||||
| 						} | ||||
| 					} | ||||
| 					if (matches.size >= max) return matches; | ||||
|  | ||||
| 					// 名前またはエイリアスにキーワードが含まれている | ||||
| 					for (const emoji of emojis) { | ||||
| 						if (keywords.every(keyword => emoji.name.includes(keyword) || emoji.aliases.some(alias => alias.includes(keyword)))) { | ||||
| 							matches.add(emoji); | ||||
| 							if (matches.size >= max) break; | ||||
| 						} | ||||
| 					} | ||||
| 				} else { | ||||
| 					for (const emoji of emojis) { | ||||
| 						if (emoji.name.startsWith(q)) { | ||||
| 							matches.add(emoji); | ||||
| 							if (matches.size >= max) break; | ||||
| 						} | ||||
| 					} | ||||
| 					if (matches.size >= max) return matches; | ||||
|  | ||||
| 					for (const emoji of emojis) { | ||||
| 						if (emoji.aliases.some(alias => alias.startsWith(q))) { | ||||
| 							matches.add(emoji); | ||||
| 							if (matches.size >= max) break; | ||||
| 						} | ||||
| 					} | ||||
| 					if (matches.size >= max) return matches; | ||||
|  | ||||
| 					for (const emoji of emojis) { | ||||
| 						if (emoji.name.includes(q)) { | ||||
| 							matches.add(emoji); | ||||
| 							if (matches.size >= max) break; | ||||
| 						} | ||||
| 					} | ||||
| 					if (matches.size >= max) return matches; | ||||
|  | ||||
| 					for (const emoji of emojis) { | ||||
| 						if (emoji.aliases.some(alias => alias.includes(q))) { | ||||
| 							matches.add(emoji); | ||||
| 							if (matches.size >= max) break; | ||||
| 						} | ||||
| 					} | ||||
| 				} | ||||
|  | ||||
| 				return matches; | ||||
| 			}; | ||||
|  | ||||
| 			const searchUnicode = () => { | ||||
| 				const max = 8; | ||||
| 				const emojis = this.emojilist; | ||||
| 				const matches = new Set(); | ||||
|  | ||||
| 				const exactMatch = emojis.find(e => e.name === q); | ||||
| 				if (exactMatch) matches.add(exactMatch); | ||||
|  | ||||
| 				if (q.includes(' ')) { // AND検索 | ||||
| 					const keywords = q.split(' '); | ||||
|  | ||||
| 					// 名前にキーワードが含まれている | ||||
| 					for (const emoji of emojis) { | ||||
| 						if (keywords.every(keyword => emoji.name.includes(keyword))) { | ||||
| 							matches.add(emoji); | ||||
| 							if (matches.size >= max) break; | ||||
| 						} | ||||
| 					} | ||||
| 					if (matches.size >= max) return matches; | ||||
|  | ||||
| 					// 名前またはエイリアスにキーワードが含まれている | ||||
| 					for (const emoji of emojis) { | ||||
| 						if (keywords.every(keyword => emoji.name.includes(keyword) || emoji.keywords.some(alias => alias.includes(keyword)))) { | ||||
| 							matches.add(emoji); | ||||
| 							if (matches.size >= max) break; | ||||
| 						} | ||||
| 					} | ||||
| 				} else { | ||||
| 					for (const emoji of emojis) { | ||||
| 						if (emoji.name.startsWith(q)) { | ||||
| 							matches.add(emoji); | ||||
| 							if (matches.size >= max) break; | ||||
| 						} | ||||
| 					} | ||||
| 					if (matches.size >= max) return matches; | ||||
|  | ||||
| 					for (const emoji of emojis) { | ||||
| 						if (emoji.keywords.some(keyword => keyword.startsWith(q))) { | ||||
| 							matches.add(emoji); | ||||
| 							if (matches.size >= max) break; | ||||
| 						} | ||||
| 					} | ||||
| 					if (matches.size >= max) return matches; | ||||
|  | ||||
| 					for (const emoji of emojis) { | ||||
| 						if (emoji.name.includes(q)) { | ||||
| 							matches.add(emoji); | ||||
| 							if (matches.size >= max) break; | ||||
| 						} | ||||
| 					} | ||||
| 					if (matches.size >= max) return matches; | ||||
|  | ||||
| 					for (const emoji of emojis) { | ||||
| 						if (emoji.keywords.some(keyword => keyword.includes(q))) { | ||||
| 							matches.add(emoji); | ||||
| 							if (matches.size >= max) break; | ||||
| 						} | ||||
| 					} | ||||
| 				} | ||||
|  | ||||
| 				return matches; | ||||
| 			}; | ||||
|  | ||||
| 			this.searchResultCustom = Array.from(searchCustom()); | ||||
| 			this.searchResultUnicode = Array.from(searchUnicode()); | ||||
| 		} | ||||
| 	}, | ||||
|  | ||||
| 	mounted() { | ||||
| 		const isIos = navigator.userAgent.includes('WebKit') && !navigator.userAgent.includes('Chrome'); | ||||
| 		if (!isIos) { | ||||
| 			this.$refs.search.focus({ | ||||
| 				preventScroll: true | ||||
| 			}); | ||||
| 		} | ||||
| 	}, | ||||
|  | ||||
| 	methods: { | ||||
| 		go(category: any) { | ||||
| 			this.goCategory(category.name); | ||||
| 		getKey(emoji: any) { | ||||
| 			return typeof emoji === 'string' ? emoji : (emoji.char || `:${emoji.name}:`); | ||||
| 		}, | ||||
|  | ||||
| 		goCategory(name: string) { | ||||
| 			let matched = false; | ||||
| 			for (const c of this.categories) { | ||||
| 				c.isActive = c.name === name; | ||||
| 				if (c.isActive) { | ||||
| 					matched = true; | ||||
| 				} | ||||
| 		chosen(emoji: any, ev) { | ||||
| 			if (ev) { | ||||
| 				const el = ev.currentTarget || ev.target; | ||||
| 				const rect = el.getBoundingClientRect(); | ||||
| 				const x = rect.left + (el.clientWidth / 2); | ||||
| 				const y = rect.top + (el.clientHeight / 2); | ||||
| 				os.popup(Particle, { x, y }, {}, 'end'); | ||||
| 			} | ||||
| 			if (!matched) { | ||||
| 				this.categories[0].isActive = true; | ||||
|  | ||||
| 			const key = this.getKey(emoji); | ||||
| 			this.$emit('done', key); | ||||
| 			this.$refs.modal.close(); | ||||
|  | ||||
| 			// 最近使った絵文字更新 | ||||
| 			if (!this.pinned.includes(key)) { | ||||
| 				let recents = this.$store.state.device.recentlyUsedEmojis; | ||||
| 				recents = recents.filter((e: any) => e !== key); | ||||
| 				recents.unshift(key); | ||||
| 				this.$store.commit('device/set', { key: 'recentlyUsedEmojis', value: recents.splice(0, 16) }); | ||||
| 			} | ||||
| 		}, | ||||
|  | ||||
| 		chosen(emoji: any) { | ||||
| 			const getKey = (emoji: any) => emoji.char || `:${emoji.name}:`; | ||||
| 			let recents = this.$store.state.device.recentEmojis || []; | ||||
| 			recents = recents.filter((e: any) => getKey(e) !== getKey(emoji)); | ||||
| 			recents.unshift(emoji) | ||||
| 			this.$store.commit('device/set', { key: 'recentEmojis', value: recents.splice(0, 16) }); | ||||
| 			this.$emit('chosen', getKey(emoji)); | ||||
| 		paste(event) { | ||||
| 			const paste = (event.clipboardData || window.clipboardData).getData('text'); | ||||
| 			if (this.done(paste)) { | ||||
| 				event.preventDefault(); | ||||
| 			} | ||||
| 		}, | ||||
|  | ||||
| 		close() { | ||||
| 			this.$refs.popup.close(); | ||||
| 		} | ||||
| 		done(query) { | ||||
| 			if (query == null) query = this.q; | ||||
| 			if (query == null) return; | ||||
| 			const q = query.replace(/:/g, ''); | ||||
| 			const exactMatchCustom = this.customEmojis.find(e => e.name === q); | ||||
| 			if (exactMatchCustom) { | ||||
| 				this.chosen(exactMatchCustom); | ||||
| 				return true; | ||||
| 			} | ||||
| 			const exactMatchUnicode = this.emojilist.find(e => e.char === q || e.name === q); | ||||
| 			if (exactMatchUnicode) { | ||||
| 				this.chosen(exactMatchUnicode); | ||||
| 				return true; | ||||
| 			} | ||||
| 			if (this.searchResultCustom.length > 0) { | ||||
| 				this.chosen(this.searchResultCustom[0]); | ||||
| 				return true; | ||||
| 			} | ||||
| 			if (this.searchResultUnicode.length > 0) { | ||||
| 				this.chosen(this.searchResultUnicode[0]); | ||||
| 				return true; | ||||
| 			} | ||||
| 		}, | ||||
| 	} | ||||
| }); | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" scoped> | ||||
| .omfetrab { | ||||
| 	width: 350px; | ||||
| 	$eachSize: 40px; | ||||
| 	$pad: 8px; | ||||
|  | ||||
| 	> header { | ||||
| 		display: flex; | ||||
| 	display: flex; | ||||
| 	flex-direction: column; | ||||
| 	width: ($eachSize * 7) + ($pad * 2); | ||||
| 	contain: content; | ||||
| 	--height: 300px; | ||||
|  | ||||
| 		> button { | ||||
| 			flex: 1; | ||||
| 			padding: 10px 0; | ||||
| 			font-size: 16px; | ||||
| 			transition: color 0.2s ease; | ||||
| 	&.compact { | ||||
| 		width: ($eachSize * 5) + ($pad * 2); | ||||
| 		--height: 210px; | ||||
| 	} | ||||
|  | ||||
| 			&:hover { | ||||
| 				color: var(--textHighlighted); | ||||
| 				transition: color 0s; | ||||
| 			} | ||||
| 	> .search { | ||||
| 		width: 100%; | ||||
| 		padding: 12px; | ||||
| 		box-sizing: border-box; | ||||
| 		font-size: 1em; | ||||
| 		outline: none; | ||||
| 		border: none; | ||||
| 		background: transparent; | ||||
| 		color: var(--fg); | ||||
|  | ||||
| 			&.active { | ||||
| 				color: var(--accent); | ||||
| 				transition: color 0s; | ||||
| 			} | ||||
| 		&:not(.filled) { | ||||
| 			order: 1; | ||||
| 			z-index: 2; | ||||
| 			box-shadow: 0px -1px 0 0px var(--divider); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	> .emojis { | ||||
| 		height: 300px; | ||||
| 		height: var(--height); | ||||
| 		overflow-y: auto; | ||||
| 		overflow-x: hidden; | ||||
|  | ||||
| 		> header.category { | ||||
| 			position: sticky; | ||||
| 			top: 0; | ||||
| 			left: 0; | ||||
| 			z-index: 1; | ||||
| 			padding: 8px; | ||||
| 			background: var(--panel); | ||||
| 			font-size: 12px; | ||||
| 		scrollbar-width: none; | ||||
|  | ||||
| 		&::-webkit-scrollbar { | ||||
| 			display: none; | ||||
| 		} | ||||
|  | ||||
| 		header.sub { | ||||
| 			padding: 4px 8px; | ||||
| 			font-size: 12px; | ||||
| 		} | ||||
| 		> .index { | ||||
| 			min-height: var(--height); | ||||
| 			position: relative; | ||||
| 			border-bottom: solid 1px var(--divider); | ||||
| 				 | ||||
| 		div.list { | ||||
| 			display: grid; | ||||
| 			grid-template-columns: 1fr 1fr 1fr 1fr 1fr 1fr 1fr 1fr; | ||||
| 			gap: 4px; | ||||
| 			padding: 8px; | ||||
|  | ||||
| 			> button { | ||||
| 				position: relative; | ||||
| 				padding: 0; | ||||
| 			> .arrow { | ||||
| 				position: absolute; | ||||
| 				bottom: 0; | ||||
| 				left: 0; | ||||
| 				width: 100%; | ||||
| 				padding: 16px 0; | ||||
| 				text-align: center; | ||||
| 				opacity: 0.5; | ||||
| 				pointer-events: none; | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 				&:before { | ||||
| 					content: ''; | ||||
| 					display: block; | ||||
| 					width: 1px; | ||||
| 					height: 0; | ||||
| 					padding-bottom: 100%; | ||||
| 				} | ||||
| 		section { | ||||
| 			> header { | ||||
| 				position: sticky; | ||||
| 				top: 0; | ||||
| 				left: 0; | ||||
| 				z-index: 1; | ||||
| 				padding: 8px; | ||||
| 				font-size: 12px; | ||||
| 			} | ||||
|  | ||||
| 			> div { | ||||
| 				padding: $pad; | ||||
|  | ||||
| 				> button { | ||||
| 					position: relative; | ||||
| 					padding: 0; | ||||
| 					width: $eachSize; | ||||
| 					height: $eachSize; | ||||
| 					border-radius: 4px; | ||||
|  | ||||
| 					&:focus { | ||||
| 						outline: solid 2px var(--focus); | ||||
| 						z-index: 1; | ||||
| 					} | ||||
|  | ||||
| 					&:hover { | ||||
| 						background: rgba(0, 0, 0, 0.05); | ||||
| 					} | ||||
|  | ||||
| 					&:active { | ||||
| 						background: var(--accent); | ||||
| 						box-shadow: inset 0 0.15em 0.3em rgba(27, 31, 35, 0.15); | ||||
| 					} | ||||
|  | ||||
| 				&:hover { | ||||
| 					> * { | ||||
| 						transform: scale(1.2); | ||||
| 						transition: transform 0s; | ||||
| 						font-size: 24px; | ||||
| 						height: 1.25em; | ||||
| 						vertical-align: -.25em; | ||||
| 						pointer-events: none; | ||||
| 					} | ||||
| 				} | ||||
| 			} | ||||
|  | ||||
| 				> * { | ||||
| 					position: absolute; | ||||
| 					top: 0; | ||||
| 					left: 0; | ||||
| 					width: 100%; | ||||
| 					height: 100%; | ||||
| 					object-fit: contain; | ||||
| 					font-size: 28px; | ||||
| 					transition: transform 0.2s ease; | ||||
| 					pointer-events: none; | ||||
| 			&.result { | ||||
| 				border-bottom: solid 1px var(--divider); | ||||
|  | ||||
| 				&:empty { | ||||
| 					display: none; | ||||
| 				} | ||||
| 			} | ||||
|  | ||||
| 			&.unicode { | ||||
| 				min-height: 384px; | ||||
| 			} | ||||
|  | ||||
| 			&.custom { | ||||
| 				min-height: 64px; | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|   | ||||
| @@ -2,23 +2,19 @@ | ||||
| <img v-if="customEmoji" class="mk-emoji custom" :class="{ normal, noStyle }" :src="url" :alt="alt" :title="alt"/> | ||||
| <img v-else-if="char && !useOsNativeEmojis" class="mk-emoji" :src="url" :alt="alt" :title="alt"/> | ||||
| <span v-else-if="char && useOsNativeEmojis">{{ char }}</span> | ||||
| <span v-else>:{{ name }}:</span> | ||||
| <span v-else>{{ emoji }}</span> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import Vue from 'vue'; | ||||
| import { getStaticImageUrl } from '../scripts/get-static-image-url'; | ||||
| import { defineComponent } from 'vue'; | ||||
| import { getStaticImageUrl } from '@/scripts/get-static-image-url'; | ||||
| import { twemojiSvgBase } from '../../misc/twemoji-base'; | ||||
|  | ||||
| export default Vue.extend({ | ||||
| export default defineComponent({ | ||||
| 	props: { | ||||
| 		name: { | ||||
| 			type: String, | ||||
| 			required: false | ||||
| 		}, | ||||
| 		emoji: { | ||||
| 			type: String, | ||||
| 			required: false | ||||
| 			required: true | ||||
| 		}, | ||||
| 		normal: { | ||||
| 			type: Boolean, | ||||
| @@ -49,6 +45,10 @@ export default Vue.extend({ | ||||
| 	}, | ||||
|  | ||||
| 	computed: { | ||||
| 		isCustom(): boolean { | ||||
| 			return this.emoji.startsWith(':'); | ||||
| 		}, | ||||
|  | ||||
| 		alt(): string { | ||||
| 			return this.customEmoji ? `:${this.customEmoji.name}:` : this.char; | ||||
| 		}, | ||||
| @@ -68,8 +68,8 @@ export default Vue.extend({ | ||||
| 	watch: { | ||||
| 		ce: { | ||||
| 			handler() { | ||||
| 				if (this.name) { | ||||
| 					const customEmoji = this.ce.find(x => x.name == this.name); | ||||
| 				if (this.isCustom) { | ||||
| 					const customEmoji = this.ce.find(x => x.name === this.emoji.substr(1, this.emoji.length - 2)); | ||||
| 					if (customEmoji) { | ||||
| 						this.customEmoji = customEmoji; | ||||
| 						this.url = this.$store.state.device.disableShowingAnimatedImages | ||||
| @@ -83,7 +83,7 @@ export default Vue.extend({ | ||||
| 	}, | ||||
|  | ||||
| 	created() { | ||||
| 		if (!this.name) { | ||||
| 		if (!this.isCustom) { | ||||
| 			this.char = this.emoji; | ||||
| 		} | ||||
|  | ||||
|   | ||||
| @@ -1,19 +1,19 @@ | ||||
| <template> | ||||
| <transition :name="$store.state.device.animation ? 'zoom' : ''" appear> | ||||
| 	<div class="mjndxjcg _panel"> | ||||
| 	<div class="mjndxjcg"> | ||||
| 		<img src="https://xn--931a.moe/assets/error.jpg" class="_ghost"/> | ||||
| 		<p><fa :icon="faExclamationTriangle"/> {{ $t('error') }}</p> | ||||
| 		<mk-button @click="() => $emit('retry')" class="button">{{ $t('retry') }}</mk-button> | ||||
| 		<p><Fa :icon="faExclamationTriangle"/> {{ $t('somethingHappened') }}</p> | ||||
| 		<MkButton @click="() => $emit('retry')" class="button">{{ $t('retry') }}</MkButton> | ||||
| 	</div> | ||||
| </transition> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import Vue from 'vue'; | ||||
| import { defineComponent } from 'vue'; | ||||
| import { faExclamationTriangle } from '@fortawesome/free-solid-svg-icons'; | ||||
| import MkButton from './ui/button.vue'; | ||||
|  | ||||
| export default Vue.extend({ | ||||
| export default defineComponent({ | ||||
| 	components: { | ||||
| 		MkButton, | ||||
| 	}, | ||||
|   | ||||
| @@ -1,14 +1,15 @@ | ||||
| <template> | ||||
| <span class="mk-file-type-icon"> | ||||
| 	<template v-if="kind == 'image'"><fa :icon="faFileImage"/></template> | ||||
| 	<template v-if="kind == 'image'"><Fa :icon="faFileImage"/></template> | ||||
| </span> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import Vue from 'vue'; | ||||
| import { defineComponent } from 'vue'; | ||||
| import { faFileImage } from '@fortawesome/free-solid-svg-icons'; | ||||
| import * as os from '@/os'; | ||||
|  | ||||
| export default Vue.extend({ | ||||
| export default defineComponent({ | ||||
| 	props: { | ||||
| 		type: { | ||||
| 			type: String, | ||||
|   | ||||
| @@ -7,32 +7,33 @@ | ||||
| > | ||||
| 	<template v-if="!wait"> | ||||
| 		<template v-if="hasPendingFollowRequestFromYou && user.isLocked"> | ||||
| 			<span v-if="full">{{ $t('followRequestPending') }}</span><fa :icon="faHourglassHalf"/> | ||||
| 			<span v-if="full">{{ $t('followRequestPending') }}</span><Fa :icon="faHourglassHalf"/> | ||||
| 		</template> | ||||
| 		<template v-else-if="hasPendingFollowRequestFromYou && !user.isLocked"> <!-- つまりリモートフォローの場合。 --> | ||||
| 			<span v-if="full">{{ $t('processing') }}</span><fa :icon="faSpinner" pulse/> | ||||
| 			<span v-if="full">{{ $t('processing') }}</span><Fa :icon="faSpinner" pulse/> | ||||
| 		</template> | ||||
| 		<template v-else-if="isFollowing"> | ||||
| 			<span v-if="full">{{ $t('unfollow') }}</span><fa :icon="faMinus"/> | ||||
| 			<span v-if="full">{{ $t('unfollow') }}</span><Fa :icon="faMinus"/> | ||||
| 		</template> | ||||
| 		<template v-else-if="!isFollowing && user.isLocked"> | ||||
| 			<span v-if="full">{{ $t('followRequest') }}</span><fa :icon="faPlus"/> | ||||
| 			<span v-if="full">{{ $t('followRequest') }}</span><Fa :icon="faPlus"/> | ||||
| 		</template> | ||||
| 		<template v-else-if="!isFollowing && !user.isLocked"> | ||||
| 			<span v-if="full">{{ $t('follow') }}</span><fa :icon="faPlus"/> | ||||
| 			<span v-if="full">{{ $t('follow') }}</span><Fa :icon="faPlus"/> | ||||
| 		</template> | ||||
| 	</template> | ||||
| 	<template v-else> | ||||
| 		<span v-if="full">{{ $t('processing') }}</span><fa :icon="faSpinner" pulse fixed-width/> | ||||
| 		<span v-if="full">{{ $t('processing') }}</span><Fa :icon="faSpinner" pulse fixed-width/> | ||||
| 	</template> | ||||
| </button> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import Vue from 'vue'; | ||||
| import { defineComponent } from 'vue'; | ||||
| import { faSpinner, faPlus, faMinus, faHourglassHalf } from '@fortawesome/free-solid-svg-icons'; | ||||
| import * as os from '@/os'; | ||||
|  | ||||
| export default Vue.extend({ | ||||
| export default defineComponent({ | ||||
| 	props: { | ||||
| 		user: { | ||||
| 			type: Object, | ||||
| @@ -58,7 +59,7 @@ export default Vue.extend({ | ||||
| 	created() { | ||||
| 		// 渡されたユーザー情報が不完全な場合 | ||||
| 		if (this.user.isFollowing == null) { | ||||
| 			this.$root.api('users/show', { | ||||
| 			os.api('users/show', { | ||||
| 				userId: this.user.id | ||||
| 			}).then(u => { | ||||
| 				this.isFollowing = u.isFollowing; | ||||
| @@ -68,13 +69,13 @@ export default Vue.extend({ | ||||
| 	}, | ||||
|  | ||||
| 	mounted() { | ||||
| 		this.connection = this.$root.stream.useSharedConnection('main'); | ||||
| 		this.connection = os.stream.useSharedConnection('main'); | ||||
|  | ||||
| 		this.connection.on('follow', this.onFollowChange); | ||||
| 		this.connection.on('unfollow', this.onFollowChange); | ||||
| 	}, | ||||
|  | ||||
| 	beforeDestroy() { | ||||
| 	beforeUnmount() { | ||||
| 		this.connection.dispose(); | ||||
| 	}, | ||||
|  | ||||
| @@ -91,7 +92,7 @@ export default Vue.extend({ | ||||
|  | ||||
| 			try { | ||||
| 				if (this.isFollowing) { | ||||
| 					const { canceled } = await this.$root.dialog({ | ||||
| 					const { canceled } = await os.dialog({ | ||||
| 						type: 'warning', | ||||
| 						text: this.$t('unfollowConfirm', { name: this.user.name || this.user.username }), | ||||
| 						showCancelButton: true | ||||
| @@ -99,21 +100,21 @@ export default Vue.extend({ | ||||
|  | ||||
| 					if (canceled) return; | ||||
|  | ||||
| 					await this.$root.api('following/delete', { | ||||
| 					await os.api('following/delete', { | ||||
| 						userId: this.user.id | ||||
| 					}); | ||||
| 				} else { | ||||
| 					if (this.hasPendingFollowRequestFromYou) { | ||||
| 						await this.$root.api('following/requests/cancel', { | ||||
| 						await os.api('following/requests/cancel', { | ||||
| 							userId: this.user.id | ||||
| 						}); | ||||
| 					} else if (this.user.isLocked) { | ||||
| 						await this.$root.api('following/create', { | ||||
| 						await os.api('following/create', { | ||||
| 							userId: this.user.id | ||||
| 						}); | ||||
| 						this.hasPendingFollowRequestFromYou = true; | ||||
| 					} else { | ||||
| 						await this.$root.api('following/create', { | ||||
| 						await os.api('following/create', { | ||||
| 							userId: this.user.id | ||||
| 						}); | ||||
| 						this.hasPendingFollowRequestFromYou = true; | ||||
|   | ||||
							
								
								
									
										106
									
								
								src/client/components/form-dialog.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										106
									
								
								src/client/components/form-dialog.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,106 @@ | ||||
| <template> | ||||
| <XModalWindow ref="dialog" | ||||
| 	:width="400" | ||||
| 	:can-close="false" | ||||
| 	:with-ok-button="true" | ||||
| 	:ok-button-disabled="false" | ||||
| 	@click="cancel()" | ||||
| 	@ok="ok()" | ||||
| 	@close="cancel()" | ||||
| 	@closed="$emit('closed')" | ||||
| > | ||||
| 	<template #header> | ||||
| 		{{ title }} | ||||
| 	</template> | ||||
| 	<div class="xkpnjxcv _section"> | ||||
| 		<label v-for="item in Object.keys(form).filter(item => !form[item].hidden)" :key="item"> | ||||
| 			<MkInput v-if="form[item].type === 'number'" v-model:value="values[item]" type="number" :step="form[item].step || 1"> | ||||
| 				<span v-text="form[item].label || item"></span><span v-if="form[item].required === false"> ({{ $t('optional') }})</span> | ||||
| 				<template v-if="form[item].description" #desc>{{ form[item].description }}</template> | ||||
| 			</MkInput> | ||||
| 			<MkInput v-else-if="form[item].type === 'string' && !form[item].multiline" v-model:value="values[item]" type="text"> | ||||
| 				<span v-text="form[item].label || item"></span><span v-if="form[item].required === false"> ({{ $t('optional') }})</span> | ||||
| 				<template v-if="form[item].description" #desc>{{ form[item].description }}</template> | ||||
| 			</MkInput> | ||||
| 			<MkTextarea v-else-if="form[item].type === 'string' && form[item].multiline" v-model:value="values[item]"> | ||||
| 				<span v-text="form[item].label || item"></span><span v-if="form[item].required === false"> ({{ $t('optional') }})</span> | ||||
| 				<template v-if="form[item].description" #desc>{{ form[item].description }}</template> | ||||
| 			</MkTextarea> | ||||
| 			<MkSwitch v-else-if="form[item].type === 'boolean'" v-model:value="values[item]"> | ||||
| 				<span v-text="form[item].label || item"></span> | ||||
| 				<template v-if="form[item].description" #desc>{{ form[item].description }}</template> | ||||
| 			</MkSwitch> | ||||
| 		</label> | ||||
| 	</div> | ||||
| </XModalWindow> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import { defineComponent } from 'vue'; | ||||
| import XModalWindow from '@/components/ui/modal-window.vue'; | ||||
| import MkInput from './ui/input.vue'; | ||||
| import MkTextarea from './ui/textarea.vue'; | ||||
| import MkSwitch from './ui/switch.vue'; | ||||
|  | ||||
| export default defineComponent({ | ||||
| 	components: { | ||||
| 		XModalWindow, | ||||
| 		MkInput, | ||||
| 		MkTextarea, | ||||
| 		MkSwitch, | ||||
| 	}, | ||||
|  | ||||
| 	props: { | ||||
| 		title: { | ||||
| 			type: String, | ||||
| 			required: true, | ||||
| 		}, | ||||
| 		form: { | ||||
| 			type: Object, | ||||
| 			required: true, | ||||
| 		}, | ||||
| 	}, | ||||
|  | ||||
| 	emits: ['done'], | ||||
|  | ||||
| 	data() { | ||||
| 		return { | ||||
| 			values: {} | ||||
| 		}; | ||||
| 	}, | ||||
|  | ||||
| 	created() { | ||||
| 		for (const item in this.form) { | ||||
| 			this.values[item] = this.form[item].hasOwnProperty('default') ? this.form[item].default : null; | ||||
| 		} | ||||
| 	}, | ||||
|  | ||||
| 	methods: { | ||||
| 		ok() { | ||||
| 			this.$emit('done', { | ||||
| 				result: this.values | ||||
| 			}); | ||||
| 			this.$refs.dialog.close(); | ||||
| 		}, | ||||
|  | ||||
| 		cancel() { | ||||
| 			this.$emit('done', { | ||||
| 				canceled: true | ||||
| 			}); | ||||
| 			this.$refs.dialog.close(); | ||||
| 		} | ||||
| 	} | ||||
| }); | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" scoped> | ||||
| .xkpnjxcv { | ||||
| 	> label { | ||||
| 		display: block; | ||||
|  | ||||
| 		&:not(:last-child) { | ||||
| 			margin-bottom: 32px; | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| </style> | ||||
| @@ -1,83 +0,0 @@ | ||||
| <template> | ||||
| <x-window ref="window" :width="400" :height="450" :no-padding="true" @closed="() => { $emit('closed'); destroyDom(); }" :with-ok-button="true" :ok-button-disabled="false" @ok="ok()" :can-close="false"> | ||||
| 	<template #header> | ||||
| 		{{ title }} | ||||
| 	</template> | ||||
| 	<div class="xkpnjxcv"> | ||||
| 		<label v-for="item in Object.keys(form).filter(item => !form[item].hidden)" :key="item"> | ||||
| 			<mk-input v-if="form[item].type === 'number'" v-model="values[item]" type="number" :step="form[item].step || 1"> | ||||
| 				<span v-text="form[item].label || item"></span> | ||||
| 				<template v-if="form[item].description" #desc>{{ form[item].description }}</template> | ||||
| 			</mk-input> | ||||
| 			<mk-input v-else-if="form[item].type === 'string' && !item.multiline" v-model="values[item]" type="text"> | ||||
| 				<span v-text="form[item].label || item"></span> | ||||
| 				<template v-if="form[item].description" #desc>{{ form[item].description }}</template> | ||||
| 			</mk-input> | ||||
| 			<mk-textarea v-else-if="form[item].type === 'string' && item.multiline" v-model="values[item]"> | ||||
| 				<span v-text="form[item].label || item"></span> | ||||
| 				<template v-if="form[item].description" #desc>{{ form[item].description }}</template> | ||||
| 			</mk-textarea> | ||||
| 			<mk-switch v-else-if="form[item].type === 'boolean'" v-model="values[item]"> | ||||
| 				<span v-text="form[item].label || item"></span> | ||||
| 				<template v-if="form[item].description" #desc>{{ form[item].description }}</template> | ||||
| 			</mk-switch> | ||||
| 		</label> | ||||
| 	</div> | ||||
| </x-window> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import Vue from 'vue'; | ||||
| import XWindow from './window.vue'; | ||||
| import MkInput from './ui/input.vue'; | ||||
| import MkTextarea from './ui/textarea.vue'; | ||||
| import MkSwitch from './ui/switch.vue'; | ||||
|  | ||||
| export default Vue.extend({ | ||||
| 	components: { | ||||
| 		XWindow, | ||||
| 		MkInput, | ||||
| 		MkTextarea, | ||||
| 		MkSwitch, | ||||
| 	}, | ||||
|  | ||||
| 	props: { | ||||
| 		title: { | ||||
| 			type: String, | ||||
| 			required: true, | ||||
| 		}, | ||||
| 		form: { | ||||
| 			type: Object, | ||||
| 			required: true, | ||||
| 		}, | ||||
| 	}, | ||||
|  | ||||
| 	data() { | ||||
| 		return { | ||||
| 			values: {} | ||||
| 		}; | ||||
| 	}, | ||||
|  | ||||
| 	created() { | ||||
| 		for (const item in this.form) { | ||||
| 			Vue.set(this.values, item, this.form[item].hasOwnProperty('default') ? this.form[item].default : null); | ||||
| 		} | ||||
| 	}, | ||||
|  | ||||
| 	methods: { | ||||
| 		ok() { | ||||
| 			this.$emit('ok', this.values); | ||||
| 			this.$refs.window.close(); | ||||
| 		}, | ||||
| 	} | ||||
| }); | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" scoped> | ||||
| .xkpnjxcv { | ||||
| 	> label { | ||||
| 		display: block; | ||||
| 		padding: 16px 24px; | ||||
| 	} | ||||
| } | ||||
| </style> | ||||
| @@ -5,9 +5,10 @@ | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import Vue from 'vue'; | ||||
| import * as katex from 'katex'; | ||||
| export default Vue.extend({ | ||||
| import { defineComponent } from 'vue'; | ||||
| import * as katex from 'katex';import * as os from '@/os'; | ||||
|  | ||||
| export default defineComponent({ | ||||
| 	props: { | ||||
| 		formula: { | ||||
| 			type: String, | ||||
|   | ||||
| @@ -1,12 +1,13 @@ | ||||
| <template> | ||||
| <x-formula :formula="formula" :block="block" /> | ||||
| <XFormula :formula="formula" :block="block" /> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import Vue from 'vue'; | ||||
| export default Vue.extend({ | ||||
| import { defineComponent, defineAsyncComponent } from 'vue';import * as os from '@/os'; | ||||
|  | ||||
| export default defineComponent({ | ||||
| 	components: { | ||||
| 		XFormula: () => import('./formula-core.vue').then(m => m.default) | ||||
| 		XFormula: defineAsyncComponent(() => import('./formula-core.vue')) | ||||
| 	}, | ||||
| 	props: { | ||||
| 		formula: { | ||||
|   | ||||
| @@ -1,15 +1,16 @@ | ||||
| <template> | ||||
| <div class="mk-google"> | ||||
| 	<input type="search" v-model="query" :placeholder="q"> | ||||
| 	<button @click="search"><fa :icon="faSearch"/> {{ $t('search') }}</button> | ||||
| 	<button @click="search"><Fa :icon="faSearch"/> {{ $t('search') }}</button> | ||||
| </div> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import Vue from 'vue'; | ||||
| import { defineComponent } from 'vue'; | ||||
| import { faSearch } from '@fortawesome/free-solid-svg-icons'; | ||||
| import * as os from '@/os'; | ||||
|  | ||||
| export default Vue.extend({ | ||||
| export default defineComponent({ | ||||
| 	props: ['q'], | ||||
| 	data() { | ||||
| 		return { | ||||
| @@ -23,7 +24,7 @@ export default Vue.extend({ | ||||
| 	methods: { | ||||
| 		search() { | ||||
| 			const engine = this.$store.state.settings.webSearchEngine || | ||||
| 				'https://www.google.com/?#q={{query}}'; | ||||
| 				'https://www.google.com/search?q={{query}}'; | ||||
| 			const url = engine.replace('{{query}}', this.query) | ||||
| 			window.open(url, '_blank'); | ||||
| 		} | ||||
|   | ||||
| @@ -1,101 +0,0 @@ | ||||
| <template> | ||||
| <div class="eqryymyo"> | ||||
| 	<div class="header"> | ||||
| 		<time ref="time" class="_ghost"> | ||||
| 			<span class="yyyymmdd">{{ yyyy }}/{{ mm }}/{{ dd }}</span> | ||||
| 			<br> | ||||
| 			<span class="hhnn">{{ hh }}<span :style="{ visibility: now.getSeconds() % 2 == 0 ? 'visible' : 'hidden' }">:</span>{{ nn }}</span> | ||||
| 		</time> | ||||
| 	</div> | ||||
| 	<div class="content _panel _ghost"> | ||||
| 		<mk-clock/> | ||||
| 	</div> | ||||
| </div> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import Vue from 'vue'; | ||||
| import MkClock from './analog-clock.vue'; | ||||
|  | ||||
| export default Vue.extend({ | ||||
| 	components: { | ||||
| 		MkClock | ||||
| 	}, | ||||
| 	data() { | ||||
| 		return { | ||||
| 			now: new Date(), | ||||
| 			clock: null | ||||
| 		}; | ||||
| 	}, | ||||
| 	computed: { | ||||
| 		yyyy(): number { | ||||
| 			return this.now.getFullYear(); | ||||
| 		}, | ||||
| 		mm(): string { | ||||
| 			return ('0' + (this.now.getMonth() + 1)).slice(-2); | ||||
| 		}, | ||||
| 		dd(): string { | ||||
| 			return ('0' + this.now.getDate()).slice(-2); | ||||
| 		}, | ||||
| 		hh(): string { | ||||
| 			return ('0' + this.now.getHours()).slice(-2); | ||||
| 		}, | ||||
| 		nn(): string { | ||||
| 			return ('0' + this.now.getMinutes()).slice(-2); | ||||
| 		} | ||||
| 	}, | ||||
| 	mounted() { | ||||
| 		this.tick(); | ||||
| 		this.clock = setInterval(this.tick, 1000); | ||||
| 	}, | ||||
| 	beforeDestroy() { | ||||
| 		clearInterval(this.clock); | ||||
| 	}, | ||||
| 	methods: { | ||||
| 		tick() { | ||||
| 			this.now = new Date(); | ||||
| 		} | ||||
| 	} | ||||
| }); | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" scoped> | ||||
| .eqryymyo { | ||||
| 	display: inline-block; | ||||
| 	overflow: visible; | ||||
|  | ||||
| 	> .header { | ||||
| 		padding: 0 12px; | ||||
| 		padding-top: 4px; | ||||
| 		text-align: center; | ||||
| 		font-size: 12px; | ||||
| 		font-family: Lucida Console, Courier, monospace; | ||||
|  | ||||
| 		&:hover + .content { | ||||
| 			opacity: 1; | ||||
| 		} | ||||
|  | ||||
| 		> time { | ||||
| 			display: table-cell; | ||||
| 			vertical-align: middle; | ||||
| 			height: 48px; | ||||
|  | ||||
| 			> .yyyymmdd { | ||||
| 				opacity: 0.7; | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	> .content { | ||||
| 		opacity: 0; | ||||
| 		display: block; | ||||
| 		position: absolute; | ||||
| 		top: auto; | ||||
| 		right: 0; | ||||
| 		margin: 16px 0 0 0; | ||||
| 		padding: 16px; | ||||
| 		width: 230px; | ||||
| 		transition: opacity 0.2s ease; | ||||
| 	} | ||||
| } | ||||
| </style> | ||||
| @@ -1,16 +1,26 @@ | ||||
| <template> | ||||
| <x-modal ref="modal" @closed="() => { $emit('closed'); destroyDom(); }"> | ||||
| 	<img class="xubzgfga" ref="img" :src="image.url" :alt="image.name" :title="image.name" @click="close" tabindex="-1"/> | ||||
| </x-modal> | ||||
| <MkModal ref="modal" @click="$refs.modal.close()" @closed="$emit('closed')"> | ||||
| 	<div class="xubzgfga"> | ||||
| 		<header>{{ image.name }}</header> | ||||
| 		<img :src="image.url" :alt="image.name" :title="image.name" @click="$refs.modal.close()"/> | ||||
| 		<footer> | ||||
| 			<span>{{ image.type }}</span> | ||||
| 			<span>{{ bytes(image.size) }}</span> | ||||
| 			<span v-if="image.properties && image.properties.width">{{ number(image.properties.width) }}px × {{ number(image.properties.height) }}px</span> | ||||
| 		</footer> | ||||
| 	</div> | ||||
| </MkModal> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import Vue from 'vue'; | ||||
| import XModal from './modal.vue'; | ||||
| import { defineComponent } from 'vue'; | ||||
| import bytes from '@/filters/bytes'; | ||||
| import number from '@/filters/number'; | ||||
| import MkModal from '@/components/ui/modal.vue'; | ||||
|  | ||||
| export default Vue.extend({ | ||||
| export default defineComponent({ | ||||
| 	components: { | ||||
| 		XModal, | ||||
| 		MkModal, | ||||
| 	}, | ||||
|  | ||||
| 	props: { | ||||
| @@ -20,32 +30,56 @@ export default Vue.extend({ | ||||
| 		}, | ||||
| 	}, | ||||
|  | ||||
| 	mounted() { | ||||
| 		this.$nextTick(() => { | ||||
| 			this.$refs.img.focus(); | ||||
| 		}); | ||||
| 	}, | ||||
| 	emits: ['closed'], | ||||
|  | ||||
| 	methods: { | ||||
| 		close() { | ||||
| 			this.$refs.modal.close(); | ||||
| 		}, | ||||
| 		bytes, | ||||
| 		number, | ||||
| 	} | ||||
| }); | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" scoped> | ||||
| .xubzgfga { | ||||
| 	position: fixed; | ||||
| 	z-index: 2; | ||||
| 	top: 0; | ||||
| 	right: 0; | ||||
| 	bottom: 0; | ||||
| 	left: 0; | ||||
| 	max-width: 100%; | ||||
| 	max-height: 100%; | ||||
| 	margin: auto; | ||||
| 	cursor: zoom-out; | ||||
| 	image-orientation: from-image; | ||||
| 	display: flex; | ||||
| 	flex-direction: column; | ||||
| 	height: 100%; | ||||
|  | ||||
| 	> header, | ||||
| 	> footer { | ||||
| 		align-self: center; | ||||
| 		display: inline-block; | ||||
| 		padding: 6px 9px; | ||||
| 		font-size: 90%; | ||||
| 		background: rgba(0, 0, 0, 0.5); | ||||
| 		border-radius: 6px; | ||||
| 		color: #fff; | ||||
| 	} | ||||
|  | ||||
| 	> header { | ||||
| 		margin-bottom: 8px; | ||||
| 		opacity: 0.9; | ||||
| 	} | ||||
|  | ||||
| 	> img { | ||||
| 		display: block; | ||||
| 		flex: 1; | ||||
| 		min-height: 0; | ||||
| 		object-fit: contain; | ||||
| 		width: 100%; | ||||
| 		cursor: zoom-out; | ||||
| 		image-orientation: from-image; | ||||
| 	} | ||||
|  | ||||
| 	> footer { | ||||
| 		margin-top: 8px; | ||||
| 		opacity: 0.8; | ||||
|  | ||||
| 		> span + span { | ||||
| 			margin-left: 0.5em; | ||||
| 			padding-left: 0.5em; | ||||
| 			border-left: solid 1px rgba(255, 255, 255, 0.5); | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| </style> | ||||
|   | ||||
| @@ -1,15 +1,15 @@ | ||||
| <template> | ||||
| <div class="xubzgfgb" :title="title"> | ||||
| <div class="xubzgfgb" :class="{ cover }" :title="title"> | ||||
| 	<canvas ref="canvas" :width="size" :height="size" :title="title" v-if="!loaded"/> | ||||
| 	<img v-if="src" :src="src" :title="title" :alt="alt" @load="onLoad"/> | ||||
| </div> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import Vue from 'vue'; | ||||
| import { defineComponent } from 'vue'; | ||||
| import { decode } from 'blurhash'; | ||||
|  | ||||
| export default Vue.extend({ | ||||
| export default defineComponent({ | ||||
| 	props: { | ||||
| 		src: { | ||||
| 			type: String, | ||||
| @@ -35,6 +35,11 @@ export default Vue.extend({ | ||||
| 			required: false, | ||||
| 			default: 64 | ||||
| 		}, | ||||
| 		cover: { | ||||
| 			type: Boolean, | ||||
| 			required: false, | ||||
| 			default: true, | ||||
| 		} | ||||
| 	}, | ||||
|  | ||||
| 	data() { | ||||
| @@ -49,6 +54,7 @@ export default Vue.extend({ | ||||
|  | ||||
| 	methods: { | ||||
| 		draw() { | ||||
| 			if (this.hash == null) return; | ||||
| 			const pixels = decode(this.hash, this.size, this.size); | ||||
| 			const ctx = (this.$refs.canvas as HTMLCanvasElement).getContext('2d'); | ||||
| 			const imageData = ctx!.createImageData(this.size, this.size); | ||||
| @@ -70,9 +76,23 @@ export default Vue.extend({ | ||||
|  | ||||
| 	> canvas, | ||||
| 	> img { | ||||
| 		display: block; | ||||
| 		width: 100%; | ||||
| 		height: 100%; | ||||
| 	} | ||||
|  | ||||
| 	> canvas { | ||||
| 		object-fit: cover; | ||||
| 	} | ||||
|  | ||||
| 	> img { | ||||
| 		object-fit: contain; | ||||
| 	} | ||||
|  | ||||
| 	&.cover { | ||||
| 		> img { | ||||
| 			object-fit: cover; | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| </style> | ||||
|   | ||||
| @@ -1,6 +1,7 @@ | ||||
| import Vue from 'vue'; | ||||
| import { App } from 'vue'; | ||||
|  | ||||
| import mfm from './misskey-flavored-markdown.vue'; | ||||
| import a from './ui/a.vue'; | ||||
| import acct from './acct.vue'; | ||||
| import avatar from './avatar.vue'; | ||||
| import emoji from './emoji.vue'; | ||||
| @@ -10,16 +11,17 @@ import time from './time.vue'; | ||||
| import url from './url.vue'; | ||||
| import loading from './loading.vue'; | ||||
| import error from './error.vue'; | ||||
| import streamIndicator from './stream-indicator.vue'; | ||||
|  | ||||
| Vue.component('mfm', mfm); | ||||
| Vue.component('mk-acct', acct); | ||||
| Vue.component('mk-avatar', avatar); | ||||
| Vue.component('mk-emoji', emoji); | ||||
| Vue.component('mk-user-name', userName); | ||||
| Vue.component('mk-ellipsis', ellipsis); | ||||
| Vue.component('mk-time', time); | ||||
| Vue.component('mk-url', url); | ||||
| Vue.component('mk-loading', loading); | ||||
| Vue.component('mk-error', error); | ||||
| Vue.component('stream-indicator', streamIndicator); | ||||
| export default function(app: App) { | ||||
| 	app.component('Mfm', mfm); | ||||
| 	app.component('MkA', a); | ||||
| 	app.component('MkAcct', acct); | ||||
| 	app.component('MkAvatar', avatar); | ||||
| 	app.component('MkEmoji', emoji); | ||||
| 	app.component('MkUserName', userName); | ||||
| 	app.component('MkEllipsis', ellipsis); | ||||
| 	app.component('MkTime', time); | ||||
| 	app.component('MkUrl', url); | ||||
| 	app.component('MkLoading', loading); | ||||
| 	app.component('MkError', error); | ||||
| } | ||||
|   | ||||
| @@ -1,93 +1,93 @@ | ||||
| <template> | ||||
| <div class="zbcjwnqg"> | ||||
| <div class="zbcjwnqg" v-size="{ max: [550, 1000] }"> | ||||
| 	<div class="stats" v-if="info"> | ||||
| 		<div class="_panel"> | ||||
| 			<div> | ||||
| 				<b><fa :icon="faUser"/>{{ $t('users') }}</b> | ||||
| 				<b><Fa :icon="faUser"/>{{ $t('users') }}</b> | ||||
| 				<small>{{ $t('local') }}</small> | ||||
| 			</div> | ||||
| 			<div> | ||||
| 				<dl class="total"> | ||||
| 					<dt>{{ $t('total') }}</dt> | ||||
| 					<dd>{{ info.originalUsersCount | number }}</dd> | ||||
| 					<dd>{{ number(info.originalUsersCount) }}</dd> | ||||
| 				</dl> | ||||
| 				<dl class="diff" :class="{ inc: usersLocalDoD > 0 }"> | ||||
| 					<dt>{{ $t('dayOverDayChanges') }}</dt> | ||||
| 					<dd>{{ usersLocalDoD | number }}</dd> | ||||
| 					<dd>{{ number(usersLocalDoD) }}</dd> | ||||
| 				</dl> | ||||
| 				<dl class="diff" :class="{ inc: usersLocalWoW > 0 }"> | ||||
| 					<dt>{{ $t('weekOverWeekChanges') }}</dt> | ||||
| 					<dd>{{ usersLocalWoW | number }}</dd> | ||||
| 					<dd>{{ number(usersLocalWoW) }}</dd> | ||||
| 				</dl> | ||||
| 			</div> | ||||
| 		</div> | ||||
| 		<div class="_panel"> | ||||
| 			<div> | ||||
| 				<b><fa :icon="faUser"/>{{ $t('users') }}</b> | ||||
| 				<b><Fa :icon="faUser"/>{{ $t('users') }}</b> | ||||
| 				<small>{{ $t('remote') }}</small> | ||||
| 			</div> | ||||
| 			<div> | ||||
| 				<dl class="total"> | ||||
| 					<dt>{{ $t('total') }}</dt> | ||||
| 					<dd>{{ (info.usersCount - info.originalUsersCount) | number }}</dd> | ||||
| 					<dd>{{ number((info.usersCount - info.originalUsersCount)) }}</dd> | ||||
| 				</dl> | ||||
| 				<dl class="diff" :class="{ inc: usersRemoteDoD > 0 }"> | ||||
| 					<dt>{{ $t('dayOverDayChanges') }}</dt> | ||||
| 					<dd>{{ usersRemoteDoD | number }}</dd> | ||||
| 					<dd>{{ number(usersRemoteDoD) }}</dd> | ||||
| 				</dl> | ||||
| 				<dl class="diff" :class="{ inc: usersRemoteWoW > 0 }"> | ||||
| 					<dt>{{ $t('weekOverWeekChanges') }}</dt> | ||||
| 					<dd>{{ usersRemoteWoW | number }}</dd> | ||||
| 					<dd>{{ number(usersRemoteWoW) }}</dd> | ||||
| 				</dl> | ||||
| 			</div> | ||||
| 		</div> | ||||
| 		<div class="_panel"> | ||||
| 			<div> | ||||
| 				<b><fa :icon="faPencilAlt"/>{{ $t('notes') }}</b> | ||||
| 				<b><Fa :icon="faPencilAlt"/>{{ $t('notes') }}</b> | ||||
| 				<small>{{ $t('local') }}</small> | ||||
| 			</div> | ||||
| 			<div> | ||||
| 				<dl class="total"> | ||||
| 					<dt>{{ $t('total') }}</dt> | ||||
| 					<dd>{{ info.originalNotesCount | number }}</dd> | ||||
| 					<dd>{{ number(info.originalNotesCount) }}</dd> | ||||
| 				</dl> | ||||
| 				<dl class="diff" :class="{ inc: notesLocalDoD > 0 }"> | ||||
| 					<dt>{{ $t('dayOverDayChanges') }}</dt> | ||||
| 					<dd>{{ notesLocalDoD | number }}</dd> | ||||
| 					<dd>{{ number(notesLocalDoD) }}</dd> | ||||
| 				</dl> | ||||
| 				<dl class="diff" :class="{ inc: notesLocalWoW > 0 }"> | ||||
| 					<dt>{{ $t('weekOverWeekChanges') }}</dt> | ||||
| 					<dd>{{ notesLocalWoW | number }}</dd> | ||||
| 					<dd>{{ number(notesLocalWoW) }}</dd> | ||||
| 				</dl> | ||||
| 			</div> | ||||
| 		</div> | ||||
| 		<div class="_panel"> | ||||
| 			<div> | ||||
| 				<b><fa :icon="faPencilAlt"/>{{ $t('notes') }}</b> | ||||
| 				<b><Fa :icon="faPencilAlt"/>{{ $t('notes') }}</b> | ||||
| 				<small>{{ $t('remote') }}</small> | ||||
| 			</div> | ||||
| 			<div> | ||||
| 				<dl class="total"> | ||||
| 					<dt>{{ $t('total') }}</dt> | ||||
| 					<dd>{{ (info.notesCount - info.originalNotesCount) | number }}</dd> | ||||
| 					<dd>{{ number((info.notesCount - info.originalNotesCount)) }}</dd> | ||||
| 				</dl> | ||||
| 				<dl class="diff" :class="{ inc: notesRemoteDoD > 0 }"> | ||||
| 					<dt>{{ $t('dayOverDayChanges') }}</dt> | ||||
| 					<dd>{{ notesRemoteDoD | number }}</dd> | ||||
| 					<dd>{{ number(notesRemoteDoD) }}</dd> | ||||
| 				</dl> | ||||
| 				<dl class="diff" :class="{ inc: notesRemoteWoW > 0 }"> | ||||
| 					<dt>{{ $t('weekOverWeekChanges') }}</dt> | ||||
| 					<dd>{{ notesRemoteWoW | number }}</dd> | ||||
| 					<dd>{{ number(notesRemoteWoW) }}</dd> | ||||
| 				</dl> | ||||
| 			</div> | ||||
| 		</div> | ||||
| 	</div> | ||||
|  | ||||
| 	<section class="_card"> | ||||
| 		<div class="_title"><fa :icon="faChartBar"/> {{ $t('statistics') }}</div> | ||||
| 		<div class="_title" style="position: relative;"><Fa :icon="faChartBar"/> {{ $t('statistics') }}<button @click="fetchChart" class="_button" style="position: absolute; right: 0; bottom: 0; top: 0; padding: inherit;"><Fa :icon="faSync"/></button></div> | ||||
| 		<div class="_content" style="margin-top: -8px;"> | ||||
| 			<div class="selects" style="display: flex;"> | ||||
| 				<mk-select v-model="chartSrc" style="margin: 0; flex: 1;"> | ||||
| 				<MkSelect v-model:value="chartSrc" style="margin: 0; flex: 1;"> | ||||
| 					<optgroup :label="$t('federation')"> | ||||
| 						<option value="federation-instances">{{ $t('_charts.federationInstancesIncDec') }}</option> | ||||
| 						<option value="federation-instances-total">{{ $t('_charts.federationInstancesTotal') }}</option> | ||||
| @@ -109,11 +109,11 @@ | ||||
| 						<option value="drive">{{ $t('_charts.storageUsageIncDec') }}</option> | ||||
| 						<option value="drive-total">{{ $t('_charts.storageUsageTotal') }}</option> | ||||
| 					</optgroup> | ||||
| 				</mk-select> | ||||
| 				<mk-select v-model="chartSpan" style="margin: 0;"> | ||||
| 				</MkSelect> | ||||
| 				<MkSelect v-model:value="chartSpan" style="margin: 0;"> | ||||
| 					<option value="hour">{{ $t('perHour') }}</option> | ||||
| 					<option value="day">{{ $t('perDay') }}</option> | ||||
| 				</mk-select> | ||||
| 				</MkSelect> | ||||
| 			</div> | ||||
| 			<canvas ref="chart"></canvas> | ||||
| 		</div> | ||||
| @@ -122,12 +122,12 @@ | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import Vue from 'vue'; | ||||
| import { faChartBar, faUser, faPencilAlt } from '@fortawesome/free-solid-svg-icons'; | ||||
| import { defineComponent, markRaw } from 'vue'; | ||||
| import { faChartBar, faUser, faPencilAlt, faSync } from '@fortawesome/free-solid-svg-icons'; | ||||
| import Chart from 'chart.js'; | ||||
| import MkSelect from './ui/select.vue'; | ||||
| import number from '@/filters/number'; | ||||
|  | ||||
| const chartLimit = 90; | ||||
| const sum = (...arr) => arr.reduce((r, a) => r.map((b, i) => a[i] + b)); | ||||
| const negate = arr => arr.map(x => -x); | ||||
| const alpha = (hex, a) => { | ||||
| @@ -137,12 +137,26 @@ const alpha = (hex, a) => { | ||||
| 	const b = parseInt(result[3], 16); | ||||
| 	return `rgba(${r}, ${g}, ${b}, ${a})`; | ||||
| }; | ||||
| import * as os from '@/os'; | ||||
|  | ||||
| export default Vue.extend({ | ||||
| export default defineComponent({ | ||||
| 	components: { | ||||
| 		MkSelect | ||||
| 	}, | ||||
|  | ||||
| 	props: { | ||||
| 		chartLimit: { | ||||
| 			type: Number, | ||||
| 			required: false, | ||||
| 			default: 90 | ||||
| 		}, | ||||
| 		detailed: { | ||||
| 			type: Boolean, | ||||
| 			required: false, | ||||
| 			default: false | ||||
| 		}, | ||||
| 	}, | ||||
|  | ||||
| 	data() { | ||||
| 		return { | ||||
| 			info: null, | ||||
| @@ -159,7 +173,7 @@ export default Vue.extend({ | ||||
| 			chartInstance: null, | ||||
| 			chartSrc: 'notes', | ||||
| 			chartSpan: 'hour', | ||||
| 			faChartBar, faUser, faPencilAlt | ||||
| 			faChartBar, faUser, faPencilAlt, faSync | ||||
| 		} | ||||
| 	}, | ||||
|  | ||||
| @@ -204,66 +218,73 @@ export default Vue.extend({ | ||||
| 	}, | ||||
|  | ||||
| 	async created() { | ||||
| 		this.info = await this.$root.api('stats'); | ||||
| 		this.info = await os.api('stats'); | ||||
|  | ||||
| 		this.now = new Date(); | ||||
|  | ||||
| 		const [perHour, perDay] = await Promise.all([Promise.all([ | ||||
| 			this.$root.api('charts/federation', { limit: chartLimit, span: 'hour' }), | ||||
| 			this.$root.api('charts/users', { limit: chartLimit, span: 'hour' }), | ||||
| 			this.$root.api('charts/active-users', { limit: chartLimit, span: 'hour' }), | ||||
| 			this.$root.api('charts/notes', { limit: chartLimit, span: 'hour' }), | ||||
| 			this.$root.api('charts/drive', { limit: chartLimit, span: 'hour' }), | ||||
| 		]), Promise.all([ | ||||
| 			this.$root.api('charts/federation', { limit: chartLimit, span: 'day' }), | ||||
| 			this.$root.api('charts/users', { limit: chartLimit, span: 'day' }), | ||||
| 			this.$root.api('charts/active-users', { limit: chartLimit, span: 'day' }), | ||||
| 			this.$root.api('charts/notes', { limit: chartLimit, span: 'day' }), | ||||
| 			this.$root.api('charts/drive', { limit: chartLimit, span: 'day' }), | ||||
| 		])]); | ||||
|  | ||||
| 		const chart = { | ||||
| 			perHour: { | ||||
| 				federation: perHour[0], | ||||
| 				users: perHour[1], | ||||
| 				activeUsers: perHour[2], | ||||
| 				notes: perHour[3], | ||||
| 				drive: perHour[4], | ||||
| 			}, | ||||
| 			perDay: { | ||||
| 				federation: perDay[0], | ||||
| 				users: perDay[1], | ||||
| 				activeUsers: perDay[2], | ||||
| 				notes: perDay[3], | ||||
| 				drive: perDay[4], | ||||
| 			} | ||||
| 		}; | ||||
|  | ||||
| 		this.notesLocalWoW = this.info.originalNotesCount - chart.perDay.notes.local.total[7]; | ||||
| 		this.notesLocalDoD = this.info.originalNotesCount - chart.perDay.notes.local.total[1]; | ||||
| 		this.notesRemoteWoW = (this.info.notesCount - this.info.originalNotesCount) - chart.perDay.notes.remote.total[7]; | ||||
| 		this.notesRemoteDoD = (this.info.notesCount - this.info.originalNotesCount) - chart.perDay.notes.remote.total[1]; | ||||
| 		this.usersLocalWoW = this.info.originalUsersCount - chart.perDay.users.local.total[7]; | ||||
| 		this.usersLocalDoD = this.info.originalUsersCount - chart.perDay.users.local.total[1]; | ||||
| 		this.usersRemoteWoW = (this.info.usersCount - this.info.originalUsersCount) - chart.perDay.users.remote.total[7]; | ||||
| 		this.usersRemoteDoD = (this.info.usersCount - this.info.originalUsersCount) - chart.perDay.users.remote.total[1]; | ||||
|  | ||||
| 		this.chart = chart; | ||||
|  | ||||
| 		this.renderChart(); | ||||
| 		this.fetchChart(); | ||||
| 	}, | ||||
|  | ||||
| 	methods: { | ||||
| 		async fetchChart() { | ||||
| 			const [perHour, perDay] = await Promise.all([Promise.all([ | ||||
| 				os.api('charts/federation', { limit: this.chartLimit, span: 'hour' }), | ||||
| 				os.api('charts/users', { limit: this.chartLimit, span: 'hour' }), | ||||
| 				os.api('charts/active-users', { limit: this.chartLimit, span: 'hour' }), | ||||
| 				os.api('charts/notes', { limit: this.chartLimit, span: 'hour' }), | ||||
| 				os.api('charts/drive', { limit: this.chartLimit, span: 'hour' }), | ||||
| 			]), Promise.all([ | ||||
| 				os.api('charts/federation', { limit: this.chartLimit, span: 'day' }), | ||||
| 				os.api('charts/users', { limit: this.chartLimit, span: 'day' }), | ||||
| 				os.api('charts/active-users', { limit: this.chartLimit, span: 'day' }), | ||||
| 				os.api('charts/notes', { limit: this.chartLimit, span: 'day' }), | ||||
| 				os.api('charts/drive', { limit: this.chartLimit, span: 'day' }), | ||||
| 			])]); | ||||
|  | ||||
| 			const chart = { | ||||
| 				perHour: { | ||||
| 					federation: perHour[0], | ||||
| 					users: perHour[1], | ||||
| 					activeUsers: perHour[2], | ||||
| 					notes: perHour[3], | ||||
| 					drive: perHour[4], | ||||
| 				}, | ||||
| 				perDay: { | ||||
| 					federation: perDay[0], | ||||
| 					users: perDay[1], | ||||
| 					activeUsers: perDay[2], | ||||
| 					notes: perDay[3], | ||||
| 					drive: perDay[4], | ||||
| 				} | ||||
| 			}; | ||||
|  | ||||
| 			this.notesLocalWoW = this.info.originalNotesCount - chart.perDay.notes.local.total[7]; | ||||
| 			this.notesLocalDoD = this.info.originalNotesCount - chart.perDay.notes.local.total[1]; | ||||
| 			this.notesRemoteWoW = (this.info.notesCount - this.info.originalNotesCount) - chart.perDay.notes.remote.total[7]; | ||||
| 			this.notesRemoteDoD = (this.info.notesCount - this.info.originalNotesCount) - chart.perDay.notes.remote.total[1]; | ||||
| 			this.usersLocalWoW = this.info.originalUsersCount - chart.perDay.users.local.total[7]; | ||||
| 			this.usersLocalDoD = this.info.originalUsersCount - chart.perDay.users.local.total[1]; | ||||
| 			this.usersRemoteWoW = (this.info.usersCount - this.info.originalUsersCount) - chart.perDay.users.remote.total[7]; | ||||
| 			this.usersRemoteDoD = (this.info.usersCount - this.info.originalUsersCount) - chart.perDay.users.remote.total[1]; | ||||
|  | ||||
| 			this.chart = chart; | ||||
|  | ||||
| 			this.renderChart(); | ||||
| 		}, | ||||
|  | ||||
| 		renderChart() { | ||||
| 			if (this.chartInstance) { | ||||
| 				this.chartInstance.destroy(); | ||||
| 			} | ||||
|  | ||||
| 			// TODO: var(--panel)の色が暗いか明るいかで判定する | ||||
| 			const gridColor = this.$store.state.device.darkMode ? 'rgba(255, 255, 255, 0.1)' : 'rgba(0, 0, 0, 0.1)'; | ||||
|  | ||||
| 			Chart.defaults.global.defaultFontColor = getComputedStyle(document.documentElement).getPropertyValue('--fg'); | ||||
| 			this.chartInstance = new Chart(this.$refs.chart, { | ||||
| 			this.chartInstance = markRaw(new Chart(this.$refs.chart, { | ||||
| 				type: 'line', | ||||
| 				data: { | ||||
| 					labels: new Array(chartLimit).fill(0).map((_, i) => this.getDate(i).toLocaleString()).slice().reverse(), | ||||
| 					labels: new Array(this.chartLimit).fill(0).map((_, i) => this.getDate(i).toLocaleString()).slice().reverse(), | ||||
| 					datasets: this.data.series.map(x => ({ | ||||
| 						label: x.name, | ||||
| 						data: x.data.slice().reverse(), | ||||
| @@ -271,7 +292,9 @@ export default Vue.extend({ | ||||
| 						lineTension: 0, | ||||
| 						borderWidth: 2, | ||||
| 						borderColor: x.color, | ||||
| 						borderDash: x.borderDash || [], | ||||
| 						backgroundColor: alpha(x.color, 0.1), | ||||
| 						fill: x.fill == null ? true : x.fill, | ||||
| 						hidden: !!x.hidden | ||||
| 					})) | ||||
| 				}, | ||||
| @@ -293,17 +316,28 @@ export default Vue.extend({ | ||||
| 					}, | ||||
| 					scales: { | ||||
| 						xAxes: [{ | ||||
| 							type: 'time', | ||||
| 							time: { | ||||
| 								stepSize: 1, | ||||
| 								unit: this.chartSpan == 'day' ? 'month' : 'day', | ||||
| 							}, | ||||
| 							gridLines: { | ||||
| 								display: false | ||||
| 								display: this.detailed, | ||||
| 								color: gridColor, | ||||
| 								zeroLineColor: gridColor, | ||||
| 							}, | ||||
| 							ticks: { | ||||
| 								display: false | ||||
| 								display: this.detailed | ||||
| 							} | ||||
| 						}], | ||||
| 						yAxes: [{ | ||||
| 							position: 'right', | ||||
| 							position: 'left', | ||||
| 							gridLines: { | ||||
| 								color: gridColor, | ||||
| 								zeroLineColor: gridColor, | ||||
| 							}, | ||||
| 							ticks: { | ||||
| 								display: false | ||||
| 								display: this.detailed | ||||
| 							} | ||||
| 						}] | ||||
| 					}, | ||||
| @@ -312,7 +346,7 @@ export default Vue.extend({ | ||||
| 						mode: 'index', | ||||
| 					} | ||||
| 				} | ||||
| 			}); | ||||
| 			})); | ||||
| 		}, | ||||
|  | ||||
| 		getDate(ago: number) { | ||||
| @@ -325,7 +359,11 @@ export default Vue.extend({ | ||||
| 		}, | ||||
|  | ||||
| 		format(arr) { | ||||
| 			return arr; | ||||
| 			const now = Date.now(); | ||||
| 			return arr.map((v, i) => ({ | ||||
| 				x: new Date(now - ((this.chartSpan == 'day' ? 86400000 :3600000 ) * i)), | ||||
| 				y: v | ||||
| 			})); | ||||
| 		}, | ||||
|  | ||||
| 		federationInstancesChart(total: boolean): any { | ||||
| @@ -347,6 +385,8 @@ export default Vue.extend({ | ||||
| 					name: 'All', | ||||
| 					type: 'line', | ||||
| 					color: '#008FFB', | ||||
| 					borderDash: [5, 5], | ||||
| 					fill: false, | ||||
| 					data: this.format(type == 'combined' | ||||
| 						? sum(this.stats.notes.local.inc, negate(this.stats.notes.local.dec), this.stats.notes.remote.inc, negate(this.stats.notes.remote.dec)) | ||||
| 						: sum(this.stats.notes[type].inc, negate(this.stats.notes[type].dec)) | ||||
| @@ -463,7 +503,9 @@ export default Vue.extend({ | ||||
| 				series: [{ | ||||
| 					name: 'All', | ||||
| 					type: 'line', | ||||
| 					color: '#008FFB', | ||||
| 					color: '#09d8e2', | ||||
| 					borderDash: [5, 5], | ||||
| 					fill: false, | ||||
| 					data: this.format( | ||||
| 						sum( | ||||
| 							this.stats.drive.local.incSize, | ||||
| @@ -480,17 +522,17 @@ export default Vue.extend({ | ||||
| 				}, { | ||||
| 					name: 'Local -', | ||||
| 					type: 'area', | ||||
| 					color: '#008FFB', | ||||
| 					color: '#FF4560', | ||||
| 					data: this.format(negate(this.stats.drive.local.decSize)) | ||||
| 				}, { | ||||
| 					name: 'Remote +', | ||||
| 					type: 'area', | ||||
| 					color: '#008FFB', | ||||
| 					color: '#00E396', | ||||
| 					data: this.format(this.stats.drive.remote.incSize) | ||||
| 				}, { | ||||
| 					name: 'Remote -', | ||||
| 					type: 'area', | ||||
| 					color: '#008FFB', | ||||
| 					color: '#FEB019', | ||||
| 					data: this.format(negate(this.stats.drive.remote.decSize)) | ||||
| 				}] | ||||
| 			}; | ||||
| @@ -525,7 +567,9 @@ export default Vue.extend({ | ||||
| 				series: [{ | ||||
| 					name: 'All', | ||||
| 					type: 'line', | ||||
| 					color: '#008FFB', | ||||
| 					color: '#09d8e2', | ||||
| 					borderDash: [5, 5], | ||||
| 					fill: false, | ||||
| 					data: this.format( | ||||
| 						sum( | ||||
| 							this.stats.drive.local.incCount, | ||||
| @@ -542,17 +586,17 @@ export default Vue.extend({ | ||||
| 				}, { | ||||
| 					name: 'Local -', | ||||
| 					type: 'area', | ||||
| 					color: '#008FFB', | ||||
| 					color: '#FF4560', | ||||
| 					data: this.format(negate(this.stats.drive.local.decCount)) | ||||
| 				}, { | ||||
| 					name: 'Remote +', | ||||
| 					type: 'area', | ||||
| 					color: '#008FFB', | ||||
| 					color: '#00E396', | ||||
| 					data: this.format(this.stats.drive.remote.incCount) | ||||
| 				}, { | ||||
| 					name: 'Remote -', | ||||
| 					type: 'area', | ||||
| 					color: '#008FFB', | ||||
| 					color: '#FEB019', | ||||
| 					data: this.format(negate(this.stats.drive.remote.decCount)) | ||||
| 				}] | ||||
| 			}; | ||||
| @@ -580,23 +624,38 @@ export default Vue.extend({ | ||||
| 				}] | ||||
| 			}; | ||||
| 		}, | ||||
|  | ||||
| 		number | ||||
| 	} | ||||
| }); | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" scoped> | ||||
| .zbcjwnqg { | ||||
| 	&.max-width_1000px { | ||||
| 		> .stats { | ||||
| 			grid-template-columns: 1fr 1fr; | ||||
| 			grid-template-rows: 1fr 1fr; | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	&.max-width_550px { | ||||
| 		> .stats { | ||||
| 			grid-template-columns: 1fr; | ||||
| 			grid-template-rows: 1fr 1fr 1fr 1fr; | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	> .stats { | ||||
| 		display: flex; | ||||
| 		justify-content: space-between; | ||||
| 		flex-wrap: wrap; | ||||
| 		margin: calc(0px - var(--margin) / 2); | ||||
| 		margin-bottom: calc(var(--margin) / 2); | ||||
| 		display: grid; | ||||
| 		grid-template-columns: 1fr 1fr 1fr 1fr; | ||||
| 		grid-template-rows: 1fr; | ||||
| 		gap: var(--margin); | ||||
| 		margin-bottom: var(--margin); | ||||
| 		font-size: 90%; | ||||
|  | ||||
| 		> div { | ||||
| 			display: flex; | ||||
| 			flex: 1 0 213px; | ||||
| 			margin: calc(var(--margin) / 2); | ||||
| 			box-sizing: border-box; | ||||
| 			padding: 16px 20px; | ||||
|  | ||||
| @@ -631,7 +690,7 @@ export default Vue.extend({ | ||||
| 							margin: 0; | ||||
| 						} | ||||
|  | ||||
| 						> dt { | ||||
| 						> dd { | ||||
| 							text-overflow: ellipsis; | ||||
| 							overflow: hidden; | ||||
| 							white-space: nowrap; | ||||
|   | ||||
							
								
								
									
										62
									
								
								src/client/components/instance-ticker.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										62
									
								
								src/client/components/instance-ticker.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,62 @@ | ||||
| <template> | ||||
| <div class="hpaizdrt" :style="bg"> | ||||
| 	<img v-if="info.faviconUrl" class="icon" :src="info.faviconUrl"/> | ||||
| 	<span class="name">{{ info.name }}</span> | ||||
| </div> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import { defineComponent } from 'vue'; | ||||
| import { instanceName } from '@/config'; | ||||
|  | ||||
| export default defineComponent({ | ||||
| 	props: { | ||||
| 		instance: { | ||||
| 			type: Object, | ||||
| 			required: false | ||||
| 		}, | ||||
| 	}, | ||||
|  | ||||
| 	data() { | ||||
| 		return { | ||||
| 			info: this.instance || { | ||||
| 				faviconUrl: '/favicon.ico', | ||||
| 				name: instanceName, | ||||
| 				themeColor: (document.querySelector('meta[name="theme-color-orig"]') as HTMLMetaElement)?.content | ||||
| 			} | ||||
| 		} | ||||
| 	}, | ||||
|  | ||||
| 	computed: { | ||||
| 		bg(): any { | ||||
| 			const themeColor = this.info.themeColor || '#777777'; | ||||
| 			return { | ||||
| 				background: `linear-gradient(90deg, ${themeColor}, ${themeColor + '00'})` | ||||
| 			}; | ||||
| 		} | ||||
| 	} | ||||
| }); | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" scoped> | ||||
| .hpaizdrt { | ||||
| 	$height: 1.1rem; | ||||
|  | ||||
| 	height: $height; | ||||
| 	border-radius: 4px 0 0 4px; | ||||
| 	overflow: hidden; | ||||
| 	color: #fff; | ||||
|  | ||||
| 	> .icon { | ||||
| 		height: 100%; | ||||
| 	} | ||||
|  | ||||
| 	> .name { | ||||
| 		margin-left: 4px; | ||||
| 		line-height: $height; | ||||
| 		font-size: 0.9em; | ||||
| 		vertical-align: top; | ||||
| 		font-weight: bold; | ||||
| 	} | ||||
| } | ||||
| </style> | ||||
| @@ -1,22 +1,22 @@ | ||||
| <template> | ||||
| <component :is="self ? 'router-link' : 'a'" class="xlcxczvw _link" :[attr]="self ? url.substr(local.length) : url" :rel="rel" :target="target" | ||||
| <component :is="self ? 'MkA' : 'a'" class="xlcxczvw _link" :[attr]="self ? url.substr(local.length) : url" :rel="rel" :target="target" | ||||
| 	@mouseover="onMouseover" | ||||
| 	@mouseleave="onMouseleave" | ||||
| 	:title="url" | ||||
| > | ||||
| 	<slot></slot> | ||||
| 	<fa :icon="faExternalLinkSquareAlt" v-if="target === '_blank'" class="icon"/> | ||||
| 	<Fa :icon="faExternalLinkSquareAlt" v-if="target === '_blank'" class="icon"/> | ||||
| </component> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import Vue from 'vue'; | ||||
| import { defineComponent } from 'vue'; | ||||
| import { faExternalLinkSquareAlt } from '@fortawesome/free-solid-svg-icons'; | ||||
| import { url as local } from '../config'; | ||||
| import MkUrlPreview from './url-preview-popup.vue'; | ||||
| import { isDeviceTouch } from '../scripts/is-device-touch'; | ||||
| import { url as local } from '@/config'; | ||||
| import { isDeviceTouch } from '@/scripts/is-device-touch'; | ||||
| import * as os from '@/os'; | ||||
|  | ||||
| export default Vue.extend({ | ||||
| export default defineComponent({ | ||||
| 	props: { | ||||
| 		url: { | ||||
| 			type: String, | ||||
| @@ -36,29 +36,34 @@ export default Vue.extend({ | ||||
| 			target: self ? null : '_blank', | ||||
| 			showTimer: null, | ||||
| 			hideTimer: null, | ||||
| 			preview: null, | ||||
| 			checkTimer: null, | ||||
| 			close: null, | ||||
| 			faExternalLinkSquareAlt | ||||
| 		}; | ||||
| 	}, | ||||
| 	methods: { | ||||
| 		showPreview() { | ||||
| 		async showPreview() { | ||||
| 			if (!document.body.contains(this.$el)) return; | ||||
| 			if (this.preview) return; | ||||
| 			if (this.close) return; | ||||
|  | ||||
| 			this.preview = new MkUrlPreview({ | ||||
| 				parent: this, | ||||
| 				propsData: { | ||||
| 					url: this.url, | ||||
| 					source: this.$el | ||||
| 				} | ||||
| 			}).$mount(); | ||||
| 			const { dispose } = await os.popup(import('@/components/url-preview-popup.vue'), { | ||||
| 				url: this.url, | ||||
| 				source: this.$el | ||||
| 			}); | ||||
|  | ||||
| 			document.body.appendChild(this.preview.$el); | ||||
| 			this.close = () => { | ||||
| 				dispose(); | ||||
| 			}; | ||||
|  | ||||
| 			this.checkTimer = setInterval(() => { | ||||
| 				if (!document.body.contains(this.$el)) this.closePreview(); | ||||
| 			}, 1000); | ||||
| 		}, | ||||
| 		closePreview() { | ||||
| 			if (this.preview) { | ||||
| 				this.preview.destroyDom(); | ||||
| 				this.preview = null; | ||||
| 			if (this.close) { | ||||
| 				clearInterval(this.checkTimer); | ||||
| 				this.close(); | ||||
| 				this.close = null; | ||||
| 			} | ||||
| 		}, | ||||
| 		onMouseover() { | ||||
|   | ||||
| @@ -5,9 +5,10 @@ | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import Vue from 'vue'; | ||||
| import { defineComponent } from 'vue'; | ||||
| import * as os from '@/os'; | ||||
|  | ||||
| export default Vue.extend({ | ||||
| export default defineComponent({ | ||||
| 	props: { | ||||
| 		inline: { | ||||
| 			type: Boolean, | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| <template> | ||||
| <div class="mk-media-banner"> | ||||
| 	<div class="sensitive" v-if="media.isSensitive && hide" @click="hide = false"> | ||||
| 		<span class="icon"><fa :icon="faExclamationTriangle"/></span> | ||||
| 		<span class="icon"><Fa :icon="faExclamationTriangle"/></span> | ||||
| 		<b>{{ $t('sensitive') }}</b> | ||||
| 		<span>{{ $t('clickToShow') }}</span> | ||||
| 	</div> | ||||
| @@ -19,17 +19,18 @@ | ||||
| 		:title="media.name" | ||||
| 		:download="media.name" | ||||
| 	> | ||||
| 		<span class="icon"><fa icon="download"/></span> | ||||
| 		<span class="icon"><Fa icon="download"/></span> | ||||
| 		<b>{{ media.name }}</b> | ||||
| 	</a> | ||||
| </div> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import Vue from 'vue'; | ||||
| import { defineComponent } from 'vue'; | ||||
| import { faExclamationTriangle } from '@fortawesome/free-solid-svg-icons'; | ||||
| import * as os from '@/os'; | ||||
|  | ||||
| export default Vue.extend({ | ||||
| export default defineComponent({ | ||||
| 	props: { | ||||
| 		media: { | ||||
| 			type: Object, | ||||
|   | ||||
| @@ -1,34 +1,36 @@ | ||||
| <template> | ||||
| <div class="qjewsnkg" v-if="hide" @click="hide = false"> | ||||
| 	<img-with-blurhash class="bg" :hash="image.blurhash" :title="image.name"/> | ||||
| 	<ImgWithBlurhash class="bg" :hash="image.blurhash" :title="image.name"/> | ||||
| 	<div class="text"> | ||||
| 		<div> | ||||
| 			<b><fa :icon="faExclamationTriangle"/> {{ $t('sensitive') }}</b> | ||||
| 			<b><Fa :icon="faExclamationTriangle"/> {{ $t('sensitive') }}</b> | ||||
| 			<span>{{ $t('clickToShow') }}</span> | ||||
| 		</div> | ||||
| 	</div> | ||||
| </div> | ||||
| <div class="gqnyydlz" v-else> | ||||
| 	<i><fa :icon="faEyeSlash" @click="hide = true"/></i> | ||||
| <div class="gqnyydlz" :style="{ background: color }" v-else> | ||||
| 	<i><Fa :icon="faEyeSlash" @click="hide = true"/></i> | ||||
| 	<a | ||||
| 		:href="image.url" | ||||
| 		:title="image.name" | ||||
| 		@click.prevent="onClick" | ||||
| 	> | ||||
| 		<img-with-blurhash :hash="image.blurhash" :src="url" :alt="image.name" :title="image.name"/> | ||||
| 		<ImgWithBlurhash :hash="image.blurhash" :src="url" :alt="image.name" :title="image.name" :cover="false"/> | ||||
| 		<div class="gif" v-if="image.type === 'image/gif'">GIF</div> | ||||
| 	</a> | ||||
| </div> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import Vue from 'vue'; | ||||
| import { defineComponent } from 'vue'; | ||||
| import { faExclamationTriangle, faEyeSlash } from '@fortawesome/free-solid-svg-icons'; | ||||
| import { getStaticImageUrl } from '../scripts/get-static-image-url'; | ||||
| import { getStaticImageUrl } from '@/scripts/get-static-image-url'; | ||||
| import { extractAvgColorFromBlurhash } from '@/scripts/extract-avg-color-from-blurhash'; | ||||
| import ImageViewer from './image-viewer.vue'; | ||||
| import ImgWithBlurhash from './img-with-blurhash.vue'; | ||||
| import * as os from '@/os'; | ||||
|  | ||||
| export default Vue.extend({ | ||||
| export default defineComponent({ | ||||
| 	components: { | ||||
| 		ImgWithBlurhash | ||||
| 	}, | ||||
| @@ -44,8 +46,8 @@ export default Vue.extend({ | ||||
| 	data() { | ||||
| 		return { | ||||
| 			hide: true, | ||||
| 			faExclamationTriangle, | ||||
| 			faEyeSlash | ||||
| 			color: null, | ||||
| 			faExclamationTriangle, faEyeSlash, | ||||
| 		}; | ||||
| 	}, | ||||
| 	computed: { | ||||
| @@ -64,19 +66,25 @@ export default Vue.extend({ | ||||
| 		} | ||||
| 	}, | ||||
| 	created() { | ||||
| 		this.hide = this.image.isSensitive && !this.$store.state.device.alwaysShowNsfw; | ||||
| 		// Plugin:register_note_view_interruptor を使って書き換えられる可能性があるためwatchする | ||||
| 		this.$watch('image', () => { | ||||
| 			this.hide = this.image.isSensitive && !this.$store.state.device.alwaysShowNsfw; | ||||
| 			if (this.image.blurhash) { | ||||
| 				this.color = extractAvgColorFromBlurhash(this.image.blurhash); | ||||
| 			} | ||||
| 		}, { | ||||
| 			deep: true, | ||||
| 			immediate: true, | ||||
| 		}); | ||||
| 	}, | ||||
| 	methods: { | ||||
| 		onClick() { | ||||
| 			if (this.$store.state.device.imageNewTab) { | ||||
| 				window.open(this.image.url, '_blank'); | ||||
| 			} else { | ||||
| 				const viewer = this.$root.new(ImageViewer, { | ||||
| 				os.popup(ImageViewer, { | ||||
| 					image: this.image | ||||
| 				}); | ||||
| 				this.$once('hook:beforeDestroy', () => { | ||||
| 					viewer.close(); | ||||
| 				}); | ||||
| 				}, {}, 'closed'); | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| @@ -117,6 +125,7 @@ export default Vue.extend({ | ||||
|  | ||||
| .gqnyydlz { | ||||
| 	position: relative; | ||||
| 	border: solid 1px var(--divider); | ||||
|  | ||||
| 	> i { | ||||
| 		display: block; | ||||
|   | ||||
| @@ -1,13 +1,11 @@ | ||||
| <template> | ||||
| <div class="mk-media-list"> | ||||
| 	<template v-for="media in mediaList.filter(media => !previewable(media))"> | ||||
| 		<x-banner :media="media" :key="media.id"/> | ||||
| 	</template> | ||||
| 	<XBanner v-for="media in mediaList.filter(media => !previewable(media))" :media="media" :key="media.id"/> | ||||
| 	<div v-if="mediaList.filter(media => previewable(media)).length > 0" class="gird-container" ref="gridOuter"> | ||||
| 		<div :data-count="mediaList.filter(media => previewable(media)).length" :style="gridInnerStyle"> | ||||
| 			<template v-for="media in mediaList"> | ||||
| 				<x-video :video="media" :key="media.id" v-if="media.type.startsWith('video')"/> | ||||
| 				<x-image :image="media" :key="media.id" v-else-if="media.type.startsWith('image')" :raw="raw"/> | ||||
| 				<XVideo :video="media" :key="media.id" v-if="media.type.startsWith('video')"/> | ||||
| 				<XImage :image="media" :key="media.id" v-else-if="media.type.startsWith('image')" :raw="raw"/> | ||||
| 			</template> | ||||
| 		</div> | ||||
| 	</div> | ||||
| @@ -15,12 +13,13 @@ | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import Vue from 'vue'; | ||||
| import { defineComponent } from 'vue'; | ||||
| import XBanner from './media-banner.vue'; | ||||
| import XImage from './media-image.vue'; | ||||
| import XVideo from './media-video.vue'; | ||||
| import * as os from '@/os'; | ||||
|  | ||||
| export default Vue.extend({ | ||||
| export default defineComponent({ | ||||
| 	components: { | ||||
| 		XBanner, | ||||
| 		XImage, | ||||
| @@ -33,8 +32,6 @@ export default Vue.extend({ | ||||
| 		raw: { | ||||
| 			default: false | ||||
| 		}, | ||||
| 		// specify the parent element | ||||
| 		parentElement: {} | ||||
| 	}, | ||||
| 	data() { | ||||
| 		return { | ||||
| @@ -46,7 +43,7 @@ export default Vue.extend({ | ||||
| 		this.size(); | ||||
| 		window.addEventListener('resize', this.size); | ||||
| 	}, | ||||
| 	beforeDestroy() { | ||||
| 	beforeUnmount() { | ||||
| 		window.removeEventListener('resize', this.size); | ||||
| 	}, | ||||
| 	activated() { | ||||
| @@ -67,7 +64,7 @@ export default Vue.extend({ | ||||
|  | ||||
| 				if (this.$refs.gridOuter) { | ||||
| 					let height = 287; | ||||
| 					const parent = this.parentElement || this.$parent.$el; | ||||
| 					const parent = this.$parent.$el; | ||||
|  | ||||
| 					if (this.$refs.gridOuter.clientHeight) { | ||||
| 						height = this.$refs.gridOuter.clientHeight; | ||||
| @@ -82,11 +79,6 @@ export default Vue.extend({ | ||||
| 			}); | ||||
| 		} | ||||
| 	}, | ||||
| 	watch: { | ||||
| 		parentElement() { | ||||
| 			this.size(); | ||||
| 		} | ||||
| 	} | ||||
| }); | ||||
| </script> | ||||
|  | ||||
|   | ||||
| @@ -1,12 +1,12 @@ | ||||
| <template> | ||||
| <div class="icozogqfvdetwohsdglrbswgrejoxbdj" v-if="hide" @click="hide = false"> | ||||
| 	<div> | ||||
| 		<b><fa :icon="faExclamationTriangle"/> {{ $t('sensitive') }}</b> | ||||
| 		<b><Fa :icon="faExclamationTriangle"/> {{ $t('sensitive') }}</b> | ||||
| 		<span>{{ $t('clickToShow') }}</span> | ||||
| 	</div> | ||||
| </div> | ||||
| <div class="kkjnbbplepmiyuadieoenjgutgcmtsvu" v-else> | ||||
| 	<i><fa :icon="faEyeSlash" @click="hide = true"/></i> | ||||
| 	<i><Fa :icon="faEyeSlash" @click="hide = true"/></i> | ||||
| 	<a | ||||
| 		:href="video.url" | ||||
| 		rel="nofollow noopener" | ||||
| @@ -14,17 +14,18 @@ | ||||
| 		:style="imageStyle" | ||||
| 		:title="video.name" | ||||
| 	> | ||||
| 		<fa :icon="faPlayCircle"/> | ||||
| 		<Fa :icon="faPlayCircle"/> | ||||
| 	</a> | ||||
| </div> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import Vue from 'vue'; | ||||
| import { defineComponent } from 'vue'; | ||||
| import { faPlayCircle } from '@fortawesome/free-regular-svg-icons'; | ||||
| import { faExclamationTriangle, faEyeSlash } from '@fortawesome/free-solid-svg-icons'; | ||||
| import * as os from '@/os'; | ||||
|  | ||||
| export default Vue.extend({ | ||||
| export default defineComponent({ | ||||
| 	props: { | ||||
| 		video: { | ||||
| 			type: Object, | ||||
|   | ||||
| @@ -1,11 +1,11 @@ | ||||
| <template> | ||||
| <router-link class="ldlomzub" :class="{ isMe }" :to="url" v-user-preview="canonical" v-if="url.startsWith('/')"> | ||||
| <MkA class="ldlomzub" :class="{ isMe }" :to="url" v-user-preview="canonical" v-if="url.startsWith('/')"> | ||||
| 	<span class="me" v-if="isMe">{{ $t('you') }}</span> | ||||
| 	<span class="main"> | ||||
| 		<span class="username">@{{ username }}</span> | ||||
| 		<span class="host" v-if="(host != localHost) || $store.state.settings.showFullAcct">@{{ toUnicode(host) }}</span> | ||||
| 	</span> | ||||
| </router-link> | ||||
| </MkA> | ||||
| <a class="ldlomzub" :href="url" target="_blank" rel="noopener" v-else> | ||||
| 	<span class="main"> | ||||
| 		<span class="username">@{{ username }}</span> | ||||
| @@ -15,11 +15,13 @@ | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import Vue from 'vue'; | ||||
| import { defineComponent } from 'vue'; | ||||
| import { toUnicode } from 'punycode'; | ||||
| import { host as localHost } from '../config'; | ||||
| import { host as localHost } from '@/config'; | ||||
| import { wellKnownServices } from '../../well-known-services'; | ||||
| import * as os from '@/os'; | ||||
|  | ||||
| export default Vue.extend({ | ||||
| export default defineComponent({ | ||||
| 	props: { | ||||
| 		username: { | ||||
| 			type: String, | ||||
| @@ -37,12 +39,11 @@ export default Vue.extend({ | ||||
| 	}, | ||||
| 	computed: { | ||||
| 		url(): string { | ||||
| 			switch (this.host) { | ||||
| 				case 'twitter.com': | ||||
| 				case 'github.com': | ||||
| 					return `https://${this.host}/${this.username}`; | ||||
| 				default: | ||||
| 					return `/${this.canonical}`; | ||||
| 			const wellKnown = wellKnownServices.find(x => x[0] === this.host); | ||||
| 			if (wellKnown) { | ||||
| 				return wellKnown[1](this.username); | ||||
| 			} else { | ||||
| 				return `/${this.canonical}`; | ||||
| 			} | ||||
| 		}, | ||||
| 		canonical(): string { | ||||
|   | ||||
| @@ -1,191 +0,0 @@ | ||||
| <template> | ||||
| <x-popup :source="source" :no-center="noCenter" :fixed="fixed" :width="width" ref="popup" @closed="() => { $emit('closed'); destroyDom(); }" v-hotkey.global="keymap"> | ||||
| 	<div class="rrevdjwt" :class="{ left: align === 'left' }" ref="items"> | ||||
| 		<template v-for="(item, i) in items.filter(item => item !== undefined)"> | ||||
| 			<div v-if="item === null" class="divider" :key="i"></div> | ||||
| 			<span v-else-if="item.type === 'label'" class="label item" :key="i"> | ||||
| 				<span>{{ item.text }}</span> | ||||
| 			</span> | ||||
| 			<router-link v-else-if="item.type === 'link'" :to="item.to" @click.native="close()" :tabindex="i" class="_button item" :key="i"> | ||||
| 				<fa v-if="item.icon" :icon="item.icon" fixed-width/> | ||||
| 				<mk-avatar v-if="item.avatar" :user="item.avatar" class="avatar"/> | ||||
| 				<span>{{ item.text }}</span> | ||||
| 				<i v-if="item.indicate"><fa :icon="faCircle"/></i> | ||||
| 			</router-link> | ||||
| 			<a v-else-if="item.type === 'a'" :href="item.href" :target="item.target" :download="item.download" @click="close()" :tabindex="i" class="_button item" :key="i"> | ||||
| 				<fa v-if="item.icon" :icon="item.icon" fixed-width/> | ||||
| 				<span>{{ item.text }}</span> | ||||
| 				<i v-if="item.indicate"><fa :icon="faCircle"/></i> | ||||
| 			</a> | ||||
| 			<button v-else-if="item.type === 'user'" @click="clicked(item.action)" :tabindex="i" class="_button item" :key="i"> | ||||
| 				<mk-avatar :user="item.user" class="avatar"/><mk-user-name :user="item.user"/> | ||||
| 				<i v-if="item.indicate"><fa :icon="faCircle"/></i> | ||||
| 			</button> | ||||
| 			<button v-else @click="clicked(item.action)" :tabindex="i" class="_button item" :key="i"> | ||||
| 				<fa v-if="item.icon" :icon="item.icon" fixed-width/> | ||||
| 				<mk-avatar v-if="item.avatar" :user="item.avatar" class="avatar"/> | ||||
| 				<span>{{ item.text }}</span> | ||||
| 				<i v-if="item.indicate"><fa :icon="faCircle"/></i> | ||||
| 			</button> | ||||
| 		</template> | ||||
| 	</div> | ||||
| </x-popup> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import Vue from 'vue'; | ||||
| import { faCircle } from '@fortawesome/free-solid-svg-icons'; | ||||
| import XPopup from './popup.vue'; | ||||
| import { focusPrev, focusNext } from '../scripts/focus'; | ||||
|  | ||||
| export default Vue.extend({ | ||||
| 	components: { | ||||
| 		XPopup | ||||
| 	}, | ||||
| 	props: { | ||||
| 		source: { | ||||
| 			required: true | ||||
| 		}, | ||||
| 		items: { | ||||
| 			type: Array, | ||||
| 			required: true | ||||
| 		}, | ||||
| 		align: { | ||||
| 			type: String, | ||||
| 			required: false | ||||
| 		}, | ||||
| 		noCenter: { | ||||
| 			type: Boolean, | ||||
| 			required: false | ||||
| 		}, | ||||
| 		fixed: { | ||||
| 			type: Boolean, | ||||
| 			required: false | ||||
| 		}, | ||||
| 		width: { | ||||
| 			type: Number, | ||||
| 			required: false | ||||
| 		}, | ||||
| 		direction: { | ||||
| 			type: String, | ||||
| 			required: false | ||||
| 		}, | ||||
| 		viaKeyboard: { | ||||
| 			type: Boolean, | ||||
| 			required: false | ||||
| 		}, | ||||
| 	}, | ||||
| 	data() { | ||||
| 		return { | ||||
| 			faCircle | ||||
| 		}; | ||||
| 	}, | ||||
| 	computed: { | ||||
| 		keymap(): any { | ||||
| 			return { | ||||
| 				'up|k|shift+tab': this.focusUp, | ||||
| 				'down|j|tab': this.focusDown, | ||||
| 			}; | ||||
| 		}, | ||||
| 	}, | ||||
| 	mounted() { | ||||
| 		if (this.viaKeyboard) { | ||||
| 			this.$nextTick(() => { | ||||
| 				focusNext(this.$refs.items.children[0], true); | ||||
| 			}); | ||||
| 		} | ||||
| 	}, | ||||
| 	methods: { | ||||
| 		clicked(fn) { | ||||
| 			fn(); | ||||
| 			this.close(); | ||||
| 		}, | ||||
| 		close() { | ||||
| 			this.$refs.popup.close(); | ||||
| 		}, | ||||
| 		focusUp() { | ||||
| 			focusPrev(document.activeElement); | ||||
| 		}, | ||||
| 		focusDown() { | ||||
| 			focusNext(document.activeElement); | ||||
| 		} | ||||
| 	} | ||||
| }); | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" scoped> | ||||
| .rrevdjwt { | ||||
| 	padding: 8px 0; | ||||
|  | ||||
| 	&.left { | ||||
| 		> .item { | ||||
| 			text-align: left; | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	> .item { | ||||
| 		display: block; | ||||
| 		position: relative; | ||||
| 		padding: 8px 16px; | ||||
| 		width: 100%; | ||||
| 		box-sizing: border-box; | ||||
| 		white-space: nowrap; | ||||
| 		font-size: 0.9em; | ||||
| 		line-height: 20px; | ||||
| 		text-align: center; | ||||
| 		overflow: hidden; | ||||
| 		text-overflow: ellipsis; | ||||
|  | ||||
| 		&:hover { | ||||
| 			color: #fff; | ||||
| 			background: var(--accent); | ||||
| 			text-decoration: none; | ||||
| 		} | ||||
|  | ||||
| 		&:active { | ||||
| 			color: #fff; | ||||
| 			background: var(--accentDarken); | ||||
| 		} | ||||
|  | ||||
| 		&:not(:active):focus { | ||||
| 			box-shadow: 0 0 0 2px var(--focus) inset; | ||||
| 		} | ||||
|  | ||||
| 		&.label { | ||||
| 			pointer-events: none; | ||||
| 			font-size: 0.7em; | ||||
| 			padding-bottom: 4px; | ||||
|  | ||||
| 			> span { | ||||
| 				opacity: 0.7; | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		> [data-icon] { | ||||
| 			margin-right: 4px; | ||||
| 			width: 20px; | ||||
| 		} | ||||
|  | ||||
| 		> .avatar { | ||||
| 			margin-right: 4px; | ||||
| 			width: 20px; | ||||
| 			height: 20px; | ||||
| 		} | ||||
|  | ||||
| 		> i { | ||||
| 			position: absolute; | ||||
| 			top: 5px; | ||||
| 			left: 13px; | ||||
| 			color: var(--indicator); | ||||
| 			font-size: 12px; | ||||
| 			animation: blink 1s infinite; | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	> .divider { | ||||
| 		margin: 8px 0; | ||||
| 		height: 1px; | ||||
| 		background: var(--divider); | ||||
| 	} | ||||
| } | ||||
| </style> | ||||
| @@ -1,16 +1,18 @@ | ||||
| import Vue, { VNode } from 'vue'; | ||||
| import { VNode, defineComponent, h } from 'vue'; | ||||
| import { MfmForest } from '../../mfm/prelude'; | ||||
| import { parse, parsePlain } from '../../mfm/parse'; | ||||
| import MkUrl from './url.vue'; | ||||
| import MkLink from './link.vue'; | ||||
| import MkMention from './mention.vue'; | ||||
| import MkEmoji from './emoji.vue'; | ||||
| import { concat } from '../../prelude/array'; | ||||
| import MkFormula from './formula.vue'; | ||||
| import MkCode from './code.vue'; | ||||
| import MkGoogle from './google.vue'; | ||||
| import { host } from '../config'; | ||||
| import MkA from './ui/a.vue'; | ||||
| import { host } from '@/config'; | ||||
|  | ||||
| export default Vue.component('misskey-flavored-markdown', { | ||||
| export default defineComponent({ | ||||
| 	props: { | ||||
| 		text: { | ||||
| 			type: String, | ||||
| @@ -41,7 +43,7 @@ export default Vue.component('misskey-flavored-markdown', { | ||||
| 		}, | ||||
| 	}, | ||||
|  | ||||
| 	render(createElement) { | ||||
| 	render() { | ||||
| 		if (this.text == null || this.text == '') return; | ||||
|  | ||||
| 		const ast = (this.plain ? parsePlain : parse)(this.text); | ||||
| @@ -53,233 +55,197 @@ export default Vue.component('misskey-flavored-markdown', { | ||||
|  | ||||
| 					if (!this.plain) { | ||||
| 						const x = text.split('\n') | ||||
| 							.map(t => t == '' ? [createElement('br')] : [this._v(t), createElement('br')]); // NOTE: this._vはHACK SEE: https://github.com/syuilo/misskey/pull/6399#issuecomment-632820283 | ||||
| 							.map(t => t == '' ? [h('br')] : [t, h('br')]); | ||||
| 						x[x.length - 1].pop(); | ||||
| 						return x; | ||||
| 					} else { | ||||
| 						return [this._v(text.replace(/\n/g, ' '))]; | ||||
| 						return [text.replace(/\n/g, ' ')]; | ||||
| 					} | ||||
| 				} | ||||
|  | ||||
| 				case 'bold': { | ||||
| 					return [createElement('b', genEl(token.children))]; | ||||
| 					return [h('b', genEl(token.children))]; | ||||
| 				} | ||||
|  | ||||
| 				case 'strike': { | ||||
| 					return [createElement('del', genEl(token.children))]; | ||||
| 					return [h('del', genEl(token.children))]; | ||||
| 				} | ||||
|  | ||||
| 				case 'italic': { | ||||
| 					return (createElement as any)('i', { | ||||
| 						attrs: { | ||||
| 							style: 'font-style: oblique;' | ||||
| 						}, | ||||
| 					return h('i', { | ||||
| 						style: 'font-style: oblique;' | ||||
| 					}, genEl(token.children)); | ||||
| 				} | ||||
|  | ||||
| 				case 'big': { | ||||
| 					return (createElement as any)('strong', { | ||||
| 						attrs: { | ||||
| 							style: `display: inline-block; font-size: 150%;` | ||||
| 						}, | ||||
| 						directives: [this.$store.state.device.animatedMfm ? { | ||||
| 							name: 'animate-css', | ||||
| 							value: { classes: 'tada', iteration: 'infinite' } | ||||
| 						}: {}] | ||||
| 					}, genEl(token.children)); | ||||
| 				case 'fn': { | ||||
| 					// TODO: CSSを文字列で組み立てていくと token.node.props.args.~~~ 経由でCSSインジェクションできるのでよしなにやる | ||||
| 					let style; | ||||
| 					switch (token.node.props.name) { | ||||
| 						case 'tada': { | ||||
| 							style = `font-size: 150%;` + (this.$store.state.device.animatedMfm ? 'animation: tada 1s linear infinite both;' : ''); | ||||
| 							break; | ||||
| 						} | ||||
| 						case 'jelly': { | ||||
| 							const speed = token.node.props.args.speed || '1s'; | ||||
| 							style = (this.$store.state.device.animatedMfm ? `animation: mfm-rubberBand ${speed} linear infinite both;` : ''); | ||||
| 							break; | ||||
| 						} | ||||
| 						case 'twitch': { | ||||
| 							const speed = token.node.props.args.speed || '0.5s'; | ||||
| 							style = this.$store.state.device.animatedMfm ? `animation: mfm-twitch ${speed} ease infinite;` : ''; | ||||
| 							break; | ||||
| 						} | ||||
| 						case 'shake': { | ||||
| 							const speed = token.node.props.args.speed || '0.5s'; | ||||
| 							style = this.$store.state.device.animatedMfm ? `animation: mfm-shake ${speed} ease infinite;` : ''; | ||||
| 							break; | ||||
| 						} | ||||
| 						case 'spin': { | ||||
| 							const direction = | ||||
| 								token.node.props.args.left ? 'reverse' : | ||||
| 								token.node.props.args.alternate ? 'alternate' : | ||||
| 								'normal'; | ||||
| 							const anime = | ||||
| 								token.node.props.args.x ? 'mfm-spinX' : | ||||
| 								token.node.props.args.y ? 'mfm-spinY' : | ||||
| 								'mfm-spin'; | ||||
| 							const speed = token.node.props.args.speed || '1.5s'; | ||||
| 							style = this.$store.state.device.animatedMfm ? `animation: ${anime} ${speed} linear infinite; animation-direction: ${direction};` : ''; | ||||
| 							break; | ||||
| 						} | ||||
| 						case 'jump': { | ||||
| 							style = this.$store.state.device.animatedMfm ? 'animation: mfm-jump 0.75s linear infinite;' : ''; | ||||
| 							break; | ||||
| 						} | ||||
| 						case 'bounce': { | ||||
| 							style = this.$store.state.device.animatedMfm ? 'animation: mfm-bounce 0.75s linear infinite; transform-origin: center bottom;' : ''; | ||||
| 							break; | ||||
| 						} | ||||
| 						case 'flip': { | ||||
| 							const transform = | ||||
| 								(token.node.props.args.h && token.node.props.args.v) ? 'scale(-1, -1)' : | ||||
| 								token.node.props.args.v ? 'scaleY(-1)' : | ||||
| 								'scaleX(-1)'; | ||||
| 							style = `transform: ${transform};`; | ||||
| 							break; | ||||
| 						} | ||||
| 					} | ||||
| 					if (style == null) { | ||||
| 						return h('span', {}, ['[', token.node.props.name, ...genEl(token.children), ']']); | ||||
| 					} else { | ||||
| 						return h('span', { | ||||
| 							style: 'display: inline-block;' + style, | ||||
| 						}, genEl(token.children)); | ||||
| 					} | ||||
| 				} | ||||
|  | ||||
| 				case 'small': { | ||||
| 					return [createElement('small', { | ||||
| 						attrs: { | ||||
| 							style: 'opacity: 0.7;' | ||||
| 						}, | ||||
| 					return [h('small', { | ||||
| 						style: 'opacity: 0.7;' | ||||
| 					}, genEl(token.children))]; | ||||
| 				} | ||||
|  | ||||
| 				case 'center': { | ||||
| 					return [createElement('div', { | ||||
| 						attrs: { | ||||
| 							style: 'text-align:center;' | ||||
| 						} | ||||
| 					return [h('div', { | ||||
| 						style: 'text-align:center;' | ||||
| 					}, genEl(token.children))]; | ||||
| 				} | ||||
|  | ||||
| 				case 'motion': { | ||||
| 					return (createElement as any)('span', { | ||||
| 						attrs: { | ||||
| 							style: 'display: inline-block;' | ||||
| 						}, | ||||
| 						directives: [this.$store.state.device.animatedMfm ? { | ||||
| 							name: 'animate-css', | ||||
| 							value: { classes: 'rubberBand', iteration: 'infinite' } | ||||
| 						} : {}] | ||||
| 					}, genEl(token.children)); | ||||
| 				} | ||||
|  | ||||
| 				case 'spin': { | ||||
| 					const direction = | ||||
| 						token.node.props.attr == 'left' ? 'reverse' : | ||||
| 						token.node.props.attr == 'alternate' ? 'alternate' : | ||||
| 						'normal'; | ||||
| 					const style = this.$store.state.device.animatedMfm | ||||
| 						? `animation: spin 1.5s linear infinite; animation-direction: ${direction};` : ''; | ||||
| 					return (createElement as any)('span', { | ||||
| 						attrs: { | ||||
| 							style: 'display: inline-block;' + style | ||||
| 						}, | ||||
| 					}, genEl(token.children)); | ||||
| 				} | ||||
|  | ||||
| 				case 'jump': { | ||||
| 					return (createElement as any)('span', { | ||||
| 						attrs: { | ||||
| 							style: this.$store.state.device.animatedMfm ? 'display: inline-block; animation: jump 0.75s linear infinite;' : 'display: inline-block;' | ||||
| 						}, | ||||
| 					}, genEl(token.children)); | ||||
| 				} | ||||
|  | ||||
| 				case 'flip': { | ||||
| 					return (createElement as any)('span', { | ||||
| 						attrs: { | ||||
| 							style: 'display: inline-block; transform: scaleX(-1);' | ||||
| 						}, | ||||
| 					}, genEl(token.children)); | ||||
| 				} | ||||
|  | ||||
| 				case 'url': { | ||||
| 					return [createElement(MkUrl, { | ||||
| 					return [h(MkUrl, { | ||||
| 						key: Math.random(), | ||||
| 						props: { | ||||
| 							url: token.node.props.url, | ||||
| 							rel: 'nofollow noopener', | ||||
| 						}, | ||||
| 						url: token.node.props.url, | ||||
| 						rel: 'nofollow noopener', | ||||
| 					})]; | ||||
| 				} | ||||
|  | ||||
| 				case 'link': { | ||||
| 					return [createElement(MkLink, { | ||||
| 					return [h(MkLink, { | ||||
| 						key: Math.random(), | ||||
| 						props: { | ||||
| 							url: token.node.props.url, | ||||
| 							rel: 'nofollow noopener', | ||||
| 						}, | ||||
| 						url: token.node.props.url, | ||||
| 						rel: 'nofollow noopener', | ||||
| 					}, genEl(token.children))]; | ||||
| 				} | ||||
|  | ||||
| 				case 'mention': { | ||||
| 					return [createElement(MkMention, { | ||||
| 					return [h(MkMention, { | ||||
| 						key: Math.random(), | ||||
| 						props: { | ||||
| 							host: (token.node.props.host == null && this.author && this.author.host != null ? this.author.host : token.node.props.host) || host, | ||||
| 							username: token.node.props.username | ||||
| 						} | ||||
| 						host: (token.node.props.host == null && this.author && this.author.host != null ? this.author.host : token.node.props.host) || host, | ||||
| 						username: token.node.props.username | ||||
| 					})]; | ||||
| 				} | ||||
|  | ||||
| 				case 'hashtag': { | ||||
| 					return [createElement('router-link', { | ||||
| 					return [h(MkA, { | ||||
| 						key: Math.random(), | ||||
| 						attrs: { | ||||
| 							to: this.isNote ? `/tags/${encodeURIComponent(token.node.props.hashtag)}` : `/explore/tags/${encodeURIComponent(token.node.props.hashtag)}`, | ||||
| 							style: 'color:var(--hashtag);' | ||||
| 						} | ||||
| 						to: this.isNote ? `/tags/${encodeURIComponent(token.node.props.hashtag)}` : `/explore/tags/${encodeURIComponent(token.node.props.hashtag)}`, | ||||
| 						style: 'color:var(--hashtag);' | ||||
| 					}, `#${token.node.props.hashtag}`)]; | ||||
| 				} | ||||
|  | ||||
| 				case 'blockCode': { | ||||
| 					return [createElement(MkCode, { | ||||
| 					return [h(MkCode, { | ||||
| 						key: Math.random(), | ||||
| 						props: { | ||||
| 							code: token.node.props.code, | ||||
| 							lang: token.node.props.lang, | ||||
| 						} | ||||
| 						code: token.node.props.code, | ||||
| 						lang: token.node.props.lang, | ||||
| 					})]; | ||||
| 				} | ||||
|  | ||||
| 				case 'inlineCode': { | ||||
| 					return [createElement(MkCode, { | ||||
| 					return [h(MkCode, { | ||||
| 						key: Math.random(), | ||||
| 						props: { | ||||
| 							code: token.node.props.code, | ||||
| 							lang: token.node.props.lang, | ||||
| 							inline: true | ||||
| 						} | ||||
| 						code: token.node.props.code, | ||||
| 						lang: token.node.props.lang, | ||||
| 						inline: true | ||||
| 					})]; | ||||
| 				} | ||||
|  | ||||
| 				case 'quote': { | ||||
| 					if (this.shouldBreak) { | ||||
| 						return [createElement('div', { | ||||
| 							attrs: { | ||||
| 								class: 'quote' | ||||
| 							} | ||||
| 					if (!this.nowrap) { | ||||
| 						return [h('div', { | ||||
| 							class: 'quote' | ||||
| 						}, genEl(token.children))]; | ||||
| 					} else { | ||||
| 						return [createElement('span', { | ||||
| 							attrs: { | ||||
| 								class: 'quote' | ||||
| 							} | ||||
| 						return [h('span', { | ||||
| 							class: 'quote' | ||||
| 						}, genEl(token.children))]; | ||||
| 					} | ||||
| 				} | ||||
|  | ||||
| 				case 'title': { | ||||
| 					return [createElement('div', { | ||||
| 						attrs: { | ||||
| 							class: 'title' | ||||
| 						} | ||||
| 					}, genEl(token.children))]; | ||||
| 				} | ||||
|  | ||||
| 				case 'emoji': { | ||||
| 					return [createElement('mk-emoji', { | ||||
| 					return [h(MkEmoji, { | ||||
| 						key: Math.random(), | ||||
| 						attrs: { | ||||
| 							emoji: token.node.props.emoji, | ||||
| 							name: token.node.props.name | ||||
| 						}, | ||||
| 						props: { | ||||
| 							customEmojis: this.customEmojis, | ||||
| 							normal: this.plain | ||||
| 						} | ||||
| 						emoji: token.node.props.name ? `:${token.node.props.name}:` : token.node.props.emoji, | ||||
| 						customEmojis: this.customEmojis, | ||||
| 						normal: this.plain | ||||
| 					})]; | ||||
| 				} | ||||
|  | ||||
| 				case 'mathInline': { | ||||
| 					//const MkFormula = () => import('./formula.vue').then(m => m.default); | ||||
| 					return [createElement(MkFormula, { | ||||
| 					return [h(MkFormula, { | ||||
| 						key: Math.random(), | ||||
| 						props: { | ||||
| 							formula: token.node.props.formula, | ||||
| 							block: false | ||||
| 						} | ||||
| 						formula: token.node.props.formula, | ||||
| 						block: false | ||||
| 					})]; | ||||
| 				} | ||||
|  | ||||
| 				case 'mathBlock': { | ||||
| 					//const MkFormula = () => import('./formula.vue').then(m => m.default); | ||||
| 					return [createElement(MkFormula, { | ||||
| 					return [h(MkFormula, { | ||||
| 						key: Math.random(), | ||||
| 						props: { | ||||
| 							formula: token.node.props.formula, | ||||
| 							block: true | ||||
| 						} | ||||
| 						formula: token.node.props.formula, | ||||
| 						block: true | ||||
| 					})]; | ||||
| 				} | ||||
|  | ||||
| 				case 'search': { | ||||
| 					//const MkGoogle = () => import('./google.vue').then(m => m.default); | ||||
| 					return [createElement(MkGoogle, { | ||||
| 					return [h(MkGoogle, { | ||||
| 						key: Math.random(), | ||||
| 						props: { | ||||
| 							q: token.node.props.query | ||||
| 						} | ||||
| 						q: token.node.props.query | ||||
| 					})]; | ||||
| 				} | ||||
|  | ||||
| 				default: { | ||||
| 					console.log('unknown ast type:', token.node.type); | ||||
| 					console.error('unrecognized ast type:', token.node.type); | ||||
|  | ||||
| 					return []; | ||||
| 				} | ||||
| @@ -287,6 +253,6 @@ export default Vue.component('misskey-flavored-markdown', { | ||||
| 		})); | ||||
|  | ||||
| 		// Parse ast to DOM | ||||
| 		return createElement('span', genEl(ast)); | ||||
| 		return h('span', genEl(ast)); | ||||
| 	} | ||||
| }); | ||||
|   | ||||
| @@ -30,10 +30,11 @@ | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import Vue from 'vue'; | ||||
| import { defineComponent } from 'vue'; | ||||
| import { v4 as uuid } from 'uuid'; | ||||
| import * as os from '@/os'; | ||||
|  | ||||
| export default Vue.extend({ | ||||
| export default defineComponent({ | ||||
| 	props: { | ||||
| 		src: { | ||||
| 			type: Array, | ||||
| @@ -64,7 +65,7 @@ export default Vue.extend({ | ||||
| 		// Vueが何故かWatchを発動させない場合があるので | ||||
| 		this.clock = setInterval(this.draw, 1000); | ||||
| 	}, | ||||
| 	beforeDestroy() { | ||||
| 	beforeUnmount() { | ||||
| 		clearInterval(this.clock); | ||||
| 	}, | ||||
| 	methods: { | ||||
|   | ||||
| @@ -3,16 +3,113 @@ | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import Vue from 'vue'; | ||||
| import { defineComponent } from 'vue'; | ||||
| import MfmCore from './mfm'; | ||||
|  | ||||
| export default Vue.extend({ | ||||
| export default defineComponent({ | ||||
| 	components: { | ||||
| 		MfmCore | ||||
| 	} | ||||
| }); | ||||
| </script> | ||||
|  | ||||
| <style lang="scss"> | ||||
| @keyframes mfm-spin { | ||||
| 	0% { transform: rotate(0deg); } | ||||
| 	100% { transform: rotate(360deg); } | ||||
| } | ||||
|  | ||||
| @keyframes mfm-spinX { | ||||
| 	0% { transform: perspective(128px) rotateX(0deg); } | ||||
| 	100% { transform: perspective(128px) rotateX(360deg); } | ||||
| } | ||||
|  | ||||
| @keyframes mfm-spinY { | ||||
| 	0% { transform: perspective(128px) rotateY(0deg); } | ||||
| 	100% { transform: perspective(128px) rotateY(360deg); } | ||||
| } | ||||
|  | ||||
| @keyframes mfm-jump { | ||||
| 	0% { transform: translateY(0); } | ||||
| 	25% { transform: translateY(-16px); } | ||||
| 	50% { transform: translateY(0); } | ||||
| 	75% { transform: translateY(-8px); } | ||||
| 	100% { transform: translateY(0); } | ||||
| } | ||||
|  | ||||
| @keyframes mfm-bounce { | ||||
| 	0% { transform: translateY(0) scale(1, 1); } | ||||
| 	25% { transform: translateY(-16px) scale(1, 1); } | ||||
| 	50% { transform: translateY(0) scale(1, 1); } | ||||
| 	75% { transform: translateY(0) scale(1.5, 0.75); } | ||||
| 	100% { transform: translateY(0) scale(1, 1); } | ||||
| } | ||||
|  | ||||
| // const val = () => `translate(${Math.floor(Math.random() * 20) - 10}px, ${Math.floor(Math.random() * 20) - 10}px)`; | ||||
| // let css = ''; | ||||
| // for (let i = 0; i <= 100; i += 5) { css += `${i}% { transform: ${val()} }\n`; } | ||||
| @keyframes mfm-twitch { | ||||
| 	0% { transform: translate(7px, -2px) } | ||||
| 	5% { transform: translate(-3px, 1px) } | ||||
| 	10% { transform: translate(-7px, -1px) } | ||||
| 	15% { transform: translate(0px, -1px) } | ||||
| 	20% { transform: translate(-8px, 6px) } | ||||
| 	25% { transform: translate(-4px, -3px) } | ||||
| 	30% { transform: translate(-4px, -6px) } | ||||
| 	35% { transform: translate(-8px, -8px) } | ||||
| 	40% { transform: translate(4px, 6px) } | ||||
| 	45% { transform: translate(-3px, 1px) } | ||||
| 	50% { transform: translate(2px, -10px) } | ||||
| 	55% { transform: translate(-7px, 0px) } | ||||
| 	60% { transform: translate(-2px, 4px) } | ||||
| 	65% { transform: translate(3px, -8px) } | ||||
| 	70% { transform: translate(6px, 7px) } | ||||
| 	75% { transform: translate(-7px, -2px) } | ||||
| 	80% { transform: translate(-7px, -8px) } | ||||
| 	85% { transform: translate(9px, 3px) } | ||||
| 	90% { transform: translate(-3px, -2px) } | ||||
| 	95% { transform: translate(-10px, 2px) } | ||||
| 	100% { transform: translate(-2px, -6px) } | ||||
| } | ||||
|  | ||||
| // const val = () => `translate(${Math.floor(Math.random() * 6) - 3}px, ${Math.floor(Math.random() * 6) - 3}px) rotate(${Math.floor(Math.random() * 24) - 12}deg)`; | ||||
| // let css = ''; | ||||
| // for (let i = 0; i <= 100; i += 5) { css += `${i}% { transform: ${val()} }\n`; } | ||||
| @keyframes mfm-shake { | ||||
| 	0% { transform: translate(-3px, -1px) rotate(-8deg) } | ||||
| 	5% { transform: translate(0px, -1px) rotate(-10deg) } | ||||
| 	10% { transform: translate(1px, -3px) rotate(0deg) } | ||||
| 	15% { transform: translate(1px, 1px) rotate(11deg) } | ||||
| 	20% { transform: translate(-2px, 1px) rotate(1deg) } | ||||
| 	25% { transform: translate(-1px, -2px) rotate(-2deg) } | ||||
| 	30% { transform: translate(-1px, 2px) rotate(-3deg) } | ||||
| 	35% { transform: translate(2px, 1px) rotate(6deg) } | ||||
| 	40% { transform: translate(-2px, -3px) rotate(-9deg) } | ||||
| 	45% { transform: translate(0px, -1px) rotate(-12deg) } | ||||
| 	50% { transform: translate(1px, 2px) rotate(10deg) } | ||||
| 	55% { transform: translate(0px, -3px) rotate(8deg) } | ||||
| 	60% { transform: translate(1px, -1px) rotate(8deg) } | ||||
| 	65% { transform: translate(0px, -1px) rotate(-7deg) } | ||||
| 	70% { transform: translate(-1px, -3px) rotate(6deg) } | ||||
| 	75% { transform: translate(0px, -2px) rotate(4deg) } | ||||
| 	80% { transform: translate(-2px, -1px) rotate(3deg) } | ||||
| 	85% { transform: translate(1px, -3px) rotate(-10deg) } | ||||
| 	90% { transform: translate(1px, 0px) rotate(3deg) } | ||||
| 	95% { transform: translate(-2px, 0px) rotate(-3deg) } | ||||
| 	100% { transform: translate(2px, 1px) rotate(2deg) } | ||||
| } | ||||
|  | ||||
| @keyframes mfm-rubberBand { | ||||
| 	from { transform: scale3d(1, 1, 1); } | ||||
| 	30% { transform: scale3d(1.25, 0.75, 1); } | ||||
| 	40% { transform: scale3d(0.75, 1.25, 1); } | ||||
| 	50% { transform: scale3d(1.15, 0.85, 1); } | ||||
| 	65% { transform: scale3d(0.95, 1.05, 1); } | ||||
| 	75% { transform: scale3d(1.05, 0.95, 1); } | ||||
| 	to { transform: scale3d(1, 1, 1); } | ||||
| } | ||||
| </style> | ||||
|  | ||||
| <style lang="scss" scoped> | ||||
| .havbbuyv { | ||||
| 	white-space: pre-wrap; | ||||
| @@ -24,7 +121,7 @@ export default Vue.extend({ | ||||
| 		text-overflow: ellipsis; | ||||
| 	} | ||||
|  | ||||
| 	::v-deep .quote { | ||||
| 	::v-deep(.quote) { | ||||
| 		display: block; | ||||
| 		margin: 8px; | ||||
| 		padding: 6px 0 6px 12px; | ||||
| @@ -33,17 +130,14 @@ export default Vue.extend({ | ||||
| 		opacity: 0.7; | ||||
| 	} | ||||
|  | ||||
| 	::v-deep pre { | ||||
| 	::v-deep(pre) { | ||||
| 		font-size: 0.8em; | ||||
| 	} | ||||
|  | ||||
| 	::v-deep > code { | ||||
| 	> ::v-deep(code) { | ||||
| 		font-size: 0.8em; | ||||
| 		word-break: break-all; | ||||
| 	} | ||||
|  | ||||
| 	::v-deep .title { | ||||
| 		text-align: center; | ||||
| 		border-bottom: solid 1px var(--divider); | ||||
| 		padding: 4px 6px; | ||||
| 	} | ||||
| } | ||||
| </style> | ||||
|   | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user