Compare commits
	
		
			31 Commits
		
	
	
		
			13.13.0-be
			...
			deck-nav
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | d8f65ca426 | ||
|   | 16b50fc6a9 | ||
|   | fb54c58a66 | ||
|   | 3a924f3dc6 | ||
|   | 11d22c7b73 | ||
|   | a879607479 | ||
|   | 98aef974df | ||
|   | cf46816687 | ||
|   | eee1e74174 | ||
|   | 8050f89d7e | ||
|   | 406e5d297b | ||
|   | 10634b3615 | ||
|   | fd03e2e1a7 | ||
|   | 6cc86272f3 | ||
|   | 06b1250d47 | ||
|   | 31a7350a10 | ||
|   | 4129ac157a | ||
|   | 30cb791e93 | ||
|   | 1c57983bfd | ||
|   | bdf08c8a54 | ||
|   | 0513ff8b4e | ||
|   | 62fe3bfb54 | ||
|   | 38a1d6693a | ||
|   | d2eec3a9e4 | ||
|   | 1de774fa3d | ||
|   | ed902658a9 | ||
|   | acdcd7c623 | ||
|   | b0344e07c4 | ||
|   | 9a6ce1e867 | ||
|   | 22a6bd6b22 | ||
|   | 38e6f3f776 | 
							
								
								
									
										6
									
								
								.github/workflows/storybook.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										6
									
								
								.github/workflows/storybook.yml
									
									
									
									
										vendored
									
									
								
							| @@ -86,7 +86,11 @@ jobs: | ||||
|         if [ "$CHROMATIC_PARAMETER" = " --skip" ]; then | ||||
|           echo "skip=true" >> $GITHUB_OUTPUT | ||||
|         fi | ||||
|         pnpm --filter frontend chromatic --exit-once-uploaded -d storybook-static $(echo "$CHROMATIC_PARAMETER") | ||||
|         BRANCH="${{ github.event.pull_request.head.user.login }}:${{ github.event.pull_request.head.ref }}" | ||||
|         if [ "$BRANCH" = "misskey-dev:${{ github.event.pull_request.head.ref }}" ]; then | ||||
|           BRANCH="${{ github.event.pull_request.head.ref }}" | ||||
|         fi | ||||
|         pnpm --filter frontend chromatic --exit-once-uploaded -d storybook-static --branch-name $BRANCH $(echo "$CHROMATIC_PARAMETER") | ||||
|       env: | ||||
|         CHROMATIC_PROJECT_TOKEN: ${{ secrets.CHROMATIC_PROJECT_TOKEN }} | ||||
|     - name: Notify that Chromatic detects changes | ||||
|   | ||||
| @@ -29,11 +29,17 @@ | ||||
| - AiScriptを0.13.3に更新 | ||||
| - Deck UIを使用している場合、`/`以外にアクセスした際にZen UIで表示するように | ||||
| 	- メインカラムを設置していない場合の問題を解決 | ||||
| - ハッシュタグのノート一覧ページから、そのハッシュタグで投稿するボタンを追加 | ||||
| - アカウント初期設定ウィザードに戻るボタンを追加 | ||||
| - アカウントの初期設定ウィザードにあとでボタンを追加 | ||||
| - Fix: URLプレビューで情報が取得できなかった際の挙動を修正 | ||||
| - Fix: Safari、Firefoxでの新規登録時、パスワードマネージャーにメールアドレスが登録されていた挙動を修正 | ||||
| - fix:ロールタイムラインが無効でも投稿が流れてしまう問題の修正 | ||||
| - fix:ロールタイムラインにて全ての投稿が流れてしまう問題の修正 | ||||
|  | ||||
| ### Server | ||||
| - Fix: お知らせの画像URLを空にできない問題を修正 | ||||
|  | ||||
| ## 13.12.2 | ||||
|  | ||||
| ## NOTE | ||||
|   | ||||
| @@ -169,20 +169,25 @@ describe('After user signed in', () => { | ||||
| 		cy.get('[data-cy-user-setup-user-description] textarea').type('ほげ'); | ||||
| 		// TODO: アイコン設定テスト | ||||
|  | ||||
| 		cy.get('[data-cy-user-setup-back]').click(); | ||||
| 		cy.get('[data-cy-user-setup-continue]').click(); | ||||
|  | ||||
| 		// プライバシー設定 | ||||
|  | ||||
| 		cy.get('[data-cy-user-setup-back]').click(); | ||||
| 		cy.get('[data-cy-user-setup-continue]').click(); | ||||
|  | ||||
| 		// フォローはスキップ | ||||
|  | ||||
| 		cy.get('[data-cy-user-setup-back]').click(); | ||||
| 		cy.get('[data-cy-user-setup-continue]').click(); | ||||
|  | ||||
| 		// プッシュ通知設定はスキップ | ||||
|  | ||||
| 		cy.get('[data-cy-user-setup-back]').click(); | ||||
| 		cy.get('[data-cy-user-setup-continue]').click(); | ||||
|  | ||||
| 		cy.get('[data-cy-user-setup-back]').click(); | ||||
| 		cy.get('[data-cy-user-setup-continue]').click(); | ||||
|   }); | ||||
| }); | ||||
|   | ||||
							
								
								
									
										72
									
								
								locales/generateDTS.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										72
									
								
								locales/generateDTS.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,72 @@ | ||||
| const fs = require('fs'); | ||||
| const yaml = require('js-yaml'); | ||||
| const ts = require('typescript'); | ||||
|  | ||||
| function createMembers(record) { | ||||
| 	return Object.entries(record) | ||||
| 		.map(([k, v]) => ts.factory.createPropertySignature( | ||||
| 			undefined, | ||||
| 			ts.factory.createStringLiteral(k), | ||||
| 			undefined, | ||||
| 			typeof v === 'string' | ||||
| 				? ts.factory.createKeywordTypeNode(ts.SyntaxKind.StringKeyword) | ||||
| 				: ts.factory.createTypeLiteralNode(createMembers(v)), | ||||
| 		)); | ||||
| } | ||||
|  | ||||
| module.exports = function generateDTS() { | ||||
| 	const locale = yaml.load(fs.readFileSync(`${__dirname}/ja-JP.yml`, 'utf-8')); | ||||
| 	const members = createMembers(locale); | ||||
| 	const elements = [ | ||||
| 		ts.factory.createInterfaceDeclaration( | ||||
| 			[ts.factory.createToken(ts.SyntaxKind.ExportKeyword)], | ||||
| 			ts.factory.createIdentifier('Locale'), | ||||
| 			undefined, | ||||
| 			undefined, | ||||
| 			members, | ||||
| 		), | ||||
| 		ts.factory.createVariableStatement( | ||||
| 			[ts.factory.createToken(ts.SyntaxKind.DeclareKeyword)], | ||||
| 			ts.factory.createVariableDeclarationList( | ||||
| 				[ts.factory.createVariableDeclaration( | ||||
| 					ts.factory.createIdentifier('locales'), | ||||
| 					undefined, | ||||
| 					ts.factory.createTypeLiteralNode([ts.factory.createIndexSignature( | ||||
| 						undefined, | ||||
| 						[ts.factory.createParameterDeclaration( | ||||
| 							undefined, | ||||
| 							undefined, | ||||
| 							ts.factory.createIdentifier('lang'), | ||||
| 							undefined, | ||||
| 							ts.factory.createKeywordTypeNode(ts.SyntaxKind.StringKeyword), | ||||
| 							undefined, | ||||
| 						)], | ||||
| 						ts.factory.createTypeReferenceNode( | ||||
| 							ts.factory.createIdentifier('Locale'), | ||||
| 							undefined, | ||||
| 						), | ||||
| 					)]), | ||||
| 					undefined, | ||||
| 				)], | ||||
| 				ts.NodeFlags.Const | ts.NodeFlags.Ambient | ts.NodeFlags.ContextFlags, | ||||
| 			), | ||||
| 		), | ||||
| 		ts.factory.createExportAssignment( | ||||
| 			undefined, | ||||
| 			true, | ||||
| 			ts.factory.createIdentifier('locales'), | ||||
| 		), | ||||
| 	]; | ||||
| 	const printed = ts.createPrinter({ | ||||
| 		newLine: ts.NewLineKind.LineFeed, | ||||
| 	}).printList( | ||||
| 		ts.ListFormat.MultiLine, | ||||
| 		ts.factory.createNodeArray(elements), | ||||
| 		ts.createSourceFile('index.d.ts', '', ts.ScriptTarget.ESNext, true, ts.ScriptKind.TS), | ||||
| 	); | ||||
|  | ||||
| 	fs.writeFileSync(`${__dirname}/index.d.ts`, `/* eslint-disable */ | ||||
| // This file is generated by locales/generateDTS.js | ||||
| // Do not edit this file directly. | ||||
| ${printed}`, 'utf-8'); | ||||
| } | ||||
							
								
								
									
										2149
									
								
								locales/index.d.ts
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2149
									
								
								locales/index.d.ts
									
									
									
									
										vendored
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -792,6 +792,7 @@ noMaintainerInformationWarning: "管理者情報が設定されていません | ||||
| noBotProtectionWarning: "Botプロテクションが設定されていません。" | ||||
| configure: "設定する" | ||||
| postToGallery: "ギャラリーへ投稿" | ||||
| postToHashtag: "このハッシュタグで投稿" | ||||
| gallery: "ギャラリー" | ||||
| recentPosts: "最近の投稿" | ||||
| popularPosts: "人気の投稿" | ||||
| @@ -1057,6 +1058,8 @@ rolesThatCanBeUsedThisEmojiAsReactionEmptyDescription: "ロールの指定が一 | ||||
| rolesThatCanBeUsedThisEmojiAsReactionPublicRoleWarn: "ロールは公開ロールである必要があります。" | ||||
| cancelReactionConfirm: "リアクションを取り消しますか?" | ||||
| changeReactionConfirm: "リアクションを変更しますか?" | ||||
| later: "あとで" | ||||
| goToMisskey: "Misskeyへ" | ||||
|  | ||||
| _initialAccountSetting: | ||||
|   accountCreated: "アカウントの作成が完了しました!" | ||||
| @@ -1072,6 +1075,7 @@ _initialAccountSetting: | ||||
|   haveFun: "{name}をお楽しみください!" | ||||
|   ifYouNeedLearnMore: "{name}(Misskey)の使い方などを詳しく知るには{link}をご覧ください。" | ||||
|   skipAreYouSure: "初期設定をスキップしますか?" | ||||
|   laterAreYouSure: "初期設定をあとでやり直しますか?" | ||||
|  | ||||
| _serverRules: | ||||
|   description: "新規登録前に表示する、サーバーの簡潔なルールを設定します。内容は利用規約の要約とすることを推奨します。" | ||||
|   | ||||
| @@ -59,7 +59,7 @@ | ||||
| 		"@typescript-eslint/eslint-plugin": "5.59.5", | ||||
| 		"@typescript-eslint/parser": "5.59.5", | ||||
| 		"cross-env": "7.0.3", | ||||
| 		"cypress": "12.12.0", | ||||
| 		"cypress": "12.13.0", | ||||
| 		"eslint": "8.40.0", | ||||
| 		"start-server-and-test": "2.0.0" | ||||
| 	}, | ||||
|   | ||||
| @@ -63,9 +63,9 @@ | ||||
| 		"@fastify/multipart": "7.6.0", | ||||
| 		"@fastify/static": "6.10.1", | ||||
| 		"@fastify/view": "7.4.1", | ||||
| 		"@nestjs/common": "9.4.1", | ||||
| 		"@nestjs/core": "9.4.1", | ||||
| 		"@nestjs/testing": "9.4.1", | ||||
| 		"@nestjs/common": "9.4.2", | ||||
| 		"@nestjs/core": "9.4.2", | ||||
| 		"@nestjs/testing": "9.4.2", | ||||
| 		"@peertube/http-signature": "1.7.0", | ||||
| 		"@sinonjs/fake-timers": "10.0.2", | ||||
| 		"@swc/cli": "0.1.62", | ||||
| @@ -103,8 +103,8 @@ | ||||
| 		"jsdom": "21.1.1", | ||||
| 		"json5": "2.2.3", | ||||
| 		"jsonld": "8.1.1", | ||||
| 		"meilisearch": "0.32.3", | ||||
| 		"jsrsasign": "10.8.6", | ||||
| 		"meilisearch": "0.32.4", | ||||
| 		"mfm-js": "0.23.3", | ||||
| 		"mime-types": "2.1.35", | ||||
| 		"misskey-js": "workspace:*", | ||||
| @@ -179,11 +179,11 @@ | ||||
| 		"@types/jsonld": "1.5.8", | ||||
| 		"@types/jsrsasign": "10.5.8", | ||||
| 		"@types/mime-types": "2.1.1", | ||||
| 		"@types/node": "20.2.1", | ||||
| 		"@types/node": "20.2.3", | ||||
| 		"@types/node-fetch": "3.0.3", | ||||
| 		"@types/nodemailer": "6.4.8", | ||||
| 		"@types/oauth": "0.9.1", | ||||
| 		"@types/pg": "8.6.6", | ||||
| 		"@types/pg": "8.10.1", | ||||
| 		"@types/pug": "2.0.6", | ||||
| 		"@types/punycode": "2.1.0", | ||||
| 		"@types/qrcode": "1.5.0", | ||||
| @@ -197,7 +197,7 @@ | ||||
| 		"@types/sinonjs__fake-timers": "8.1.2", | ||||
| 		"@types/tinycolor2": "1.4.3", | ||||
| 		"@types/tmp": "0.2.3", | ||||
| 		"@types/unzipper": "0.10.5", | ||||
| 		"@types/unzipper": "0.10.6", | ||||
| 		"@types/uuid": "9.0.1", | ||||
| 		"@types/vary": "1.1.0", | ||||
| 		"@types/web-push": "3.3.2", | ||||
|   | ||||
| @@ -4,7 +4,7 @@ import * as Redis from 'ioredis'; | ||||
| import { DataSource } from 'typeorm'; | ||||
| import { MeiliSearch } from 'meilisearch'; | ||||
| import { DI } from './di-symbols.js'; | ||||
| import { loadConfig } from './config.js'; | ||||
| import { Config, loadConfig } from './config.js'; | ||||
| import { createPostgresDataSource } from './postgres.js'; | ||||
| import { RepositoryModule } from './models/RepositoryModule.js'; | ||||
| import type { Provider, OnApplicationShutdown } from '@nestjs/common'; | ||||
| @@ -25,7 +25,7 @@ const $db: Provider = { | ||||
|  | ||||
| const $meilisearch: Provider = { | ||||
| 	provide: DI.meilisearch, | ||||
| 	useFactory: (config) => { | ||||
| 	useFactory: (config: Config) => { | ||||
| 		if (config.meilisearch) { | ||||
| 			return new MeiliSearch({ | ||||
| 				host: `${config.meilisearch.ssl ? 'https' : 'http' }://${config.meilisearch.host}:${config.meilisearch.port}`, | ||||
| @@ -40,7 +40,7 @@ const $meilisearch: Provider = { | ||||
|  | ||||
| const $redis: Provider = { | ||||
| 	provide: DI.redis, | ||||
| 	useFactory: (config) => { | ||||
| 	useFactory: (config: Config) => { | ||||
| 		return new Redis.Redis({ | ||||
| 			port: config.redis.port, | ||||
| 			host: config.redis.host, | ||||
| @@ -55,7 +55,7 @@ const $redis: Provider = { | ||||
|  | ||||
| const $redisForPub: Provider = { | ||||
| 	provide: DI.redisForPub, | ||||
| 	useFactory: (config) => { | ||||
| 	useFactory: (config: Config) => { | ||||
| 		const redis = new Redis.Redis({ | ||||
| 			port: config.redisForPubsub.port, | ||||
| 			host: config.redisForPubsub.host, | ||||
| @@ -71,7 +71,7 @@ const $redisForPub: Provider = { | ||||
|  | ||||
| const $redisForSub: Provider = { | ||||
| 	provide: DI.redisForSub, | ||||
| 	useFactory: (config) => { | ||||
| 	useFactory: (config: Config) => { | ||||
| 		const redis = new Redis.Redis({ | ||||
| 			port: config.redisForPubsub.port, | ||||
| 			host: config.redisForPubsub.host, | ||||
|   | ||||
| @@ -25,7 +25,7 @@ export const paramDef = { | ||||
| 		id: { type: 'string', format: 'misskey:id' }, | ||||
| 		title: { type: 'string', minLength: 1 }, | ||||
| 		text: { type: 'string', minLength: 1 }, | ||||
| 		imageUrl: { type: 'string', nullable: true, minLength: 1 }, | ||||
| 		imageUrl: { type: 'string', nullable: true, minLength: 0 }, | ||||
| 	}, | ||||
| 	required: ['id', 'title', 'text', 'imageUrl'], | ||||
| } as const; | ||||
| @@ -46,7 +46,8 @@ export default class extends Endpoint<typeof meta, typeof paramDef> { | ||||
| 				updatedAt: new Date(), | ||||
| 				title: ps.title, | ||||
| 				text: ps.text, | ||||
| 				imageUrl: ps.imageUrl, | ||||
| 				/* eslint-disable-next-line @typescript-eslint/prefer-nullish-coalescing -- 空の文字列の場合、nullを渡すようにするため */ | ||||
| 				imageUrl: ps.imageUrl || null,  | ||||
| 			}); | ||||
| 		}); | ||||
| 	} | ||||
|   | ||||
| @@ -160,37 +160,41 @@ | ||||
| 				<path d="M12 9v2m0 4v.01"></path> | ||||
| 				<path d="M5 19h14a2 2 0 0 0 1.84 -2.75l-7.1 -12.25a2 2 0 0 0 -3.5 0l-7.1 12.25a2 2 0 0 0 1.75 2.75"></path> | ||||
| 			</svg> | ||||
| 			<h1>An error has occurred!</h1> | ||||
| 			<button class="button-big" onclick="location.reload();"> | ||||
| 				<span class="button-label-big">Refresh</span> | ||||
| 			<h1>Failed to load<br>読み込みに失敗しました</h1> | ||||
| 			<button class="button-big" onclick="location.reload(true);"> | ||||
| 				<span class="button-label-big">Reload / リロード</span> | ||||
| 			</button> | ||||
| 			<p class="dont-worry">Don't worry, it's (probably) not your fault.</p> | ||||
| 			<p>If the problem persists after refreshing, please contact your instance's administrator.<br>You may also try the following options:</p> | ||||
| 			<p>Update your os and browser.</p> | ||||
| 			<p>Disable an adblocker.</p> | ||||
| 			<a href="/flush"> | ||||
| 				<button class="button-small"> | ||||
| 					<span class="button-label-small">Clear preferences and cache</span> | ||||
| 				</button> | ||||
| 			</a> | ||||
| 			<br> | ||||
| 			<a href="/cli"> | ||||
| 				<button class="button-small"> | ||||
| 					<span class="button-label-small">Start the simple client</span> | ||||
| 				</button> | ||||
| 			</a> | ||||
| 			<br> | ||||
| 			<a href="/bios"> | ||||
| 				<button class="button-small"> | ||||
| 					<span class="button-label-small">Start the repair tool</span> | ||||
| 				</button> | ||||
| 			</a> | ||||
| 			<p><b>The following actions may solve the problem. / 以下を行うと解決する可能性があります。</b></p> | ||||
| 			<p>Clear the browser cache / ブラウザのキャッシュをクリアする</p> | ||||
| 			<p>Update your os and browser / ブラウザおよびOSを最新バージョンに更新する</p> | ||||
| 			<p>Disable an adblocker / アドブロッカーを無効にする</p> | ||||
| 			<details style="color: #86b300;"> | ||||
| 				<summary>Other options / その他のオプション</summary> | ||||
| 				<a href="/flush"> | ||||
| 					<button class="button-small"> | ||||
| 						<span class="button-label-small">Clear preferences and cache</span> | ||||
| 					</button> | ||||
| 				</a> | ||||
| 				<br> | ||||
| 				<a href="/cli"> | ||||
| 					<button class="button-small"> | ||||
| 						<span class="button-label-small">Start the simple client</span> | ||||
| 					</button> | ||||
| 				</a> | ||||
| 				<br> | ||||
| 				<a href="/bios"> | ||||
| 					<button class="button-small"> | ||||
| 						<span class="button-label-small">Start the repair tool</span> | ||||
| 					</button> | ||||
| 				</a> | ||||
| 			</details> | ||||
| 			<br> | ||||
| 			<div id="errors"></div> | ||||
| 			`; | ||||
| 			errorsElement = document.getElementById('errors'); | ||||
| 		} | ||||
| 		const detailsElement = document.createElement('details'); | ||||
| 		detailsElement.id = 'errorInfo'; | ||||
| 		detailsElement.innerHTML = ` | ||||
| 		<br> | ||||
| 		<summary> | ||||
| @@ -247,7 +251,7 @@ | ||||
| 		.button-label-big { | ||||
| 			color: #222; | ||||
| 			font-weight: bold; | ||||
| 			font-size: 20px; | ||||
| 			font-size: 1.2em; | ||||
| 			padding: 12px; | ||||
| 		} | ||||
|  | ||||
| @@ -267,11 +271,6 @@ | ||||
| 			font-size: 16px; | ||||
| 		} | ||||
|  | ||||
| 		.dont-worry, | ||||
| 		#msg { | ||||
| 			font-size: 18px; | ||||
| 		} | ||||
|  | ||||
| 		.icon-warning { | ||||
| 			color: #dec340; | ||||
| 			height: 4rem; | ||||
| @@ -279,14 +278,15 @@ | ||||
| 		} | ||||
|  | ||||
| 		h1 { | ||||
| 			font-size: 32px; | ||||
| 			font-size: 1.5em; | ||||
| 			margin: 1em; | ||||
| 		} | ||||
|  | ||||
| 		code { | ||||
| 			font-family: Fira, FiraCode, monospace; | ||||
| 		} | ||||
|  | ||||
| 		details { | ||||
| 		#errorInfo { | ||||
| 			background: #333; | ||||
| 			margin-bottom: 2rem; | ||||
| 			padding: 0.5rem 1rem; | ||||
| @@ -296,16 +296,16 @@ | ||||
| 			margin: auto; | ||||
| 		} | ||||
|  | ||||
| 		summary { | ||||
| 		#errorInfo summary { | ||||
| 			cursor: pointer; | ||||
| 		} | ||||
|  | ||||
| 		summary > * { | ||||
| 		#errorInfo summary > * { | ||||
| 			display: inline; | ||||
| 		} | ||||
|  | ||||
| 		@media screen and (max-width: 500px) { | ||||
| 			details { | ||||
| 			#errorInfo { | ||||
| 				width: 50%; | ||||
| 			} | ||||
| 		`) | ||||
|   | ||||
| @@ -22,7 +22,7 @@ | ||||
| 		"@syuilo/aiscript": "0.13.3", | ||||
| 		"@tabler/icons-webfont": "2.17.0", | ||||
| 		"@vitejs/plugin-vue": "4.2.3", | ||||
| 		"@vue-macros/reactivity-transform": "0.3.7", | ||||
| 		"@vue-macros/reactivity-transform": "0.3.8", | ||||
| 		"@vue/compiler-sfc": "3.3.4", | ||||
| 		"autosize": "6.0.1", | ||||
| 		"broadcast-channel": "4.20.2", | ||||
| @@ -53,7 +53,7 @@ | ||||
| 		"punycode": "2.3.0", | ||||
| 		"querystring": "0.2.1", | ||||
| 		"rndstr": "1.0.0", | ||||
| 		"rollup": "3.22.0", | ||||
| 		"rollup": "3.23.0", | ||||
| 		"s-age": "1.1.2", | ||||
| 		"sanitize-html": "2.10.0", | ||||
| 		"sass": "1.62.1", | ||||
| @@ -77,37 +77,37 @@ | ||||
| 		"vuedraggable": "next" | ||||
| 	}, | ||||
| 	"devDependencies": { | ||||
| 		"@storybook/addon-actions": "7.0.12", | ||||
| 		"@storybook/addon-essentials": "7.0.12", | ||||
| 		"@storybook/addon-interactions": "7.0.12", | ||||
| 		"@storybook/addon-links": "7.0.12", | ||||
| 		"@storybook/addon-storysource": "7.0.12", | ||||
| 		"@storybook/addons": "7.0.12", | ||||
| 		"@storybook/blocks": "7.0.12", | ||||
| 		"@storybook/core-events": "7.0.12", | ||||
| 		"@storybook/addon-actions": "7.0.15", | ||||
| 		"@storybook/addon-essentials": "7.0.15", | ||||
| 		"@storybook/addon-interactions": "7.0.15", | ||||
| 		"@storybook/addon-links": "7.0.15", | ||||
| 		"@storybook/addon-storysource": "7.0.15", | ||||
| 		"@storybook/addons": "7.0.15", | ||||
| 		"@storybook/blocks": "7.0.15", | ||||
| 		"@storybook/core-events": "7.0.15", | ||||
| 		"@storybook/jest": "0.1.0", | ||||
| 		"@storybook/manager-api": "7.0.12", | ||||
| 		"@storybook/preview-api": "7.0.12", | ||||
| 		"@storybook/react": "7.0.12", | ||||
| 		"@storybook/react-vite": "7.0.12", | ||||
| 		"@storybook/manager-api": "7.0.15", | ||||
| 		"@storybook/preview-api": "7.0.15", | ||||
| 		"@storybook/react": "7.0.15", | ||||
| 		"@storybook/react-vite": "7.0.15", | ||||
| 		"@storybook/testing-library": "0.1.0", | ||||
| 		"@storybook/theming": "7.0.12", | ||||
| 		"@storybook/types": "7.0.12", | ||||
| 		"@storybook/vue3": "7.0.12", | ||||
| 		"@storybook/vue3-vite": "7.0.12", | ||||
| 		"@storybook/theming": "7.0.15", | ||||
| 		"@storybook/types": "7.0.15", | ||||
| 		"@storybook/vue3": "7.0.15", | ||||
| 		"@storybook/vue3-vite": "7.0.15", | ||||
| 		"@testing-library/jest-dom": "5.16.5", | ||||
| 		"@testing-library/vue": "7.0.0", | ||||
| 		"@types/escape-regexp": "0.0.1", | ||||
| 		"@types/estree": "1.0.1", | ||||
| 		"@types/gulp": "4.0.10", | ||||
| 		"@types/gulp-rename": "2.0.2", | ||||
| 		"@types/matter-js": "0.18.3", | ||||
| 		"@types/matter-js": "0.18.4", | ||||
| 		"@types/micromatch": "4.0.2", | ||||
| 		"@types/node": "20.2.1", | ||||
| 		"@types/node": "20.2.3", | ||||
| 		"@types/punycode": "2.1.0", | ||||
| 		"@types/sanitize-html": "2.9.0", | ||||
| 		"@types/seedrandom": "3.0.5", | ||||
| 		"@types/testing-library__jest-dom": "^5.14.5", | ||||
| 		"@types/testing-library__jest-dom": "^5.14.6", | ||||
| 		"@types/throttle-debounce": "5.0.0", | ||||
| 		"@types/tinycolor2": "1.4.3", | ||||
| 		"@types/uuid": "9.0.1", | ||||
| @@ -117,13 +117,13 @@ | ||||
| 		"@typescript-eslint/parser": "5.59.5", | ||||
| 		"@vitest/coverage-c8": "0.31.1", | ||||
| 		"@vue/runtime-core": "3.3.4", | ||||
| 		"astring": "1.8.4", | ||||
| 		"astring": "1.8.5", | ||||
| 		"chokidar-cli": "3.0.0", | ||||
| 		"cross-env": "7.0.3", | ||||
| 		"cypress": "12.12.0", | ||||
| 		"cypress": "12.13.0", | ||||
| 		"eslint": "8.40.0", | ||||
| 		"eslint-plugin-import": "2.27.5", | ||||
| 		"eslint-plugin-vue": "9.13.0", | ||||
| 		"eslint-plugin-vue": "9.14.0", | ||||
| 		"fast-glob": "3.2.12", | ||||
| 		"happy-dom": "9.19.2", | ||||
| 		"micromatch": "3.1.10", | ||||
| @@ -133,7 +133,7 @@ | ||||
| 		"react": "18.2.0", | ||||
| 		"react-dom": "18.2.0", | ||||
| 		"start-server-and-test": "2.0.0", | ||||
| 		"storybook": "7.0.12", | ||||
| 		"storybook": "7.0.15", | ||||
| 		"storybook-addon-misskey-theme": "github:misskey-dev/storybook-addon-misskey-theme", | ||||
| 		"summaly": "github:misskey-dev/summaly", | ||||
| 		"vite-plugin-turbosnap": "1.0.2", | ||||
|   | ||||
| @@ -5,7 +5,9 @@ import '@/style.scss'; | ||||
| import { mainBoot } from './boot/main-boot'; | ||||
| import { subBoot } from './boot/sub-boot'; | ||||
|  | ||||
| if (['/share', '/auth', '/miauth'].includes(location.pathname)) { | ||||
| const subBootPaths = ['/share', '/auth', '/miauth', '/signup-complete']; | ||||
|  | ||||
| if (subBootPaths.some(i => location.pathname === i || location.pathname.startsWith(i + '/'))) { | ||||
| 	subBoot(); | ||||
| } else { | ||||
| 	mainBoot(); | ||||
|   | ||||
| @@ -16,7 +16,7 @@ import { initializeSw } from '@/scripts/initialize-sw'; | ||||
|  | ||||
| export async function mainBoot() { | ||||
| 	const { isClientUpdated } = await common(() => createApp( | ||||
| 		new URLSearchParams(window.location.search).has('zen') || (ui === 'deck' && location.pathname !== '/') ? defineAsyncComponent(() => import('@/ui/zen.vue')) : | ||||
| 		new URLSearchParams(window.location.search).has('zen') ? defineAsyncComponent(() => import('@/ui/zen.vue')) : | ||||
| 		!$i ? defineAsyncComponent(() => import('@/ui/visitor.vue')) : | ||||
| 		ui === 'deck' ? defineAsyncComponent(() => import('@/ui/deck.vue')) : | ||||
| 		ui === 'classic' ? defineAsyncComponent(() => import('@/ui/classic.vue')) : | ||||
|   | ||||
| @@ -2,7 +2,7 @@ | ||||
| <button | ||||
| 	v-if="!link" | ||||
| 	ref="el" class="_button" | ||||
| 	:class="[$style.root, { [$style.inline]: inline, [$style.primary]: primary, [$style.gradate]: gradate, [$style.danger]: danger, [$style.rounded]: rounded, [$style.full]: full, [$style.small]: small, [$style.large]: large, [$style.asLike]: asLike }]" | ||||
| 	:class="[$style.root, { [$style.inline]: inline, [$style.primary]: primary, [$style.gradate]: gradate, [$style.danger]: danger, [$style.rounded]: rounded, [$style.full]: full, [$style.small]: small, [$style.large]: large, [$style.transparent]: transparent, [$style.asLike]: asLike }]" | ||||
| 	:type="type" | ||||
| 	@click="emit('click', $event)" | ||||
| 	@mousedown="onMousedown" | ||||
| @@ -14,7 +14,7 @@ | ||||
| </button> | ||||
| <MkA | ||||
| 	v-else class="_button" | ||||
| 	:class="[$style.root, { [$style.inline]: inline, [$style.primary]: primary, [$style.gradate]: gradate, [$style.danger]: danger, [$style.rounded]: rounded, [$style.full]: full, [$style.small]: small, [$style.large]: large, [$style.asLike]: asLike }]" | ||||
| 	:class="[$style.root, { [$style.inline]: inline, [$style.primary]: primary, [$style.gradate]: gradate, [$style.danger]: danger, [$style.rounded]: rounded, [$style.full]: full, [$style.small]: small, [$style.large]: large, [$style.transparent]: transparent, [$style.asLike]: asLike }]" | ||||
| 	:to="to" | ||||
| 	@mousedown="onMousedown" | ||||
| > | ||||
| @@ -44,6 +44,7 @@ const props = defineProps<{ | ||||
| 	full?: boolean; | ||||
| 	small?: boolean; | ||||
| 	large?: boolean; | ||||
| 	transparent?: boolean; | ||||
| 	asLike?: boolean; | ||||
| }>(); | ||||
|  | ||||
| @@ -194,6 +195,10 @@ function onMousedown(evt: MouseEvent): void { | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	&.transparent { | ||||
| 		background: transparent; | ||||
| 	} | ||||
|  | ||||
| 	&.gradate { | ||||
| 		font-weight: bold; | ||||
| 		color: var(--fgOnAccent) !important; | ||||
|   | ||||
| @@ -14,8 +14,8 @@ | ||||
| > | ||||
| 	<MkEmojiPicker | ||||
| 		ref="picker" | ||||
| 		class="ryghynhb _popup _shadow" | ||||
| 		:class="{ drawer: type === 'drawer' }" | ||||
| 		class="_popup _shadow" | ||||
| 		:class="{ [$style.drawer]: type === 'drawer' }" | ||||
| 		:showPinned="showPinned" | ||||
| 		:asReactionPicker="asReactionPicker" | ||||
| 		:asDrawer="type === 'drawer'" | ||||
| @@ -67,12 +67,10 @@ function opening() { | ||||
| } | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" scoped> | ||||
| .ryghynhb { | ||||
| 	&.drawer { | ||||
| 		border-radius: 24px; | ||||
| 		border-bottom-right-radius: 0; | ||||
| 		border-bottom-left-radius: 0; | ||||
| 	} | ||||
| <style lang="scss" module> | ||||
| .drawer { | ||||
| 	border-radius: 24px; | ||||
| 	border-bottom-right-radius: 0; | ||||
| 	border-bottom-left-radius: 0; | ||||
| } | ||||
| </style> | ||||
|   | ||||
| @@ -8,27 +8,28 @@ | ||||
| > | ||||
| 	<template #header>{{ i18n.ts.forgotPassword }}</template> | ||||
|  | ||||
| 	<form v-if="instance.enableEmail" class="bafeceda" @submit.prevent="onSubmit"> | ||||
| 		<div class="main _gaps_m"> | ||||
| 			<MkInput v-model="username" type="text" pattern="^[a-zA-Z0-9_]+$" :spellcheck="false" autofocus required> | ||||
| 				<template #label>{{ i18n.ts.username }}</template> | ||||
| 				<template #prefix>@</template> | ||||
| 			</MkInput> | ||||
| 	<MkSpacer :marginMin="20" :marginMax="28"> | ||||
| 		<form v-if="instance.enableEmail" @submit.prevent="onSubmit"> | ||||
| 			<div class="_gaps_m"> | ||||
| 				<MkInput v-model="username" type="text" pattern="^[a-zA-Z0-9_]+$" :spellcheck="false" autofocus required> | ||||
| 					<template #label>{{ i18n.ts.username }}</template> | ||||
| 					<template #prefix>@</template> | ||||
| 				</MkInput> | ||||
|  | ||||
| 			<MkInput v-model="email" type="email" :spellcheck="false" required> | ||||
| 				<template #label>{{ i18n.ts.emailAddress }}</template> | ||||
| 				<template #caption>{{ i18n.ts._forgotPassword.enterEmail }}</template> | ||||
| 			</MkInput> | ||||
| 				<MkInput v-model="email" type="email" :spellcheck="false" required> | ||||
| 					<template #label>{{ i18n.ts.emailAddress }}</template> | ||||
| 					<template #caption>{{ i18n.ts._forgotPassword.enterEmail }}</template> | ||||
| 				</MkInput> | ||||
|  | ||||
| 			<MkButton type="submit" :disabled="processing" primary style="margin: 0 auto;">{{ i18n.ts.send }}</MkButton> | ||||
| 				<MkButton type="submit" rounded :disabled="processing" primary style="margin: 0 auto;">{{ i18n.ts.send }}</MkButton> | ||||
|  | ||||
| 				<MkInfo>{{ i18n.ts._forgotPassword.ifNoEmail }}</MkInfo> | ||||
| 			</div> | ||||
| 		</form> | ||||
| 		<div v-else> | ||||
| 			{{ i18n.ts._forgotPassword.contactAdmin }} | ||||
| 		</div> | ||||
| 		<div class="sub"> | ||||
| 			<MkA to="/about" class="_link">{{ i18n.ts._forgotPassword.ifNoEmail }}</MkA> | ||||
| 		</div> | ||||
| 	</form> | ||||
| 	<div v-else class="bafecedb"> | ||||
| 		{{ i18n.ts._forgotPassword.contactAdmin }} | ||||
| 	</div> | ||||
| 	</MkSpacer> | ||||
| </MkModalWindow> | ||||
| </template> | ||||
|  | ||||
| @@ -37,6 +38,7 @@ import { } from 'vue'; | ||||
| import MkModalWindow from '@/components/MkModalWindow.vue'; | ||||
| import MkButton from '@/components/MkButton.vue'; | ||||
| import MkInput from '@/components/MkInput.vue'; | ||||
| import MkInfo from '@/components/MkInfo.vue'; | ||||
| import * as os from '@/os'; | ||||
| import { instance } from '@/instance'; | ||||
| import { i18n } from '@/i18n'; | ||||
| @@ -62,20 +64,3 @@ async function onSubmit() { | ||||
| 	dialog.close(); | ||||
| } | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" scoped> | ||||
| .bafeceda { | ||||
| 	> .main { | ||||
| 		padding: 24px; | ||||
| 	} | ||||
|  | ||||
| 	> .sub { | ||||
| 		border-top: solid 0.5px var(--divider); | ||||
| 		padding: 24px; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| .bafecedb { | ||||
| 	padding: 24px; | ||||
| } | ||||
| </style> | ||||
|   | ||||
| @@ -1,78 +0,0 @@ | ||||
| <template> | ||||
| <MkModal ref="modal" :zPriority="'middle'" @click="modal.close()" @closed="emit('closed')"> | ||||
| 	<div class="xubzgfga"> | ||||
| 		<header>{{ image.name }}</header> | ||||
| 		<img :src="image.url" :alt="image.comment" :title="image.comment" @click="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" setup> | ||||
| import { } from 'vue'; | ||||
| import * as misskey from 'misskey-js'; | ||||
| import bytes from '@/filters/bytes'; | ||||
| import number from '@/filters/number'; | ||||
| import MkModal from '@/components/MkModal.vue'; | ||||
|  | ||||
| const props = withDefaults(defineProps<{ | ||||
| 	image: misskey.entities.DriveFile; | ||||
| }>(), { | ||||
| }); | ||||
|  | ||||
| const emit = defineEmits<{ | ||||
| 	(ev: 'closed'): void; | ||||
| }>(); | ||||
|  | ||||
| const modal = $shallowRef<InstanceType<typeof MkModal>>(); | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" scoped> | ||||
| .xubzgfga { | ||||
| 	margin: auto; | ||||
| 	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,182 +0,0 @@ | ||||
| <template> | ||||
| <MkModal ref="modal" @click="$emit('click')" @closed="$emit('closed')"> | ||||
| 	<div ref="rootEl" class="hrmcaedk" :style="{ width: `${width}px`, height: (height ? `min(${height}px, 100%)` : '100%') }"> | ||||
| 		<div class="header" @contextmenu="onContextmenu"> | ||||
| 			<button v-if="history.length > 0" v-tooltip="i18n.ts.goBack" class="_button" @click="back()"><i class="ti ti-arrow-left"></i></button> | ||||
| 			<span v-else style="display: inline-block; width: 20px"></span> | ||||
| 			<span v-if="pageMetadata?.value" class="title"> | ||||
| 				<i v-if="pageMetadata?.value.icon" class="icon" :class="pageMetadata?.value.icon"></i> | ||||
| 				<span>{{ pageMetadata?.value.title }}</span> | ||||
| 			</span> | ||||
| 			<button class="_button" @click="$refs.modal.close()"><i class="ti ti-x"></i></button> | ||||
| 		</div> | ||||
| 		<div class="body" style="container-type: inline-size;"> | ||||
| 			<MkStickyContainer> | ||||
| 				<template #header><MkPageHeader v-if="pageMetadata?.value && !pageMetadata?.value.hideHeader" :info="pageMetadata?.value"/></template> | ||||
| 				<RouterView :router="router"/> | ||||
| 			</MkStickyContainer> | ||||
| 		</div> | ||||
| 	</div> | ||||
| </MkModal> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts" setup> | ||||
| import { ComputedRef, provide } from 'vue'; | ||||
| import MkModal from '@/components/MkModal.vue'; | ||||
| import { popout as _popout } from '@/scripts/popout'; | ||||
| import copyToClipboard from '@/scripts/copy-to-clipboard'; | ||||
| import { url } from '@/config'; | ||||
| import * as os from '@/os'; | ||||
| import { mainRouter, routes } from '@/router'; | ||||
| import { i18n } from '@/i18n'; | ||||
| import { PageMetadata, provideMetadataReceiver } from '@/scripts/page-metadata'; | ||||
| import { Router } from '@/nirax'; | ||||
|  | ||||
| const props = defineProps<{ | ||||
| 	initialPath: string; | ||||
| }>(); | ||||
|  | ||||
| defineEmits<{ | ||||
| 	(ev: 'closed'): void; | ||||
| 	(ev: 'click'): void; | ||||
| }>(); | ||||
|  | ||||
| const router = new Router(routes, props.initialPath); | ||||
|  | ||||
| router.addListener('push', ctx => { | ||||
| 	 | ||||
| }); | ||||
|  | ||||
| let pageMetadata = $ref<null | ComputedRef<PageMetadata>>(); | ||||
| let rootEl = $ref(); | ||||
| let modal = $shallowRef<InstanceType<typeof MkModal>>(); | ||||
| let path = $ref(props.initialPath); | ||||
| let width = $ref(860); | ||||
| let height = $ref(660); | ||||
| const history = []; | ||||
|  | ||||
| provide('router', router); | ||||
| provideMetadataReceiver((info) => { | ||||
| 	pageMetadata = info; | ||||
| }); | ||||
| provide('shouldOmitHeaderTitle', true); | ||||
| provide('shouldHeaderThin', true); | ||||
|  | ||||
| const pageUrl = $computed(() => url + path); | ||||
| const contextmenu = $computed(() => { | ||||
| 	return [{ | ||||
| 		type: 'label', | ||||
| 		text: path, | ||||
| 	}, { | ||||
| 		icon: 'ti ti-player-eject', | ||||
| 		text: i18n.ts.showInPage, | ||||
| 		action: expand, | ||||
| 	}, { | ||||
| 		icon: 'ti ti-window-maximize', | ||||
| 		text: i18n.ts.popout, | ||||
| 		action: popout, | ||||
| 	}, null, { | ||||
| 		icon: 'ti ti-external-link', | ||||
| 		text: i18n.ts.openInNewTab, | ||||
| 		action: () => { | ||||
| 			window.open(pageUrl, '_blank'); | ||||
| 			modal.close(); | ||||
| 		}, | ||||
| 	}, { | ||||
| 		icon: 'ti ti-link', | ||||
| 		text: i18n.ts.copyLink, | ||||
| 		action: () => { | ||||
| 			copyToClipboard(pageUrl); | ||||
| 		}, | ||||
| 	}]; | ||||
| }); | ||||
|  | ||||
| function navigate(path, record = true) { | ||||
| 	if (record) history.push(router.getCurrentPath()); | ||||
| 	router.push(path); | ||||
| } | ||||
|  | ||||
| function back() { | ||||
| 	navigate(history.pop(), false); | ||||
| } | ||||
|  | ||||
| function expand() { | ||||
| 	mainRouter.push(path); | ||||
| 	modal.close(); | ||||
| } | ||||
|  | ||||
| function popout() { | ||||
| 	_popout(path, rootEl); | ||||
| 	modal.close(); | ||||
| } | ||||
|  | ||||
| function onContextmenu(ev: MouseEvent) { | ||||
| 	os.contextMenu(contextmenu, ev); | ||||
| } | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" scoped> | ||||
| .hrmcaedk { | ||||
| 	margin: auto; | ||||
| 	overflow: hidden; | ||||
| 	display: flex; | ||||
| 	flex-direction: column; | ||||
| 	contain: content; | ||||
| 	border-radius: var(--radius); | ||||
|  | ||||
| 	--root-margin: 24px; | ||||
|  | ||||
| 	@media (max-width: 500px) { | ||||
| 		--root-margin: 16px; | ||||
| 	} | ||||
|  | ||||
| 	> .header { | ||||
| 		$height: 52px; | ||||
| 		$height-narrow: 42px; | ||||
| 		display: flex; | ||||
| 		flex-shrink: 0; | ||||
| 		height: $height; | ||||
| 		line-height: $height; | ||||
| 		font-weight: bold; | ||||
| 		white-space: nowrap; | ||||
| 		overflow: hidden; | ||||
| 		text-overflow: ellipsis; | ||||
| 		background: var(--windowHeader); | ||||
| 		-webkit-backdrop-filter: var(--blur, blur(15px)); | ||||
| 		backdrop-filter: var(--blur, blur(15px)); | ||||
|  | ||||
| 		> button { | ||||
| 			height: $height; | ||||
| 			width: $height; | ||||
|  | ||||
| 			&:hover { | ||||
| 				color: var(--fgHighlighted); | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		@media (max-width: 500px) { | ||||
| 			height: $height-narrow; | ||||
| 			line-height: $height-narrow; | ||||
| 			padding-left: 16px; | ||||
|  | ||||
| 			> button { | ||||
| 				height: $height-narrow; | ||||
| 				width: $height-narrow; | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		> .title { | ||||
| 			flex: 1; | ||||
|  | ||||
| 			> .icon { | ||||
| 				margin-right: 0.5em; | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	> .body { | ||||
| 		overflow: auto; | ||||
| 		background: var(--bg); | ||||
| 	} | ||||
| } | ||||
| </style> | ||||
| @@ -1,16 +1,16 @@ | ||||
| <template> | ||||
| <div v-show="props.modelValue.length != 0" class="skeikyzd"> | ||||
| 	<Sortable :modelValue="props.modelValue" class="files" itemKey="id" :animation="150" :delay="100" :delayOnTouchOnly="true" @update:modelValue="v => emit('update:modelValue', v)"> | ||||
| <div v-show="props.modelValue.length != 0" :class="$style.root"> | ||||
| 	<Sortable :modelValue="props.modelValue" :class="$style.files" itemKey="id" :animation="150" :delay="100" :delayOnTouchOnly="true" @update:modelValue="v => emit('update:modelValue', v)"> | ||||
| 		<template #item="{element}"> | ||||
| 			<div class="file" @click="showFileMenu(element, $event)" @contextmenu.prevent="showFileMenu(element, $event)"> | ||||
| 				<MkDriveFileThumbnail :data-id="element.id" class="thumbnail" :file="element" fit="cover"/> | ||||
| 				<div v-if="element.isSensitive" class="sensitive"> | ||||
| 					<i class="ti ti-alert-triangle icon"></i> | ||||
| 			<div :class="$style.file" @click="showFileMenu(element, $event)" @contextmenu.prevent="showFileMenu(element, $event)"> | ||||
| 				<MkDriveFileThumbnail :data-id="element.id" :class="$style.thumbnail" :file="element" fit="cover"/> | ||||
| 				<div v-if="element.isSensitive" :class="$style.sensitive"> | ||||
| 					<i class="ti ti-alert-triangle" style="margin: auto;"></i> | ||||
| 				</div> | ||||
| 			</div> | ||||
| 		</template> | ||||
| 	</Sortable> | ||||
| 	<p class="remain">{{ 16 - props.modelValue.length }}/16</p> | ||||
| 	<p :class="$style.remain">{{ 16 - props.modelValue.length }}/16</p> | ||||
| </div> | ||||
| </template> | ||||
|  | ||||
| @@ -108,60 +108,53 @@ function showFileMenu(file, ev: MouseEvent) { | ||||
| } | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" scoped> | ||||
| .skeikyzd { | ||||
| <style lang="scss" module> | ||||
| .root { | ||||
| 	padding: 8px 16px; | ||||
| 	position: relative; | ||||
| } | ||||
|  | ||||
| 	> .files { | ||||
| 		display: flex; | ||||
| 		flex-wrap: wrap; | ||||
| .files { | ||||
| 	display: flex; | ||||
| 	flex-wrap: wrap; | ||||
| } | ||||
|  | ||||
| 		> .file { | ||||
| 			position: relative; | ||||
| 			width: 64px; | ||||
| 			height: 64px; | ||||
| 			margin-right: 4px; | ||||
| 			border-radius: 4px; | ||||
| 			overflow: hidden; | ||||
| 			cursor: move; | ||||
| .file { | ||||
| 	position: relative; | ||||
| 	width: 64px; | ||||
| 	height: 64px; | ||||
| 	margin-right: 4px; | ||||
| 	border-radius: 4px; | ||||
| 	overflow: hidden; | ||||
| 	cursor: move; | ||||
| } | ||||
|  | ||||
| 			&:hover > .remove { | ||||
| 				display: block; | ||||
| 			} | ||||
| .thumbnail { | ||||
| 	width: 100%; | ||||
| 	height: 100%; | ||||
| 	z-index: 1; | ||||
| 	color: var(--fg); | ||||
| } | ||||
|  | ||||
| 			> .thumbnail { | ||||
| 				width: 100%; | ||||
| 				height: 100%; | ||||
| 				z-index: 1; | ||||
| 				color: var(--fg); | ||||
| 			} | ||||
| .sensitive { | ||||
| 	display: flex; | ||||
| 	position: absolute; | ||||
| 	width: 64px; | ||||
| 	height: 64px; | ||||
| 	top: 0; | ||||
| 	left: 0; | ||||
| 	z-index: 2; | ||||
| 	background: rgba(17, 17, 17, .7); | ||||
| 	color: #fff; | ||||
| } | ||||
|  | ||||
| 			> .sensitive { | ||||
| 				display: flex; | ||||
| 				position: absolute; | ||||
| 				width: 64px; | ||||
| 				height: 64px; | ||||
| 				top: 0; | ||||
| 				left: 0; | ||||
| 				z-index: 2; | ||||
| 				background: rgba(17, 17, 17, .7); | ||||
| 				color: #fff; | ||||
|  | ||||
| 				> .icon { | ||||
| 					margin: auto; | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	> .remain { | ||||
| 		display: block; | ||||
| 		position: absolute; | ||||
| 		top: 8px; | ||||
| 		right: 8px; | ||||
| 		margin: 0; | ||||
| 		padding: 0; | ||||
| 	} | ||||
| .remain { | ||||
| 	display: block; | ||||
| 	position: absolute; | ||||
| 	top: 8px; | ||||
| 	right: 8px; | ||||
| 	margin: 0; | ||||
| 	padding: 0; | ||||
| 	font-size: 90%; | ||||
| } | ||||
| </style> | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| <template> | ||||
| <div ref="rootEl" class="meijqfqm"> | ||||
| 	<canvas :id="idForCanvas" ref="canvasEl" class="canvas" :width="width" height="300" @contextmenu.prevent="() => {}"></canvas> | ||||
| 	<div :id="idForTags" ref="tagsEl" class="tags"> | ||||
| <div ref="rootEl" :class="$style.root"> | ||||
| 	<canvas :id="idForCanvas" ref="canvasEl" style="display: block;" :width="width" height="300" @contextmenu.prevent="() => {}"></canvas> | ||||
| 	<div :id="idForTags" ref="tagsEl" :class="$style.tags"> | ||||
| 		<ul> | ||||
| 			<slot></slot> | ||||
| 		</ul> | ||||
| @@ -70,21 +70,17 @@ defineExpose({ | ||||
| }); | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" scoped> | ||||
| .meijqfqm { | ||||
| <style lang="scss" module> | ||||
| .root { | ||||
| 	position: relative; | ||||
| 	overflow: clip; | ||||
| 	display: grid; | ||||
| 	place-items: center; | ||||
| } | ||||
|  | ||||
| 	> .canvas { | ||||
| 		display: block; | ||||
| 	} | ||||
|  | ||||
| 	> .tags { | ||||
| 		position: absolute; | ||||
| 		top: 999px; | ||||
| 		left: 999px; | ||||
| 	} | ||||
| .tags { | ||||
| 	position: absolute; | ||||
| 	top: 999px; | ||||
| 	left: 999px; | ||||
| } | ||||
| </style> | ||||
|   | ||||
| @@ -1,12 +1,12 @@ | ||||
| <template> | ||||
| <div class="adhpbeos"> | ||||
| 	<div class="label" @click="focus"><slot name="label"></slot></div> | ||||
| 	<div class="input" :class="{ disabled, focused, tall, pre }"> | ||||
| <div> | ||||
| 	<div :class="$style.label" @click="focus"><slot name="label"></slot></div> | ||||
| 	<div :class="{ [$style.disabled]: disabled, [$style.focused]: focused, [$style.tall]: tall, [$style.pre]: pre }" style="position: relative;"> | ||||
| 		<textarea | ||||
| 			ref="inputEl" | ||||
| 			v-model="v" | ||||
| 			v-adaptive-border | ||||
| 			:class="{ code, _monospace: code }" | ||||
| 			:class="[$style.textarea, { [$style.code]: code, _monospace: code }]" | ||||
| 			:disabled="disabled" | ||||
| 			:required="required" | ||||
| 			:readonly="readonly" | ||||
| @@ -20,9 +20,9 @@ | ||||
| 			@input="onInput" | ||||
| 		></textarea> | ||||
| 	</div> | ||||
| 	<div class="caption"><slot name="caption"></slot></div> | ||||
| 	<div :class="$style.caption"><slot name="caption"></slot></div> | ||||
|  | ||||
| 	<MkButton v-if="manualSave && changed" primary class="save" @click="updated"><i class="ti ti-device-floppy"></i> {{ i18n.ts.save }}</MkButton> | ||||
| 	<MkButton v-if="manualSave && changed" primary :class="$style.save" @click="updated"><i class="ti ti-device-floppy"></i> {{ i18n.ts.save }}</MkButton> | ||||
| </div> | ||||
| </template> | ||||
|  | ||||
| @@ -111,87 +111,82 @@ onMounted(() => { | ||||
| }); | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" scoped> | ||||
| .adhpbeos { | ||||
| 	> .label { | ||||
| 		font-size: 0.85em; | ||||
| 		padding: 0 0 8px 0; | ||||
| 		user-select: none; | ||||
| <style lang="scss" module> | ||||
| .label { | ||||
| 	font-size: 0.85em; | ||||
| 	padding: 0 0 8px 0; | ||||
| 	user-select: none; | ||||
|  | ||||
| 		&:empty { | ||||
| 			display: none; | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	> .caption { | ||||
| 		font-size: 0.85em; | ||||
| 		padding: 8px 0 0 0; | ||||
| 		color: var(--fgTransparentWeak); | ||||
|  | ||||
| 		&:empty { | ||||
| 			display: none; | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	> .input { | ||||
| 		position: relative; | ||||
|  | ||||
| 		> textarea { | ||||
| 			appearance: none; | ||||
| 			-webkit-appearance: none; | ||||
| 			display: block; | ||||
| 			width: 100%; | ||||
| 			min-width: 100%; | ||||
| 			max-width: 100%; | ||||
| 			min-height: 130px; | ||||
| 			margin: 0; | ||||
| 			padding: 12px; | ||||
| 			font: inherit; | ||||
| 			font-weight: normal; | ||||
| 			font-size: 1em; | ||||
| 			color: var(--fg); | ||||
| 			background: var(--panel); | ||||
| 			border: solid 1px var(--panel); | ||||
| 			border-radius: 6px; | ||||
| 			outline: none; | ||||
| 			box-shadow: none; | ||||
| 			box-sizing: border-box; | ||||
| 			transition: border-color 0.1s ease-out; | ||||
|  | ||||
| 			&:hover { | ||||
| 				border-color: var(--inputBorderHover) !important; | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		&.focused { | ||||
| 			> textarea { | ||||
| 				border-color: var(--accent) !important; | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		&.disabled { | ||||
| 			opacity: 0.7; | ||||
|  | ||||
| 			&, * { | ||||
| 				cursor: not-allowed !important; | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		&.tall { | ||||
| 			> textarea { | ||||
| 				min-height: 200px; | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		&.pre { | ||||
| 			> textarea { | ||||
| 				white-space: pre; | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	> .save { | ||||
| 		margin: 8px 0 0 0; | ||||
| 	&:empty { | ||||
| 		display: none; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| .caption { | ||||
| 	font-size: 0.85em; | ||||
| 	padding: 8px 0 0 0; | ||||
| 	color: var(--fgTransparentWeak); | ||||
|  | ||||
| 	&:empty { | ||||
| 		display: none; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| .textarea { | ||||
| 	appearance: none; | ||||
| 	-webkit-appearance: none; | ||||
| 	display: block; | ||||
| 	width: 100%; | ||||
| 	min-width: 100%; | ||||
| 	max-width: 100%; | ||||
| 	min-height: 130px; | ||||
| 	margin: 0; | ||||
| 	padding: 12px; | ||||
| 	font: inherit; | ||||
| 	font-weight: normal; | ||||
| 	font-size: 1em; | ||||
| 	color: var(--fg); | ||||
| 	background: var(--panel); | ||||
| 	border: solid 1px var(--panel); | ||||
| 	border-radius: 6px; | ||||
| 	outline: none; | ||||
| 	box-shadow: none; | ||||
| 	box-sizing: border-box; | ||||
| 	transition: border-color 0.1s ease-out; | ||||
|  | ||||
| 	&:hover { | ||||
| 		border-color: var(--inputBorderHover) !important; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| .focused { | ||||
| 	> .textarea { | ||||
| 		border-color: var(--accent) !important; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| .disabled { | ||||
| 	opacity: 0.7; | ||||
| 	cursor: not-allowed !important; | ||||
|  | ||||
| 	> .textarea { | ||||
| 		cursor: not-allowed !important; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| .tall { | ||||
| 	> .textarea { | ||||
| 		min-height: 200px; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| .pre { | ||||
| 	> .textarea { | ||||
| 		white-space: pre; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| .save { | ||||
| 	margin: 8px 0 0 0; | ||||
| } | ||||
| </style> | ||||
|   | ||||
| @@ -46,12 +46,6 @@ const onUserRemoved = () => { | ||||
| 	tlComponent.pagingComponent?.reload(); | ||||
| }; | ||||
|  | ||||
| const onChangeFollowing = () => { | ||||
| 	if (!tlComponent.pagingComponent?.backed) { | ||||
| 		tlComponent.pagingComponent?.reload(); | ||||
| 	} | ||||
| }; | ||||
|  | ||||
| let endpoint; | ||||
| let query; | ||||
| let connection; | ||||
| @@ -79,8 +73,6 @@ if (props.src === 'antenna') { | ||||
| 	connection.on('note', prepend); | ||||
|  | ||||
| 	connection2 = stream.useChannel('main'); | ||||
| 	connection2.on('follow', onChangeFollowing); | ||||
| 	connection2.on('unfollow', onChangeFollowing); | ||||
| } else if (props.src === 'local') { | ||||
| 	endpoint = 'notes/local-timeline'; | ||||
| 	query = { | ||||
|   | ||||
| @@ -34,6 +34,7 @@ | ||||
| 							<div style="font-size: 120%;">{{ i18n.ts._initialAccountSetting.accountCreated }}</div> | ||||
| 							<div>{{ i18n.ts._initialAccountSetting.letsStartAccountSetup }}</div> | ||||
| 							<MkButton primary rounded gradate style="margin: 16px auto 0 auto;" data-cy-user-setup-continue @click="page++">{{ i18n.ts._initialAccountSetting.profileSetting }} <i class="ti ti-arrow-right"></i></MkButton> | ||||
| 							<MkButton style="margin: 0 auto;" transparent rounded @click="later(true)">{{ i18n.ts.later }}</MkButton> | ||||
| 						</div> | ||||
| 					</MkSpacer> | ||||
| 				</div> | ||||
| @@ -43,6 +44,7 @@ | ||||
| 					<MkSpacer :marginMin="20" :marginMax="28"> | ||||
| 						<XProfile/> | ||||
| 						<div class="_buttonsCenter" style="margin-top: 16px;"> | ||||
| 							<MkButton rounded data-cy-user-setup-back @click="page--"><i class="ti ti-arrow-left"></i> {{ i18n.ts.goBack }}</MkButton> | ||||
| 							<MkButton primary rounded gradate data-cy-user-setup-continue @click="page++">{{ i18n.ts.continue }} <i class="ti ti-arrow-right"></i></MkButton> | ||||
| 						</div> | ||||
| 					</MkSpacer> | ||||
| @@ -53,6 +55,7 @@ | ||||
| 					<MkSpacer :marginMin="20" :marginMax="28"> | ||||
| 						<XPrivacy/> | ||||
| 						<div class="_buttonsCenter" style="margin-top: 16px;"> | ||||
| 							<MkButton rounded data-cy-user-setup-back @click="page--"><i class="ti ti-arrow-left"></i> {{ i18n.ts.goBack }}</MkButton> | ||||
| 							<MkButton primary rounded gradate data-cy-user-setup-continue @click="page++">{{ i18n.ts.continue }} <i class="ti ti-arrow-right"></i></MkButton> | ||||
| 						</div> | ||||
| 					</MkSpacer> | ||||
| @@ -64,7 +67,10 @@ | ||||
| 						<XFollow/> | ||||
| 					</MkSpacer> | ||||
| 					<div :class="$style.pageFooter"> | ||||
| 						<MkButton primary rounded gradate style="margin: 0 auto;" data-cy-user-setup-continue @click="page++">{{ i18n.ts.continue }} <i class="ti ti-arrow-right"></i></MkButton> | ||||
| 						<div class="_buttonsCenter"> | ||||
| 							<MkButton rounded data-cy-user-setup-back @click="page--"><i class="ti ti-arrow-left"></i> {{ i18n.ts.goBack }}</MkButton> | ||||
| 							<MkButton primary rounded gradate style="" data-cy-user-setup-continue @click="page++">{{ i18n.ts.continue }} <i class="ti ti-arrow-right"></i></MkButton> | ||||
| 						</div> | ||||
| 					</div> | ||||
| 				</div> | ||||
| 			</template> | ||||
| @@ -76,7 +82,10 @@ | ||||
| 							<div style="font-size: 120%;">{{ i18n.ts.pushNotification }}</div> | ||||
| 							<div style="padding: 0 16px;">{{ i18n.t('_initialAccountSetting.pushNotificationDescription', { name: instance.name ?? host }) }}</div> | ||||
| 							<MkPushNotificationAllowButton primary showOnlyToRegister style="margin: 0 auto;"/> | ||||
| 							<MkButton primary rounded gradate style="margin: 16px auto 0 auto;" data-cy-user-setup-continue @click="page++">{{ i18n.ts.continue }} <i class="ti ti-arrow-right"></i></MkButton> | ||||
| 							<div class="_buttonsCenter" style="margin-top: 16px;"> | ||||
| 								<MkButton rounded data-cy-user-setup-back @click="page--"><i class="ti ti-arrow-left"></i> {{ i18n.ts.goBack }}</MkButton> | ||||
| 								<MkButton primary rounded gradate data-cy-user-setup-continue @click="page++">{{ i18n.ts.continue }} <i class="ti ti-arrow-right"></i></MkButton> | ||||
| 							</div> | ||||
| 						</div> | ||||
| 					</MkSpacer> | ||||
| 				</div> | ||||
| @@ -95,7 +104,10 @@ | ||||
| 								</template> | ||||
| 							</I18n> | ||||
| 							<div>{{ i18n.t('_initialAccountSetting.haveFun', { name: instance.name ?? host }) }}</div> | ||||
| 							<MkButton primary rounded gradate style="margin: 16px auto 0 auto;" data-cy-user-setup-continue @click="close(false)">{{ i18n.ts.close }}</MkButton> | ||||
| 							<div class="_buttonsCenter" style="margin-top: 16px;"> | ||||
| 								<MkButton rounded data-cy-user-setup-back @click="page--"><i class="ti ti-arrow-left"></i> {{ i18n.ts.goBack }}</MkButton> | ||||
| 								<MkButton primary rounded gradate data-cy-user-setup-continue @click="close(false)">{{ i18n.ts.close }}</MkButton> | ||||
| 							</div> | ||||
| 						</div> | ||||
| 					</MkSpacer> | ||||
| 				</div> | ||||
| @@ -144,6 +156,19 @@ async function close(skip: boolean) { | ||||
| 	dialog.value.close(); | ||||
| 	defaultStore.set('accountSetupWizard', -1); | ||||
| } | ||||
|  | ||||
| async function later(later: boolean) { | ||||
| 	if (later) { | ||||
| 		const { canceled } = await os.confirm({ | ||||
| 			type: 'warning', | ||||
| 			text: i18n.ts._initialAccountSetting.laterAreYouSure, | ||||
| 		}); | ||||
| 		if (canceled) return; | ||||
| 	} | ||||
|  | ||||
| 	dialog.value.close(); | ||||
| 	defaultStore.set('accountSetupWizard', 0); | ||||
| } | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" module> | ||||
| @@ -190,7 +215,7 @@ async function close(skip: boolean) { | ||||
| 	left: 0; | ||||
| 	padding: 12px; | ||||
| 	border-top: solid 0.5px var(--divider); | ||||
| 	-webkit-backdrop-filter: var(--blur, blur(15px)); | ||||
| 	backdrop-filter: var(--blur, blur(15px)); | ||||
| 	-webkit-backdrop-filter: blur(15px); | ||||
| 	backdrop-filter: blur(15px); | ||||
| } | ||||
| </style> | ||||
|   | ||||
| @@ -1,19 +1,19 @@ | ||||
| <template> | ||||
| <div class="ffcbddfc" :class="{ inline }"> | ||||
| 	<a v-if="external" class="main _button" :href="to" target="_blank"> | ||||
| 		<span class="icon"><slot name="icon"></slot></span> | ||||
| 		<span class="text"><slot></slot></span> | ||||
| 		<span class="right"> | ||||
| 			<span class="text"><slot name="suffix"></slot></span> | ||||
| 			<i class="ti ti-external-link icon"></i> | ||||
| <div :class="[$style.root, { [$style.inline]: inline }]"> | ||||
| 	<a v-if="external" :class="$style.main" class="_button" :href="to" target="_blank"> | ||||
| 		<span :class="$style.icon"><slot name="icon"></slot></span> | ||||
| 		<span :class="$style.text"><slot></slot></span> | ||||
| 		<span :class="$style.suffix"> | ||||
| 			<span :class="$style.suffixText"><slot name="suffix"></slot></span> | ||||
| 			<i class="ti ti-external-link" :class="$style.suffixIcon"></i> | ||||
| 		</span> | ||||
| 	</a> | ||||
| 	<MkA v-else class="main _button" :class="{ active }" :to="to" :behavior="behavior"> | ||||
| 		<span class="icon"><slot name="icon"></slot></span> | ||||
| 		<span class="text"><slot></slot></span> | ||||
| 		<span class="right"> | ||||
| 			<span class="text"><slot name="suffix"></slot></span> | ||||
| 			<i class="ti ti-chevron-right icon"></i> | ||||
| 	<MkA v-else :class="[$style.main, { [$style.active]: active }]" class="_button" :to="to" :behavior="behavior"> | ||||
| 		<span :class="$style.icon"><slot name="icon"></slot></span> | ||||
| 		<span :class="$style.text"><slot></slot></span> | ||||
| 		<span :class="$style.suffix"> | ||||
| 			<span :class="$style.suffixText"><slot name="suffix"></slot></span> | ||||
| 			<i class="ti ti-chevron-right" :class="$style.suffixIcon"></i> | ||||
| 		</span> | ||||
| 	</MkA> | ||||
| </div> | ||||
| @@ -26,70 +26,70 @@ const props = defineProps<{ | ||||
| 	to: string; | ||||
| 	active?: boolean; | ||||
| 	external?: boolean; | ||||
| 	behavior?: null | 'window' | 'browser' | 'modalWindow'; | ||||
| 	behavior?: null | 'window' | 'browser'; | ||||
| 	inline?: boolean; | ||||
| }>(); | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" scoped> | ||||
| .ffcbddfc { | ||||
| <style lang="scss" module> | ||||
| .root { | ||||
| 	display: block; | ||||
|  | ||||
| 	&.inline { | ||||
| 		display: inline-block; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| 	> .main { | ||||
| 		display: flex; | ||||
| 		align-items: center; | ||||
| 		width: 100%; | ||||
| 		box-sizing: border-box; | ||||
| 		padding: 10px 14px; | ||||
| 		background: var(--buttonBg); | ||||
| 		border-radius: 6px; | ||||
| 		font-size: 0.9em; | ||||
| .main { | ||||
| 	display: flex; | ||||
| 	align-items: center; | ||||
| 	width: 100%; | ||||
| 	box-sizing: border-box; | ||||
| 	padding: 10px 14px; | ||||
| 	background: var(--buttonBg); | ||||
| 	border-radius: 6px; | ||||
| 	font-size: 0.9em; | ||||
|  | ||||
| 		&:hover { | ||||
| 			text-decoration: none; | ||||
| 			background: var(--buttonHoverBg); | ||||
| 		} | ||||
| 	&:hover { | ||||
| 		text-decoration: none; | ||||
| 		background: var(--buttonHoverBg); | ||||
| 	} | ||||
|  | ||||
| 		&.active { | ||||
| 			color: var(--accent); | ||||
| 			background: var(--buttonHoverBg); | ||||
| 		} | ||||
| 	&.active { | ||||
| 		color: var(--accent); | ||||
| 		background: var(--buttonHoverBg); | ||||
| 	} | ||||
| } | ||||
|  | ||||
| 		> .icon { | ||||
| 			margin-right: 0.75em; | ||||
| 			flex-shrink: 0; | ||||
| 			text-align: center; | ||||
| 			color: var(--fgTransparentWeak); | ||||
| .icon { | ||||
| 	margin-right: 0.75em; | ||||
| 	flex-shrink: 0; | ||||
| 	text-align: center; | ||||
| 	color: var(--fgTransparentWeak); | ||||
|  | ||||
| 			&:empty { | ||||
| 				display: none; | ||||
| 	&:empty { | ||||
| 		display: none; | ||||
|  | ||||
| 				& + .text { | ||||
| 					padding-left: 4px; | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		> .text { | ||||
| 			flex-shrink: 1; | ||||
| 			white-space: normal; | ||||
| 			padding-right: 12px; | ||||
| 			text-align: center; | ||||
| 		} | ||||
|  | ||||
| 		> .right { | ||||
| 			margin-left: auto; | ||||
| 			opacity: 0.7; | ||||
| 			white-space: nowrap; | ||||
|  | ||||
| 			> .text:not(:empty) { | ||||
| 				margin-right: 0.75em; | ||||
| 			} | ||||
| 		& + .text { | ||||
| 			padding-left: 4px; | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| .text { | ||||
| 	flex-shrink: 1; | ||||
| 	white-space: normal; | ||||
| 	padding-right: 12px; | ||||
| 	text-align: center; | ||||
| } | ||||
|  | ||||
| .suffix { | ||||
| 	margin-left: auto; | ||||
| 	opacity: 0.7; | ||||
| 	white-space: nowrap; | ||||
|  | ||||
| 	> .suffixText:not(:empty) { | ||||
| 		margin-right: 0.75em; | ||||
| 	} | ||||
| } | ||||
| </style> | ||||
|   | ||||
| @@ -1,10 +1,10 @@ | ||||
| <template> | ||||
| <div class="adhpbeou"> | ||||
| 	<div class="label" @click="focus"><slot name="label"></slot></div> | ||||
| 	<div class="content"> | ||||
| <div> | ||||
| 	<div :class="$style.label" @click="focus"><slot name="label"></slot></div> | ||||
| 	<div :class="$style.content"> | ||||
| 		<slot></slot> | ||||
| 	</div> | ||||
| 	<div class="caption"><slot name="caption"></slot></div> | ||||
| 	<div :class="$style.caption"><slot name="caption"></slot></div> | ||||
| </div> | ||||
| </template> | ||||
|  | ||||
| @@ -16,26 +16,24 @@ function focus() { | ||||
| } | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" scoped> | ||||
| .adhpbeou { | ||||
| 	> .label { | ||||
| 		font-size: 0.85em; | ||||
| 		padding: 0 0 8px 0; | ||||
| 		user-select: none; | ||||
| <style lang="scss" module> | ||||
| .label { | ||||
| 	font-size: 0.85em; | ||||
| 	padding: 0 0 8px 0; | ||||
| 	user-select: none; | ||||
|  | ||||
| 		&:empty { | ||||
| 			display: none; | ||||
| 		} | ||||
| 	&:empty { | ||||
| 		display: none; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| 	> .caption { | ||||
| 		font-size: 0.85em; | ||||
| 		padding: 8px 0 0 0; | ||||
| 		color: var(--fgTransparentWeak); | ||||
| .caption { | ||||
| 	font-size: 0.85em; | ||||
| 	padding: 8px 0 0 0; | ||||
| 	color: var(--fgTransparentWeak); | ||||
|  | ||||
| 		&:empty { | ||||
| 			display: none; | ||||
| 		} | ||||
| 	&:empty { | ||||
| 		display: none; | ||||
| 	} | ||||
| } | ||||
| </style> | ||||
|   | ||||
| @@ -1,18 +1,16 @@ | ||||
| <template> | ||||
| <Transition :name="defaultStore.state.animation ? 'fade' : ''" mode="out-in"> | ||||
| 	<div v-if="pending"> | ||||
| 		<MkLoading/> | ||||
| <div v-if="pending"> | ||||
| 	<MkLoading/> | ||||
| </div> | ||||
| <div v-else-if="resolved"> | ||||
| 	<slot :result="result"></slot> | ||||
| </div> | ||||
| <div v-else> | ||||
| 	<div :class="$style.error"> | ||||
| 		<div><i class="ti ti-alert-triangle"></i> {{ i18n.ts.somethingHappened }}</div> | ||||
| 		<MkButton inline style="margin-top: 16px;" @click="retry"><i class="ti ti-reload"></i> {{ i18n.ts.retry }}</MkButton> | ||||
| 	</div> | ||||
| 	<div v-else-if="resolved"> | ||||
| 		<slot :result="result"></slot> | ||||
| 	</div> | ||||
| 	<div v-else> | ||||
| 		<div class="wszdbhzo"> | ||||
| 			<div><i class="ti ti-alert-triangle"></i> {{ i18n.ts.somethingHappened }}</div> | ||||
| 			<MkButton inline class="retry" @click="retry"><i class="ti ti-reload"></i> {{ i18n.ts.retry }}</MkButton> | ||||
| 		</div> | ||||
| 	</div> | ||||
| </Transition> | ||||
| </div> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts" setup> | ||||
| @@ -60,22 +58,9 @@ const retry = () => { | ||||
| }; | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" scoped> | ||||
| .fade-enter-active, | ||||
| .fade-leave-active { | ||||
| 	transition: opacity 0.125s ease; | ||||
| } | ||||
| .fade-enter-from, | ||||
| .fade-leave-to { | ||||
| 	opacity: 0; | ||||
| } | ||||
|  | ||||
| .wszdbhzo { | ||||
| <style lang="scss" module> | ||||
| .error { | ||||
| 	padding: 16px; | ||||
| 	text-align: center; | ||||
|  | ||||
| 	> .retry { | ||||
| 		margin-top: 16px; | ||||
| 	} | ||||
| } | ||||
| </style> | ||||
|   | ||||
| @@ -15,7 +15,7 @@ import { useRouter } from '@/router'; | ||||
| const props = withDefaults(defineProps<{ | ||||
| 	to: string; | ||||
| 	activeClass?: null | string; | ||||
| 	behavior?: null | 'window' | 'browser' | 'modalWindow'; | ||||
| 	behavior?: null | 'window' | 'browser'; | ||||
| }>(), { | ||||
| 	activeClass: null, | ||||
| 	behavior: null, | ||||
| @@ -70,14 +70,6 @@ function openWindow() { | ||||
| 	os.pageWindow(props.to); | ||||
| } | ||||
|  | ||||
| function modalWindow() { | ||||
| 	os.modalPageWindow(props.to); | ||||
| } | ||||
|  | ||||
| function popout() { | ||||
| 	popout_(props.to); | ||||
| } | ||||
|  | ||||
| function nav(ev: MouseEvent) { | ||||
| 	if (props.behavior === 'browser') { | ||||
| 		location.href = props.to; | ||||
| @@ -87,8 +79,6 @@ function nav(ev: MouseEvent) { | ||||
| 	if (props.behavior) { | ||||
| 		if (props.behavior === 'window') { | ||||
| 			return openWindow(); | ||||
| 		} else if (props.behavior === 'modalWindow') { | ||||
| 			return modalWindow(); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
|   | ||||
| @@ -1,8 +1,8 @@ | ||||
| <template> | ||||
| <section class="sdgxphyu"> | ||||
| 	<component :is="'h' + h">{{ block.title }}</component> | ||||
| <section> | ||||
| 	<component :is="'h' + h" :class="h < 5 ? $style['h' + h] : null">{{ block.title }}</component> | ||||
|  | ||||
| 	<div class="children"> | ||||
| 	<div class="_gaps"> | ||||
| 		<XBlock v-for="child in block.children" :key="child.id" :page="page" :block="child" :h="h + 1"/> | ||||
| 	</div> | ||||
| </section> | ||||
| @@ -22,27 +22,19 @@ defineProps<{ | ||||
| }>(); | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" scoped> | ||||
| .sdgxphyu { | ||||
| 	margin: 1.5em 0; | ||||
| <style lang="scss" module> | ||||
| .h2 { | ||||
| 	font-size: 1.35em; | ||||
| 	margin: 0 0 0.5em 0; | ||||
| } | ||||
|  | ||||
| 	> h2 { | ||||
| 		font-size: 1.35em; | ||||
| 		margin: 0 0 0.5em 0; | ||||
| 	} | ||||
| .h3 { | ||||
| 	font-size: 1em; | ||||
| 	margin: 0 0 0.5em 0; | ||||
| } | ||||
|  | ||||
| 	> h3 { | ||||
| 		font-size: 1em; | ||||
| 		margin: 0 0 0.5em 0; | ||||
| 	} | ||||
|  | ||||
| 	> h4 { | ||||
| 		font-size: 1em; | ||||
| 		margin: 0 0 0.5em 0; | ||||
| 	} | ||||
|  | ||||
| 	> .children { | ||||
| 		//padding 16px | ||||
| 	} | ||||
| .h4 { | ||||
| 	font-size: 1em; | ||||
| 	margin: 0 0 0.5em 0; | ||||
| } | ||||
| </style> | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| <template> | ||||
| <div class="mrdgzndn"> | ||||
| <div class="_gaps"> | ||||
| 	<Mfm :text="block.text" :isNote="false" :i="$i"/> | ||||
| 	<MkUrlPreview v-for="url in urls" :key="url" :url="url" class="url"/> | ||||
| 	<MkUrlPreview v-for="url in urls" :key="url" :url="url"/> | ||||
| </div> | ||||
| </template> | ||||
|  | ||||
| @@ -22,19 +22,3 @@ const props = defineProps<{ | ||||
|  | ||||
| const urls = props.block.text ? extractUrlFromMfm(mfm.parse(props.block.text)) : []; | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" scoped> | ||||
| .mrdgzndn { | ||||
| 	&:not(:first-child) { | ||||
| 		margin-top: 0.5em; | ||||
| 	} | ||||
|  | ||||
| 	&:not(:last-child) { | ||||
| 		margin-bottom: 0.5em; | ||||
| 	} | ||||
|  | ||||
| 	> .url { | ||||
| 		margin: 0.5em 0; | ||||
| 	} | ||||
| } | ||||
| </style> | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| <template> | ||||
| <div class="iroscrza" :class="{ center: page.alignCenter, serif: page.font === 'serif' }"> | ||||
| <div :class="{ [$style.center]: page.alignCenter, [$style.serif]: page.font === 'serif' }"> | ||||
| 	<XBlock v-for="child in page.content" :key="child.id" :block="child" :h="2"/> | ||||
| </div> | ||||
| </template> | ||||
| @@ -14,16 +14,12 @@ defineProps<{ | ||||
| }>(); | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" scoped> | ||||
| .iroscrza { | ||||
| 	&.serif { | ||||
| 		> div { | ||||
| 			font-family: serif; | ||||
| 		} | ||||
| 	} | ||||
| <style lang="scss" module> | ||||
| .serif { | ||||
| 	font-family: serif; | ||||
| } | ||||
|  | ||||
| 	&.center { | ||||
| 		text-align: center; | ||||
| 	} | ||||
| .center { | ||||
| 	text-align: center; | ||||
| } | ||||
| </style> | ||||
|   | ||||
| @@ -1,8 +1,9 @@ | ||||
| import { markRaw } from 'vue'; | ||||
| import type { Locale } from '../../../locales'; | ||||
| import { locale } from '@/config'; | ||||
| import { I18n } from '@/scripts/i18n'; | ||||
|  | ||||
| export const i18n = markRaw(new I18n(locale)); | ||||
| export const i18n = markRaw(new I18n<Locale>(locale)); | ||||
|  | ||||
| export function updateI18n(newLocale) { | ||||
| 	i18n.ts = newLocale; | ||||
|   | ||||
| @@ -172,12 +172,6 @@ export function pageWindow(path: string) { | ||||
| 	}, {}, 'closed'); | ||||
| } | ||||
|  | ||||
| export function modalPageWindow(path: string) { | ||||
| 	popup(defineAsyncComponent(() => import('@/components/MkModalPageWindow.vue')), { | ||||
| 		initialPath: path, | ||||
| 	}, {}, 'closed'); | ||||
| } | ||||
|  | ||||
| export function toast(message: string) { | ||||
| 	popup(MkToast, { | ||||
| 		message, | ||||
|   | ||||
| @@ -3,7 +3,7 @@ | ||||
| 	<MkStickyContainer> | ||||
| 		<template #header><XHeader :actions="headerActions"/></template> | ||||
| 		<MkSpacer :contentMax="900"> | ||||
| 			<div> | ||||
| 			<div class="_gaps"> | ||||
| 				<div> | ||||
| 					<MkInput v-model="host" :debounce="true" class=""> | ||||
| 						<template #prefix><i class="ti ti-search"></i></template> | ||||
|   | ||||
| @@ -3,29 +3,27 @@ | ||||
| 	<MkStickyContainer> | ||||
| 		<template #header><XHeader :actions="headerActions"/></template> | ||||
| 		<MkSpacer :contentMax="900"> | ||||
| 			<div> | ||||
| 				<div> | ||||
| 					<div class="inputs" style="display: flex; gap: var(--margin); flex-wrap: wrap;"> | ||||
| 						<MkSelect v-model="origin" style="margin: 0; flex: 1;"> | ||||
| 							<template #label>{{ i18n.ts.instance }}</template> | ||||
| 							<option value="combined">{{ i18n.ts.all }}</option> | ||||
| 							<option value="local">{{ i18n.ts.local }}</option> | ||||
| 							<option value="remote">{{ i18n.ts.remote }}</option> | ||||
| 						</MkSelect> | ||||
| 						<MkInput v-model="searchHost" :debounce="true" type="search" style="margin: 0; flex: 1;" :disabled="pagination.params.origin === 'local'"> | ||||
| 							<template #label>{{ i18n.ts.host }}</template> | ||||
| 						</MkInput> | ||||
| 					</div> | ||||
| 					<div class="inputs" style="display: flex; gap: var(--margin); flex-wrap: wrap; padding-top: 1.2em;"> | ||||
| 						<MkInput v-model="userId" :debounce="true" type="search" style="margin: 0; flex: 1;"> | ||||
| 							<template #label>User ID</template> | ||||
| 						</MkInput> | ||||
| 						<MkInput v-model="type" :debounce="true" type="search" style="margin: 0; flex: 1;"> | ||||
| 							<template #label>MIME type</template> | ||||
| 						</MkInput> | ||||
| 					</div> | ||||
| 					<MkFileListForAdmin :pagination="pagination" :viewMode="viewMode"/> | ||||
| 			<div class="_gaps"> | ||||
| 				<div class="inputs" style="display: flex; gap: var(--margin); flex-wrap: wrap;"> | ||||
| 					<MkSelect v-model="origin" style="margin: 0; flex: 1;"> | ||||
| 						<template #label>{{ i18n.ts.instance }}</template> | ||||
| 						<option value="combined">{{ i18n.ts.all }}</option> | ||||
| 						<option value="local">{{ i18n.ts.local }}</option> | ||||
| 						<option value="remote">{{ i18n.ts.remote }}</option> | ||||
| 					</MkSelect> | ||||
| 					<MkInput v-model="searchHost" :debounce="true" type="search" style="margin: 0; flex: 1;" :disabled="pagination.params.origin === 'local'"> | ||||
| 						<template #label>{{ i18n.ts.host }}</template> | ||||
| 					</MkInput> | ||||
| 				</div> | ||||
| 				<div class="inputs" style="display: flex; gap: var(--margin); flex-wrap: wrap;"> | ||||
| 					<MkInput v-model="userId" :debounce="true" type="search" style="margin: 0; flex: 1;"> | ||||
| 						<template #label>User ID</template> | ||||
| 					</MkInput> | ||||
| 					<MkInput v-model="type" :debounce="true" type="search" style="margin: 0; flex: 1;"> | ||||
| 						<template #label>MIME type</template> | ||||
| 					</MkInput> | ||||
| 				</div> | ||||
| 				<MkFileListForAdmin :pagination="pagination" :viewMode="viewMode"/> | ||||
| 			</div> | ||||
| 		</MkSpacer> | ||||
| 	</MkStickyContainer> | ||||
|   | ||||
| @@ -5,10 +5,10 @@ | ||||
| 		<div class="_gaps"> | ||||
| 			<div v-for="relay in relays" :key="relay.inbox" class="relaycxt _panel" style="padding: 16px;"> | ||||
| 				<div>{{ relay.inbox }}</div> | ||||
| 				<div class="status"> | ||||
| 					<i v-if="relay.status === 'accepted'" class="ti ti-check icon accepted"></i> | ||||
| 					<i v-else-if="relay.status === 'rejected'" class="ti ti-ban icon rejected"></i> | ||||
| 					<i v-else class="ti ti-clock icon requesting"></i> | ||||
| 				<div style="margin: 8px 0;"> | ||||
| 					<i v-if="relay.status === 'accepted'" class="ti ti-check" :class="$style.icon" style="color: var(--success);"></i> | ||||
| 					<i v-else-if="relay.status === 'rejected'" class="ti ti-ban" :class="$style.icon" style="color: var(--error);"></i> | ||||
| 					<i v-else class="ti ti-clock" :class="$style.icon"></i> | ||||
| 					<span>{{ i18n.t(`_relayStatus.${relay.status}`) }}</span> | ||||
| 				</div> | ||||
| 				<MkButton class="button" inline danger @click="remove(relay.inbox)"><i class="ti ti-trash"></i> {{ i18n.ts.remove }}</MkButton> | ||||
| @@ -83,23 +83,9 @@ definePageMetadata({ | ||||
| }); | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" scoped> | ||||
| .relaycxt { | ||||
| 	> .status { | ||||
| 		margin: 8px 0; | ||||
|  | ||||
| 		> .icon { | ||||
| 			width: 1em; | ||||
| 			margin-right: 0.75em; | ||||
|  | ||||
| 			&.accepted { | ||||
| 				color: var(--success); | ||||
| 			} | ||||
|  | ||||
| 			&.rejected { | ||||
| 				color: var(--error); | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| <style lang="scss" module> | ||||
| .icon { | ||||
| 	width: 1em; | ||||
| 	margin-right: 0.75em; | ||||
| } | ||||
| </style> | ||||
|   | ||||
| @@ -2,29 +2,29 @@ | ||||
| <MkStickyContainer> | ||||
| 	<template #header><MkPageHeader v-model:tab="tab" :actions="headerActions" :tabs="headerTabs"/></template> | ||||
| 	<MkSpacer :contentMax="700"> | ||||
| 		<div v-if="tab === 'featured'" class=""> | ||||
| 		<div v-if="tab === 'featured'"> | ||||
| 			<MkPagination v-slot="{items}" :pagination="featuredFlashsPagination"> | ||||
| 				<div class="_gaps_s"> | ||||
| 					<MkFlashPreview v-for="flash in items" :key="flash.id" class="" :flash="flash"/> | ||||
| 					<MkFlashPreview v-for="flash in items" :key="flash.id" :flash="flash"/> | ||||
| 				</div> | ||||
| 			</MkPagination> | ||||
| 		</div> | ||||
|  | ||||
| 		<div v-else-if="tab === 'my'" class="my"> | ||||
| 		<div v-else-if="tab === 'my'"> | ||||
| 			<div class="_gaps"> | ||||
| 				<MkButton class="new" gradate rounded style="margin: 0 auto;" @click="create()"><i class="ti ti-plus"></i></MkButton> | ||||
| 				<MkButton gradate rounded style="margin: 0 auto;" @click="create()"><i class="ti ti-plus"></i></MkButton> | ||||
| 				<MkPagination v-slot="{items}" :pagination="myFlashsPagination"> | ||||
| 					<div class="_gaps_s"> | ||||
| 						<MkFlashPreview v-for="flash in items" :key="flash.id" class="" :flash="flash"/> | ||||
| 						<MkFlashPreview v-for="flash in items" :key="flash.id" :flash="flash"/> | ||||
| 					</div> | ||||
| 				</MkPagination> | ||||
| 			</div> | ||||
| 		</div> | ||||
|  | ||||
| 		<div v-else-if="tab === 'liked'" class=""> | ||||
| 		<div v-else-if="tab === 'liked'"> | ||||
| 			<MkPagination v-slot="{items}" :pagination="likedFlashsPagination"> | ||||
| 				<div class="_gaps_s"> | ||||
| 					<MkFlashPreview v-for="like in items" :key="like.flash.id" class="" :flash="like.flash"/> | ||||
| 					<MkFlashPreview v-for="like in items" :key="like.flash.id" :flash="like.flash"/> | ||||
| 				</div> | ||||
| 			</MkPagination> | ||||
| 		</div> | ||||
| @@ -87,21 +87,3 @@ definePageMetadata(computed(() => ({ | ||||
| 	icon: 'ti ti-player-play', | ||||
| }))); | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" scoped> | ||||
| .rknalgpo { | ||||
| 	&.my .ckltabjg:first-child { | ||||
| 		margin-top: 16px; | ||||
| 	} | ||||
|  | ||||
| 	.ckltabjg:not(:last-child) { | ||||
| 		margin-bottom: 8px; | ||||
| 	} | ||||
|  | ||||
| 	@media (min-width: 500px) { | ||||
| 		.ckltabjg:not(:last-child) { | ||||
| 			margin-bottom: 16px; | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| </style> | ||||
|   | ||||
| @@ -2,14 +2,16 @@ | ||||
| <MkStickyContainer> | ||||
| 	<template #header><MkPageHeader :actions="headerActions" :tabs="headerTabs"/></template> | ||||
| 	<MkSpacer :contentMax="700"> | ||||
| 		<div class="qkcjvfiv"> | ||||
| 			<MkButton primary class="add" @click="create"><i class="ti ti-plus"></i> {{ i18n.ts.createList }}</MkButton> | ||||
| 		<div class="_gaps"> | ||||
| 			<MkButton primary rounded style="margin: 0 auto;" @click="create"><i class="ti ti-plus"></i> {{ i18n.ts.createList }}</MkButton> | ||||
|  | ||||
| 			<MkPagination v-slot="{items}" ref="pagingComponent" :pagination="pagination" class="lists"> | ||||
| 				<MkA v-for="list in items" :key="list.id" class="list _panel" :to="`/my/lists/${ list.id }`"> | ||||
| 					<div class="name">{{ list.name }}</div> | ||||
| 					<MkAvatars :userIds="list.userIds"/> | ||||
| 				</MkA> | ||||
| 			<MkPagination v-slot="{items}" ref="pagingComponent" :pagination="pagination"> | ||||
| 				<div class="_gaps"> | ||||
| 					<MkA v-for="list in items" :key="list.id" class="_panel" :class="$style.list" :to="`/my/lists/${ list.id }`"> | ||||
| 						<div style="margin-bottom: 4px;">{{ list.name }}</div> | ||||
| 						<MkAvatars :userIds="list.userIds"/> | ||||
| 					</MkA> | ||||
| 				</div> | ||||
| 			</MkPagination> | ||||
| 		</div> | ||||
| 	</MkSpacer> | ||||
| @@ -58,29 +60,17 @@ definePageMetadata({ | ||||
| }); | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" scoped> | ||||
| .qkcjvfiv { | ||||
| 	> .add { | ||||
| 		margin: 0 auto var(--margin) auto; | ||||
| 	} | ||||
| <style lang="scss" module> | ||||
| .list { | ||||
| 	display: block; | ||||
| 	padding: 16px; | ||||
| 	border: solid 1px var(--divider); | ||||
| 	border-radius: 6px; | ||||
| 	margin-bottom: 8px; | ||||
|  | ||||
| 	> .lists { | ||||
| 		> .list { | ||||
| 			display: block; | ||||
| 			padding: 16px; | ||||
| 			border: solid 1px var(--divider); | ||||
| 			border-radius: 6px; | ||||
| 			margin-bottom: 8px; | ||||
|  | ||||
| 			&:hover { | ||||
| 				border: solid 1px var(--accent); | ||||
| 				text-decoration: none; | ||||
| 			} | ||||
|  | ||||
| 			> .name { | ||||
| 				margin-bottom: 4px; | ||||
| 			} | ||||
| 		} | ||||
| 	&:hover { | ||||
| 		border: solid 1px var(--accent); | ||||
| 		text-decoration: none; | ||||
| 	} | ||||
| } | ||||
| </style> | ||||
|   | ||||
| @@ -2,22 +2,28 @@ | ||||
| <MkStickyContainer> | ||||
| 	<template #header><MkPageHeader v-model:tab="tab" :actions="headerActions" :tabs="headerTabs"/></template> | ||||
| 	<MkSpacer :contentMax="700"> | ||||
| 		<div v-if="tab === 'featured'" class="rknalgpo"> | ||||
| 		<div v-if="tab === 'featured'"> | ||||
| 			<MkPagination v-slot="{items}" :pagination="featuredPagesPagination"> | ||||
| 				<MkPagePreview v-for="page in items" :key="page.id" class="ckltabjg" :page="page"/> | ||||
| 				<div class="_gaps"> | ||||
| 					<MkPagePreview v-for="page in items" :key="page.id" :page="page"/> | ||||
| 				</div> | ||||
| 			</MkPagination> | ||||
| 		</div> | ||||
|  | ||||
| 		<div v-else-if="tab === 'my'" class="rknalgpo my"> | ||||
| 		<div v-else-if="tab === 'my'" class="_gaps"> | ||||
| 			<MkButton class="new" @click="create()"><i class="ti ti-plus"></i></MkButton> | ||||
| 			<MkPagination v-slot="{items}" :pagination="myPagesPagination"> | ||||
| 				<MkPagePreview v-for="page in items" :key="page.id" class="ckltabjg" :page="page"/> | ||||
| 				<div class="_gaps"> | ||||
| 					<MkPagePreview v-for="page in items" :key="page.id" :page="page"/> | ||||
| 				</div> | ||||
| 			</MkPagination> | ||||
| 		</div> | ||||
|  | ||||
| 		<div v-else-if="tab === 'liked'" class="rknalgpo"> | ||||
| 		<div v-else-if="tab === 'liked'"> | ||||
| 			<MkPagination v-slot="{items}" :pagination="likedPagesPagination"> | ||||
| 				<MkPagePreview v-for="like in items" :key="like.page.id" class="ckltabjg" :page="like.page"/> | ||||
| 				<div class="_gaps"> | ||||
| 					<MkPagePreview v-for="like in items" :key="like.page.id" :page="like.page"/> | ||||
| 				</div> | ||||
| 			</MkPagination> | ||||
| 		</div> | ||||
| 	</MkSpacer> | ||||
| @@ -79,21 +85,3 @@ definePageMetadata(computed(() => ({ | ||||
| 	icon: 'ti ti-note', | ||||
| }))); | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" scoped> | ||||
| .rknalgpo { | ||||
| 	&.my .ckltabjg:first-child { | ||||
| 		margin-top: 16px; | ||||
| 	} | ||||
|  | ||||
| 	.ckltabjg:not(:last-child) { | ||||
| 		margin-bottom: 8px; | ||||
| 	} | ||||
|  | ||||
| 	@media (min-width: 500px) { | ||||
| 		.ckltabjg:not(:last-child) { | ||||
| 			margin-bottom: 16px; | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| </style> | ||||
|   | ||||
| @@ -9,11 +9,11 @@ | ||||
| 		</template> | ||||
| 		<template #default="{items}"> | ||||
| 			<div class="_gaps"> | ||||
| 				<div v-for="token in items" :key="token.id" class="_panel bfomjevm"> | ||||
| 					<img v-if="token.iconUrl" class="icon" :src="token.iconUrl" alt=""/> | ||||
| 					<div class="body"> | ||||
| 						<div class="name">{{ token.name }}</div> | ||||
| 						<div class="description">{{ token.description }}</div> | ||||
| 				<div v-for="token in items" :key="token.id" class="_panel" :class="$style.app"> | ||||
| 					<img v-if="token.iconUrl" :class="$style.appIcon" :src="token.iconUrl" alt=""/> | ||||
| 					<div :class="$style.appBody"> | ||||
| 						<div :class="$style.appName">{{ token.name }}</div> | ||||
| 						<div>{{ token.description }}</div> | ||||
| 						<MkKeyValue oneline> | ||||
| 							<template #key>{{ i18n.ts.installedDate }}</template> | ||||
| 							<template #value><MkTime :time="token.createdAt"/></template> | ||||
| @@ -28,7 +28,7 @@ | ||||
| 								<li v-for="p in token.permission" :key="p">{{ i18n.t(`_permissions.${p}`) }}</li> | ||||
| 							</ul> | ||||
| 						</details> | ||||
| 						<div class="actions"> | ||||
| 						<div> | ||||
| 							<MkButton inline danger @click="revoke(token)"><i class="ti ti-trash"></i></MkButton> | ||||
| 						</div> | ||||
| 					</div> | ||||
| @@ -75,27 +75,27 @@ definePageMetadata({ | ||||
| }); | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" scoped> | ||||
| .bfomjevm { | ||||
| <style lang="scss" module> | ||||
| .app { | ||||
| 	display: flex; | ||||
| 	padding: 16px; | ||||
| } | ||||
|  | ||||
| 	> .icon { | ||||
| 		display: block; | ||||
| 		flex-shrink: 0; | ||||
| 		margin: 0 12px 0 0; | ||||
| 		width: 50px; | ||||
| 		height: 50px; | ||||
| 		border-radius: 8px; | ||||
| 	} | ||||
| .appIcon { | ||||
| 	display: block; | ||||
| 	flex-shrink: 0; | ||||
| 	margin: 0 12px 0 0; | ||||
| 	width: 50px; | ||||
| 	height: 50px; | ||||
| 	border-radius: 8px; | ||||
| } | ||||
|  | ||||
| 	> .body { | ||||
| 		width: calc(100% - 62px); | ||||
| 		position: relative; | ||||
| .appBody { | ||||
| 	width: calc(100% - 62px); | ||||
| 	position: relative; | ||||
| } | ||||
|  | ||||
| 		> .name { | ||||
| 			font-weight: bold; | ||||
| 		} | ||||
| 	} | ||||
| .appName { | ||||
| 	font-weight: bold; | ||||
| } | ||||
| </style> | ||||
|   | ||||
| @@ -16,7 +16,10 @@ | ||||
| 			class="_panel" | ||||
| 			@posted="state = 'posted'" | ||||
| 		/> | ||||
| 		<MkButton v-else-if="state === 'posted'" primary :class="$style.close" @click="close()">{{ i18n.ts.close }}</MkButton> | ||||
| 		<div v-else-if="state === 'posted'" class="_buttonsCenter"> | ||||
| 			<MkButton primary @click="close">{{ i18n.ts.close }}</MkButton> | ||||
| 			<MkButton @click="goToMisskey">{{ i18n.ts.goToMisskey }}</MkButton> | ||||
| 		</div> | ||||
| 	</MkSpacer> | ||||
| </MkStickyContainer> | ||||
| </template> | ||||
| @@ -148,10 +151,14 @@ function close(): void { | ||||
|  | ||||
| 	// 閉じなければ100ms後タイムラインに | ||||
| 	window.setTimeout(() => { | ||||
| 		mainRouter.push('/'); | ||||
| 		location.href = '/'; | ||||
| 	}, 100); | ||||
| } | ||||
|  | ||||
| function goToMisskey(): void { | ||||
| 	location.href = '/'; | ||||
| } | ||||
|  | ||||
| const headerActions = $computed(() => []); | ||||
|  | ||||
| const headerTabs = $computed(() => []); | ||||
| @@ -161,9 +168,3 @@ definePageMetadata({ | ||||
| 	icon: 'ti ti-share', | ||||
| }); | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" module> | ||||
| .close { | ||||
| 	margin: 16px auto; | ||||
| } | ||||
| </style> | ||||
|   | ||||
| @@ -1,37 +1,83 @@ | ||||
| <template> | ||||
| <div> | ||||
| 	{{ i18n.ts.processing }} | ||||
| <div :class="$style.root"> | ||||
| 	<MkAnimBg style="position: fixed; top: 0;"/> | ||||
| 	<div :class="$style.formContainer"> | ||||
| 		<form :class="$style.form" class="_panel" @submit.prevent="submit()"> | ||||
| 			<div :class="$style.banner"> | ||||
| 				<i class="ti ti-user-check"></i> | ||||
| 			</div> | ||||
| 			<div class="_gaps_m" style="padding: 32px;"> | ||||
| 				<div>{{ i18n.t('clickToFinishEmailVerification', { ok: i18n.ts.gotIt }) }}</div> | ||||
| 				<div> | ||||
| 					<MkButton gradate large rounded type="submit" :disabled="submitting" data-cy-admin-ok style="margin: 0 auto;"> | ||||
| 						{{ submitting ? i18n.ts.processing : i18n.ts.gotIt }}<MkEllipsis v-if="submitting"/> | ||||
| 					</MkButton> | ||||
| 				</div> | ||||
| 			</div> | ||||
| 		</form> | ||||
| 	</div> | ||||
| </div> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts" setup> | ||||
| import { onMounted } from 'vue'; | ||||
| import * as os from '@/os'; | ||||
| import { } from 'vue'; | ||||
| import MkButton from '@/components/MkButton.vue'; | ||||
| import MkAnimBg from '@/components/MkAnimBg.vue'; | ||||
| import { login } from '@/account'; | ||||
| import { i18n } from '@/i18n'; | ||||
| import { definePageMetadata } from '@/scripts/page-metadata'; | ||||
| import * as os from '@/os'; | ||||
|  | ||||
| let submitting = $ref(false); | ||||
|  | ||||
| const props = defineProps<{ | ||||
| 	code: string; | ||||
| }>(); | ||||
|  | ||||
| onMounted(async () => { | ||||
| 	await os.alert({ | ||||
| 		type: 'info', | ||||
| 		text: i18n.t('clickToFinishEmailVerification', { ok: i18n.ts.gotIt }), | ||||
| 	}); | ||||
| 	const res = await os.apiWithDialog('signup-pending', { | ||||
| function submit() { | ||||
| 	if (submitting) return; | ||||
| 	submitting = true; | ||||
|  | ||||
| 	os.api('signup-pending', { | ||||
| 		code: props.code, | ||||
| 	}).then(res => { | ||||
| 		return login(res.i, '/'); | ||||
| 	}).catch(() => { | ||||
| 		submitting = false; | ||||
|  | ||||
| 		os.alert({ | ||||
| 			type: 'error', | ||||
| 			text: i18n.ts.somethingHappened, | ||||
| 		}); | ||||
| 	}); | ||||
| 	login(res.i, '/'); | ||||
| }); | ||||
|  | ||||
| const headerActions = $computed(() => []); | ||||
|  | ||||
| const headerTabs = $computed(() => []); | ||||
|  | ||||
| definePageMetadata({ | ||||
| 	title: i18n.ts.signup, | ||||
| 	icon: 'ti ti-user', | ||||
| }); | ||||
| } | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" module> | ||||
| .root { | ||||
| } | ||||
|  | ||||
| .formContainer { | ||||
| 	min-height: 100svh; | ||||
| 	padding: 32px 32px 64px 32px; | ||||
| 	box-sizing: border-box; | ||||
| display: grid; | ||||
| place-content: center; | ||||
| } | ||||
|  | ||||
| .form { | ||||
| 	position: relative; | ||||
| 	z-index: 10; | ||||
| 	border-radius: var(--radius); | ||||
| 	box-shadow: 0 8px 16px rgba(0, 0, 0, 0.1); | ||||
| 	overflow: clip; | ||||
| 	max-width: 500px; | ||||
| } | ||||
|  | ||||
| .banner { | ||||
| 	padding: 16px; | ||||
| 	text-align: center; | ||||
| 	font-size: 26px; | ||||
| 	background-color: var(--accentedBg); | ||||
| 	color: var(--accent); | ||||
| } | ||||
| </style> | ||||
|   | ||||
| @@ -2,15 +2,27 @@ | ||||
| <MkStickyContainer> | ||||
| 	<template #header><MkPageHeader :actions="headerActions" :tabs="headerTabs"/></template> | ||||
| 	<MkSpacer :contentMax="800"> | ||||
| 		<MkNotes class="" :pagination="pagination"/> | ||||
| 		<MkNotes ref="notes" class="" :pagination="pagination"/> | ||||
| 	</MkSpacer> | ||||
| 	<template v-if="$i" #footer> | ||||
| 		<div :class="$style.footer"> | ||||
| 			<MkSpacer :contentMax="800" :marginMin="16" :marginMax="16"> | ||||
| 				<MkButton rounded primary :class="$style.button" @click="post()"><i class="ti ti-pencil"></i>{{ i18n.ts.postToHashtag }}</MkButton> | ||||
| 			</MkSpacer> | ||||
| 		</div> | ||||
| 	</template> | ||||
| </MkStickyContainer> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts" setup> | ||||
| import { computed } from 'vue'; | ||||
| import { computed, ref } from 'vue'; | ||||
| import MkNotes from '@/components/MkNotes.vue'; | ||||
| import MkButton from '@/components/MkButton.vue'; | ||||
| import { definePageMetadata } from '@/scripts/page-metadata'; | ||||
| import { i18n } from '@/i18n'; | ||||
| import { $i } from '@/account'; | ||||
| import { defaultStore } from '@/store'; | ||||
| import * as os from '@/os'; | ||||
|  | ||||
| const props = defineProps<{ | ||||
| 	tag: string; | ||||
| @@ -23,6 +35,16 @@ const pagination = { | ||||
| 		tag: props.tag, | ||||
| 	})), | ||||
| }; | ||||
| const notes = ref<InstanceType<typeof MkNotes>>(); | ||||
|  | ||||
| async function post() { | ||||
| 	defaultStore.set('postFormHashtags', props.tag); | ||||
| 	defaultStore.set('postFormWithHashtags', true); | ||||
| 	await os.post(); | ||||
| 	defaultStore.set('postFormHashtags', ''); | ||||
| 	defaultStore.set('postFormWithHashtags', false); | ||||
| 	notes.value?.pagingComponent?.reload(); | ||||
| } | ||||
|  | ||||
| const headerActions = $computed(() => []); | ||||
|  | ||||
| @@ -33,3 +55,16 @@ definePageMetadata(computed(() => ({ | ||||
| 	icon: 'ti ti-hash', | ||||
| }))); | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" module> | ||||
| .footer { | ||||
| 	-webkit-backdrop-filter: var(--blur, blur(15px)); | ||||
| 	backdrop-filter: var(--blur, blur(15px)); | ||||
| 	border-top: solid 0.5px var(--divider); | ||||
| 	display: flex; | ||||
| } | ||||
|  | ||||
| .button { | ||||
| 		margin: 0 auto var(--margin) auto; | ||||
| } | ||||
| </style> | ||||
|   | ||||
| @@ -34,7 +34,3 @@ const props = defineProps<{ | ||||
| }>(); | ||||
|  | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" scoped> | ||||
|  | ||||
| </style> | ||||
|   | ||||
| @@ -56,6 +56,3 @@ definePageMetadata(computed(() => user ? { | ||||
| 	avatar: user, | ||||
| } : null)); | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" scoped> | ||||
| </style> | ||||
|   | ||||
| @@ -56,6 +56,3 @@ definePageMetadata(computed(() => user ? { | ||||
| 	avatar: user, | ||||
| } : null)); | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" scoped> | ||||
| </style> | ||||
|   | ||||
| @@ -2,21 +2,19 @@ | ||||
| <MkStickyContainer> | ||||
| 	<template #header><MkPageHeader v-model:tab="tab" :actions="headerActions" :tabs="headerTabs"/></template> | ||||
| 	<div> | ||||
| 		<Transition name="fade" mode="out-in"> | ||||
| 			<div v-if="user"> | ||||
| 				<XHome v-if="tab === 'home'" :user="user"/> | ||||
| 				<XTimeline v-else-if="tab === 'notes'" :user="user"/> | ||||
| 				<XActivity v-else-if="tab === 'activity'" :user="user"/> | ||||
| 				<XAchievements v-else-if="tab === 'achievements'" :user="user"/> | ||||
| 				<XReactions v-else-if="tab === 'reactions'" :user="user"/> | ||||
| 				<XClips v-else-if="tab === 'clips'" :user="user"/> | ||||
| 				<XLists v-else-if="tab === 'lists'" :user="user"/> | ||||
| 				<XPages v-else-if="tab === 'pages'" :user="user"/> | ||||
| 				<XGallery v-else-if="tab === 'gallery'" :user="user"/> | ||||
| 			</div> | ||||
| 			<MkError v-else-if="error" @retry="fetchUser()"/> | ||||
| 			<MkLoading v-else/> | ||||
| 		</Transition> | ||||
| 		<div v-if="user"> | ||||
| 			<XHome v-if="tab === 'home'" :user="user"/> | ||||
| 			<XTimeline v-else-if="tab === 'notes'" :user="user"/> | ||||
| 			<XActivity v-else-if="tab === 'activity'" :user="user"/> | ||||
| 			<XAchievements v-else-if="tab === 'achievements'" :user="user"/> | ||||
| 			<XReactions v-else-if="tab === 'reactions'" :user="user"/> | ||||
| 			<XClips v-else-if="tab === 'clips'" :user="user"/> | ||||
| 			<XLists v-else-if="tab === 'lists'" :user="user"/> | ||||
| 			<XPages v-else-if="tab === 'pages'" :user="user"/> | ||||
| 			<XGallery v-else-if="tab === 'gallery'" :user="user"/> | ||||
| 		</div> | ||||
| 		<MkError v-else-if="error" @retry="fetchUser()"/> | ||||
| 		<MkLoading v-else/> | ||||
| 	</div> | ||||
| </MkStickyContainer> | ||||
| </template> | ||||
| @@ -118,14 +116,3 @@ definePageMetadata(computed(() => user ? { | ||||
| 	}, | ||||
| } : null)); | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" scoped> | ||||
| .fade-enter-active, | ||||
| .fade-leave-active { | ||||
| 	transition: opacity 0.125s ease; | ||||
| } | ||||
| .fade-enter-from, | ||||
| .fade-leave-to { | ||||
| 	opacity: 0; | ||||
| } | ||||
| </style> | ||||
|   | ||||
| @@ -77,7 +77,7 @@ | ||||
| 		codeString: '#ffb675', | ||||
| 		codeNumber: '#cfff9e', | ||||
| 		codeBoolean: '#c59eff', | ||||
| 		deckDivider: '#000', | ||||
| 		deckBg: '#000', | ||||
| 		htmlThemeColor: '@bg', | ||||
| 		X2: ':darken<2<@panel', | ||||
| 		X3: 'rgba(255, 255, 255, 0.05)', | ||||
|   | ||||
| @@ -77,7 +77,7 @@ | ||||
| 		codeString: '#b98710', | ||||
| 		codeNumber: '#0fbbbb', | ||||
| 		codeBoolean: '#62b70c', | ||||
| 		deckDivider: ':darken<3<@bg', | ||||
| 		deckBg: ':darken<3<@bg', | ||||
| 		htmlThemeColor: '@bg', | ||||
| 		X2: ':darken<2<@panel', | ||||
| 		X3: 'rgba(0, 0, 0, 0.05)', | ||||
|   | ||||
| @@ -83,6 +83,6 @@ | ||||
| 		fgTransparentWeak: ':alpha<0.75<@fg', | ||||
| 		panelHeaderDivider: 'rgba(0, 0, 0, 0)', | ||||
| 		scrollbarHandleHover: 'rgba(255, 255, 255, 0.4)', | ||||
| 		deckDivider: '#142022', | ||||
| 		deckBg: '#142022', | ||||
| 	}, | ||||
| } | ||||
|   | ||||
| @@ -1,18 +1,18 @@ | ||||
| <template> | ||||
| <div class="dlrsnxqu"> | ||||
| <div :class="$style.root"> | ||||
| 	<div | ||||
| 		v-for="x in defaultStore.reactiveState.statusbars.value" :key="x.id" class="item" :class="[{ black: x.black }, { | ||||
| 			verySmall: x.size === 'verySmall', | ||||
| 			small: x.size === 'small', | ||||
| 			medium: x.size === 'medium', | ||||
| 			large: x.size === 'large', | ||||
| 			veryLarge: x.size === 'veryLarge', | ||||
| 		v-for="x in defaultStore.reactiveState.statusbars.value" :key="x.id" :class="[$style.item, { [$style.black]: x.black, | ||||
| 			[$style.verySmall]: x.size === 'verySmall', | ||||
| 			[$style.small]: x.size === 'small', | ||||
| 			[$style.medium]: x.size === 'medium', | ||||
| 			[$style.large]: x.size === 'large', | ||||
| 			[$style.veryLarge]: x.size === 'veryLarge', | ||||
| 		}]" | ||||
| 	> | ||||
| 		<span class="name">{{ x.name }}</span> | ||||
| 		<XRss v-if="x.type === 'rss'" class="body" :refreshIntervalSec="x.props.refreshIntervalSec" :marqueeDuration="x.props.marqueeDuration" :marqueeReverse="x.props.marqueeReverse" :display="x.props.display" :url="x.props.url" :shuffle="x.props.shuffle"/> | ||||
| 		<XFederation v-else-if="x.type === 'federation'" class="body" :refreshIntervalSec="x.props.refreshIntervalSec" :marqueeDuration="x.props.marqueeDuration" :marqueeReverse="x.props.marqueeReverse" :display="x.props.display" :colored="x.props.colored"/> | ||||
| 		<XUserList v-else-if="x.type === 'userList'" class="body" :refreshIntervalSec="x.props.refreshIntervalSec" :marqueeDuration="x.props.marqueeDuration" :marqueeReverse="x.props.marqueeReverse" :display="x.props.display" :userListId="x.props.userListId"/> | ||||
| 		<span :class="$style.name">{{ x.name }}</span> | ||||
| 		<XRss v-if="x.type === 'rss'" :class="$style.body" :refreshIntervalSec="x.props.refreshIntervalSec" :marqueeDuration="x.props.marqueeDuration" :marqueeReverse="x.props.marqueeReverse" :display="x.props.display" :url="x.props.url" :shuffle="x.props.shuffle"/> | ||||
| 		<XFederation v-else-if="x.type === 'federation'" :class="$style.body" :refreshIntervalSec="x.props.refreshIntervalSec" :marqueeDuration="x.props.marqueeDuration" :marqueeReverse="x.props.marqueeReverse" :display="x.props.display" :colored="x.props.colored"/> | ||||
| 		<XUserList v-else-if="x.type === 'userList'" :class="$style.body" :refreshIntervalSec="x.props.refreshIntervalSec" :marqueeDuration="x.props.marqueeDuration" :marqueeReverse="x.props.marqueeReverse" :display="x.props.display" :userListId="x.props.userListId"/> | ||||
| 	</div> | ||||
| </div> | ||||
| </template> | ||||
| @@ -25,67 +25,67 @@ const XFederation = defineAsyncComponent(() => import('./statusbar-federation.vu | ||||
| const XUserList = defineAsyncComponent(() => import('./statusbar-user-list.vue')); | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" scoped> | ||||
| .dlrsnxqu { | ||||
| <style lang="scss" module> | ||||
| .root { | ||||
| 	font-size: 15px; | ||||
| 	background: var(--panel); | ||||
| } | ||||
|  | ||||
| 	> .item { | ||||
| 		--height: 24px; | ||||
| 		--nameMargin: 10px; | ||||
| 		font-size: 0.85em; | ||||
| .item { | ||||
| 	--height: 24px; | ||||
| 	--nameMargin: 10px; | ||||
| 	font-size: 0.85em; | ||||
|  | ||||
| 		&.verySmall { | ||||
| 			--nameMargin: 7px; | ||||
| 			--height: 16px; | ||||
| 			font-size: 0.75em; | ||||
| 		} | ||||
| 	&.verySmall { | ||||
| 		--nameMargin: 7px; | ||||
| 		--height: 16px; | ||||
| 		font-size: 0.75em; | ||||
| 	} | ||||
|  | ||||
| 		&.small { | ||||
| 			--nameMargin: 8px; | ||||
| 			--height: 20px; | ||||
| 			font-size: 0.8em; | ||||
| 		} | ||||
| 	&.small { | ||||
| 		--nameMargin: 8px; | ||||
| 		--height: 20px; | ||||
| 		font-size: 0.8em; | ||||
| 	} | ||||
|  | ||||
| 		&.large { | ||||
| 			--nameMargin: 12px; | ||||
| 			--height: 26px; | ||||
| 			font-size: 0.875em; | ||||
| 		} | ||||
| 	&.large { | ||||
| 		--nameMargin: 12px; | ||||
| 		--height: 26px; | ||||
| 		font-size: 0.875em; | ||||
| 	} | ||||
|  | ||||
| 		&.veryLarge { | ||||
| 			--nameMargin: 14px; | ||||
| 			--height: 30px; | ||||
| 			font-size: 0.9em; | ||||
| 		} | ||||
| 	&.veryLarge { | ||||
| 		--nameMargin: 14px; | ||||
| 		--height: 30px; | ||||
| 		font-size: 0.9em; | ||||
| 	} | ||||
|  | ||||
| 		display: flex; | ||||
| 		vertical-align: bottom; | ||||
| 		width: 100%; | ||||
| 		line-height: var(--height); | ||||
| 		height: var(--height); | ||||
| 		overflow: clip; | ||||
| 		contain: strict; | ||||
| 	display: flex; | ||||
| 	vertical-align: bottom; | ||||
| 	width: 100%; | ||||
| 	line-height: var(--height); | ||||
| 	height: var(--height); | ||||
| 	overflow: clip; | ||||
| 	contain: strict; | ||||
|  | ||||
| 		> .name { | ||||
| 			padding: 0 var(--nameMargin); | ||||
| 			font-weight: bold; | ||||
| 			color: var(--accent); | ||||
|  | ||||
| 			&:empty { | ||||
| 				display: none; | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		> .body { | ||||
| 			min-width: 0; | ||||
| 			flex: 1; | ||||
| 		} | ||||
|  | ||||
| 		&.black { | ||||
| 			background: #000; | ||||
| 			color: #fff; | ||||
| 		} | ||||
| 	&.black { | ||||
| 		background: #000; | ||||
| 		color: #fff; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| .name { | ||||
| 	padding: 0 var(--nameMargin); | ||||
| 	font-weight: bold; | ||||
| 	color: var(--accent); | ||||
|  | ||||
| 	&:empty { | ||||
| 		display: none; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| .body { | ||||
| 	min-width: 0; | ||||
| 	flex: 1; | ||||
| } | ||||
| </style> | ||||
|   | ||||
| @@ -16,7 +16,7 @@ | ||||
| 				</component> | ||||
| 			</template> | ||||
| 			<div class="divider"></div> | ||||
| 			<MkA v-if="$i.isAdmin || $i.isModerator" v-click-anime v-tooltip="i18n.ts.controlPanel" class="item" activeClass="active" to="/admin" :behavior="settingsWindowed ? 'modalWindow' : null"> | ||||
| 			<MkA v-if="$i.isAdmin || $i.isModerator" v-click-anime v-tooltip="i18n.ts.controlPanel" class="item" activeClass="active" to="/admin" :behavior="settingsWindowed ? 'window' : null"> | ||||
| 				<i class="ti ti-dashboard ti-fw"></i> | ||||
| 			</MkA> | ||||
| 			<button v-click-anime class="item _button" @click="more"> | ||||
| @@ -25,7 +25,7 @@ | ||||
| 			</button> | ||||
| 		</div> | ||||
| 		<div class="right"> | ||||
| 			<MkA v-click-anime v-tooltip="i18n.ts.settings" class="item" activeClass="active" to="/settings" :behavior="settingsWindowed ? 'modalWindow' : null"> | ||||
| 			<MkA v-click-anime v-tooltip="i18n.ts.settings" class="item" activeClass="active" to="/settings" :behavior="settingsWindowed ? 'window' : null"> | ||||
| 				<i class="ti ti-settings ti-fw"></i> | ||||
| 			</MkA> | ||||
| 			<button v-click-anime class="item _button account" @click="openAccountMenu"> | ||||
|   | ||||
| @@ -20,14 +20,14 @@ | ||||
| 		</component> | ||||
| 	</template> | ||||
| 	<div class="divider"></div> | ||||
| 	<MkA v-if="$i.isAdmin || $i.isModerator" v-click-anime class="item" activeClass="active" to="/admin" :behavior="settingsWindowed ? 'modalWindow' : null"> | ||||
| 	<MkA v-if="$i.isAdmin || $i.isModerator" v-click-anime class="item" activeClass="active" to="/admin" :behavior="settingsWindowed ? 'window' : null"> | ||||
| 		<i class="ti ti-dashboard ti-fw"></i><span class="text">{{ i18n.ts.controlPanel }}</span> | ||||
| 	</MkA> | ||||
| 	<button v-click-anime class="item _button" @click="more"> | ||||
| 		<i class="ti ti-dots ti-fw"></i><span class="text">{{ i18n.ts.more }}</span> | ||||
| 		<span v-if="otherNavItemIndicated" class="indicator"><i class="_indicatorCircle"></i></span> | ||||
| 	</button> | ||||
| 	<MkA v-click-anime class="item" activeClass="active" to="/settings" :behavior="settingsWindowed ? 'modalWindow' : null"> | ||||
| 	<MkA v-click-anime class="item" activeClass="active" to="/settings" :behavior="settingsWindowed ? 'window' : null"> | ||||
| 		<i class="ti ti-settings ti-fw"></i><span class="text">{{ i18n.ts.settings }}</span> | ||||
| 	</MkA> | ||||
| 	<div class="divider"></div> | ||||
|   | ||||
| @@ -4,7 +4,7 @@ | ||||
|  | ||||
| 	<div :class="$style.main"> | ||||
| 		<XStatusBars/> | ||||
| 		<div ref="columnsEl" :class="[$style.sections, deckStore.reactiveState.columnAlign.value, { [$style.snapScroll]: snapScroll }]" @contextmenu.self.prevent="onContextmenu"> | ||||
| 		<div ref="columnsEl" :class="[$style.sections, { [$style.center]: deckStore.reactiveState.columnAlign.value === 'center', [$style.snapScroll]: snapScroll }]" @contextmenu.self.prevent="onContextmenu"> | ||||
| 			<!-- sectionを利用しているのは、deck.vue側でcolumnに対してfirst-of-typeを効かせるため --> | ||||
| 			<section | ||||
| 				v-for="ids in layout" | ||||
| @@ -130,6 +130,14 @@ mainRouter.navHook = (path, flag): boolean => { | ||||
| 	return false; | ||||
| }; | ||||
|  | ||||
| if (mainRouter.currentRoute.value.path !== '/') { | ||||
| 	const noMainColumn = !deckStore.state.columns.some(x => x.type === 'main'); | ||||
| 	if (deckStore.state.navWindow || noMainColumn) { | ||||
| 		os.pageWindow(location.pathname + location.search + location.hash); | ||||
| 		mainRouter.replace('/'); | ||||
| 	} | ||||
| } | ||||
|  | ||||
| const isMobile = ref(window.innerWidth <= 500); | ||||
| window.addEventListener('resize', () => { | ||||
| 	isMobile.value = window.innerWidth <= 500; | ||||
| @@ -282,7 +290,7 @@ async function deleteProfile() { | ||||
|  | ||||
| 	--margin: var(--marginHalf); | ||||
|  | ||||
| 	--deckDividerThickness: 5px; | ||||
| 	--columnGap: 6px; | ||||
|  | ||||
| 	display: flex; | ||||
| 	height: 100dvh; | ||||
| @@ -306,14 +314,15 @@ async function deleteProfile() { | ||||
| 	display: flex; | ||||
| 	overflow-x: auto; | ||||
| 	overflow-y: clip; | ||||
| 	background: var(--deckBg); | ||||
|  | ||||
| 	&.center { | ||||
| 		> .section:first-of-type { | ||||
| 			margin-left: auto; | ||||
| 			margin-left: auto !important; | ||||
| 		} | ||||
|  | ||||
| 		> .section:last-of-type { | ||||
| 			margin-right: auto; | ||||
| 			margin-right: auto !important; | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| @@ -327,14 +336,16 @@ async function deleteProfile() { | ||||
| 	flex-direction: column; | ||||
| 	scroll-snap-align: start; | ||||
| 	flex-shrink: 0; | ||||
| 	border-right: solid var(--deckDividerThickness) var(--deckDivider); | ||||
| 	padding-top: var(--columnGap); | ||||
| 	padding-bottom: var(--columnGap); | ||||
| 	padding-right: var(--columnGap); | ||||
|  | ||||
| 	&:first-of-type { | ||||
| 		border-left: solid var(--deckDividerThickness) var(--deckDivider); | ||||
| 		padding-left: var(--columnGap); | ||||
| 	} | ||||
|  | ||||
| 	> .column:not(:last-of-type) { | ||||
| 		border-bottom: solid var(--deckDividerThickness) var(--deckDivider); | ||||
| 		margin-bottom: var(--columnGap); | ||||
| 	} | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -13,6 +13,12 @@ | ||||
| 		@dragend="onDragend" | ||||
| 		@contextmenu.prevent.stop="onContextmenu" | ||||
| 	> | ||||
| 		<svg viewBox="0 0 256 128" :class="$style.tabShape"> | ||||
| 			<g transform="matrix(6.2431,0,0,6.2431,-677.417,-29.3839)"> | ||||
| 				<path d="M149.512,4.707L108.507,4.707C116.252,4.719 118.758,14.958 118.758,14.958C118.758,14.958 121.381,25.283 129.009,25.209L149.512,25.209L149.512,4.707Z" style="fill:var(--deckBg);"/> | ||||
| 			</g> | ||||
| 		</svg> | ||||
| 		<div :class="$style.color"></div> | ||||
| 		<button v-if="isStacked && !isMainColumn" :class="$style.toggleActive" class="_button" @click="toggleActive"> | ||||
| 			<template v-if="active"><i class="ti ti-chevron-up"></i></template> | ||||
| 			<template v-else><i class="ti ti-chevron-down"></i></template> | ||||
| @@ -235,6 +241,7 @@ function onDrop(ev) { | ||||
| 	height: 100%; | ||||
| 	overflow: clip; | ||||
| 	contain: strict; | ||||
| 	border-radius: 10px; | ||||
|  | ||||
| 	&.draghover { | ||||
| 		&:after { | ||||
| @@ -274,6 +281,7 @@ function onDrop(ev) { | ||||
| 	&:not(.active) { | ||||
| 		flex-basis: var(--deckColumnHeaderHeight); | ||||
| 		min-height: var(--deckColumnHeaderHeight); | ||||
| 		border-bottom-right-radius: 0; | ||||
| 	} | ||||
|  | ||||
| 	&.naked { | ||||
| @@ -286,10 +294,22 @@ function onDrop(ev) { | ||||
| 			box-shadow: none; | ||||
| 			color: var(--fg); | ||||
| 		} | ||||
|  | ||||
| 		> .body { | ||||
| 			&::-webkit-scrollbar-track { | ||||
| 				background: inherit; | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	&.paged { | ||||
| 		background: var(--bg) !important; | ||||
|  | ||||
| 		> .body { | ||||
| 			&::-webkit-scrollbar-track { | ||||
| 				background: inherit; | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @@ -299,7 +319,7 @@ function onDrop(ev) { | ||||
| 	z-index: 2; | ||||
| 	line-height: var(--deckColumnHeaderHeight); | ||||
| 	height: var(--deckColumnHeaderHeight); | ||||
| 	padding: 0 16px; | ||||
| 	padding: 0 16px 0 30px; | ||||
| 	font-size: 0.9em; | ||||
| 	color: var(--panelHeaderFg); | ||||
| 	background: var(--panelHeaderBg); | ||||
| @@ -308,6 +328,24 @@ function onDrop(ev) { | ||||
| 	user-select: none; | ||||
| } | ||||
|  | ||||
| .color { | ||||
| 	position: absolute; | ||||
| 	top: 12px; | ||||
| 	left: 12px; | ||||
| 	width: 3px; | ||||
| 	height: calc(100% - 24px); | ||||
| 	background: var(--accent); | ||||
| 	border-radius: 999px; | ||||
| } | ||||
|  | ||||
| .tabShape { | ||||
| 	position: absolute; | ||||
| 	top: 0; | ||||
| 	right: -8px; | ||||
| 	width: auto; | ||||
| 	height: calc(100% - 6px); | ||||
| } | ||||
|  | ||||
| .title { | ||||
| 	display: inline-block; | ||||
| 	align-items: center; | ||||
| @@ -351,5 +389,9 @@ function onDrop(ev) { | ||||
| 	box-sizing: border-box; | ||||
| 	container-type: size; | ||||
| 	background-color: var(--bg); | ||||
|  | ||||
| 	&::-webkit-scrollbar-track { | ||||
| 		background: var(--panel); | ||||
| 	} | ||||
| } | ||||
| </style> | ||||
|   | ||||
| @@ -1,9 +1,9 @@ | ||||
| <template> | ||||
| <MkContainer :naked="widgetProps.transparent" :showHeader="false" data-cy-mkw-clock class="mkw-clock"> | ||||
| 	<div class="vubelbmv" :class="widgetProps.size"> | ||||
| 		<div v-if="widgetProps.label === 'tz' || widgetProps.label === 'timeAndTz'" class="_monospace label a abbrev">{{ tzAbbrev }}</div> | ||||
| <MkContainer :naked="widgetProps.transparent" :showHeader="false" data-cy-mkw-clock> | ||||
| 	<div :class="[$style.root, $style[widgetProps.size]]"> | ||||
| 		<div v-if="widgetProps.label === 'tz' || widgetProps.label === 'timeAndTz'" class="_monospace" :class="[$style.label, $style.a]">{{ tzAbbrev }}</div> | ||||
| 		<MkAnalogClock | ||||
| 			class="clock" | ||||
| 			:class="$style.clock" | ||||
| 			:thickness="widgetProps.thickness" | ||||
| 			:offset="tzOffset" | ||||
| 			:graduations="widgetProps.graduations" | ||||
| @@ -11,8 +11,8 @@ | ||||
| 			:twentyfour="widgetProps.twentyFour" | ||||
| 			:sAnimation="widgetProps.sAnimation" | ||||
| 		/> | ||||
| 		<MkDigitalClock v-if="widgetProps.label === 'time' || widgetProps.label === 'timeAndTz'" class="_monospace label c time" :showS="false" :offset="tzOffset"/> | ||||
| 		<div v-if="widgetProps.label === 'tz' || widgetProps.label === 'timeAndTz'" class="_monospace label d offset">{{ tzOffsetLabel }}</div> | ||||
| 		<MkDigitalClock v-if="widgetProps.label === 'time' || widgetProps.label === 'timeAndTz'" :class="[$style.label, $style.c]" class="_monospace" :showS="false" :offset="tzOffset"/> | ||||
| 		<div v-if="widgetProps.label === 'tz' || widgetProps.label === 'timeAndTz'" class="_monospace" :class="[$style.label, $style.d]">{{ tzOffsetLabel }}</div> | ||||
| 	</div> | ||||
| </MkContainer> | ||||
| </template> | ||||
| @@ -140,39 +140,10 @@ defineExpose<WidgetComponentExpose>({ | ||||
| }); | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" scoped> | ||||
| .vubelbmv { | ||||
| <style lang="scss" module> | ||||
| .root { | ||||
| 	position: relative; | ||||
|  | ||||
| 	> .label { | ||||
| 		position: absolute; | ||||
| 		opacity: 0.7; | ||||
|  | ||||
| 		&.a { | ||||
| 			top: 14px; | ||||
| 			left: 14px; | ||||
| 		} | ||||
|  | ||||
| 		&.b { | ||||
| 			top: 14px; | ||||
| 			right: 14px; | ||||
| 		} | ||||
|  | ||||
| 		&.c { | ||||
| 			bottom: 14px; | ||||
| 			left: 14px; | ||||
| 		} | ||||
|  | ||||
| 		&.d { | ||||
| 			bottom: 14px; | ||||
| 			right: 14px; | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	> .clock { | ||||
| 		margin: auto; | ||||
| 	} | ||||
|  | ||||
| 	&.small { | ||||
| 		padding: 12px; | ||||
|  | ||||
| @@ -197,4 +168,33 @@ defineExpose<WidgetComponentExpose>({ | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| .label { | ||||
| 	position: absolute; | ||||
| 	opacity: 0.7; | ||||
|  | ||||
| 	&.a { | ||||
| 		top: 14px; | ||||
| 		left: 14px; | ||||
| 	} | ||||
|  | ||||
| 	&.b { | ||||
| 		top: 14px; | ||||
| 		right: 14px; | ||||
| 	} | ||||
|  | ||||
| 	&.c { | ||||
| 		bottom: 14px; | ||||
| 		left: 14px; | ||||
| 	} | ||||
|  | ||||
| 	&.d { | ||||
| 		bottom: 14px; | ||||
| 		right: 14px; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| .clock { | ||||
| 	margin: auto; | ||||
| } | ||||
| </style> | ||||
|   | ||||
| @@ -72,7 +72,3 @@ defineExpose<WidgetComponentExpose>({ | ||||
| 	id: props.widget ? props.widget.id : null, | ||||
| }); | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" scoped> | ||||
|  | ||||
| </style> | ||||
|   | ||||
| @@ -1,8 +1,10 @@ | ||||
| <template> | ||||
| <div data-cy-mkw-onlineUsers class="mkw-onlineUsers" :class="{ _panel: !widgetProps.transparent, pad: !widgetProps.transparent }"> | ||||
| 	<I18n v-if="onlineUsersCount" :src="i18n.ts.onlineUsersCount" textTag="span" class="text"> | ||||
| 		<template #n><b>{{ number(onlineUsersCount) }}</b></template> | ||||
| 	</I18n> | ||||
| <div data-cy-mkw-onlineUsers :class="[$style.root, { _panel: !widgetProps.transparent, [$style.pad]: !widgetProps.transparent }]"> | ||||
| 	<span :class="$style.text"> | ||||
| 		<I18n v-if="onlineUsersCount" :src="i18n.ts.onlineUsersCount" textTag="span"> | ||||
| 			<template #n><b style="color: #41b781;">{{ number(onlineUsersCount) }}</b></template> | ||||
| 		</I18n> | ||||
| 	</span> | ||||
| </div> | ||||
| </template> | ||||
|  | ||||
| @@ -55,22 +57,16 @@ defineExpose<WidgetComponentExpose>({ | ||||
| }); | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" scoped> | ||||
| .mkw-onlineUsers { | ||||
| <style lang="scss" module> | ||||
| .root { | ||||
| 	text-align: center; | ||||
|  | ||||
| 	&.pad { | ||||
| 		padding: 16px 0; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| 	> .text { | ||||
| 		::v-deep(b) { | ||||
| 			color: #41b781; | ||||
| 		} | ||||
|  | ||||
| 		::v-deep(span) { | ||||
| 			opacity: 0.7; | ||||
| 		} | ||||
| 	} | ||||
| .text { | ||||
| 	color: var(--fgTransparentWeak); | ||||
| } | ||||
| </style> | ||||
|   | ||||
| @@ -1,11 +1,12 @@ | ||||
| <template> | ||||
| <svg class="hsalcinq" viewBox="0 0 1 1" preserveAspectRatio="none"> | ||||
| <svg :class="$style.root" viewBox="0 0 1 1" preserveAspectRatio="none"> | ||||
| 	<circle | ||||
| 		:r="r" | ||||
| 		cx="50%" cy="50%" | ||||
| 		fill="none" | ||||
| 		stroke-width="0.1" | ||||
| 		stroke="rgba(0, 0, 0, 0.05)" | ||||
| 		:class="$style.circle" | ||||
| 	/> | ||||
| 	<circle | ||||
| 		:r="r" | ||||
| @@ -16,7 +17,7 @@ | ||||
| 		stroke-width="0.1" | ||||
| 		:stroke="color" | ||||
| 	/> | ||||
| 	<text x="50%" y="50%" dy="0.05" text-anchor="middle">{{ (value * 100).toFixed(0) }}%</text> | ||||
| 	<text x="50%" y="50%" dy="0.05" text-anchor="middle" :class="$style.text">{{ (value * 100).toFixed(0) }}%</text> | ||||
| </svg> | ||||
| </template> | ||||
|  | ||||
| @@ -33,20 +34,20 @@ const color = $computed(() => `hsl(${180 - (props.value * 180)}, 80%, 70%)`); | ||||
| const strokeDashoffset = $computed(() => (1 - props.value) * (Math.PI * (r * 2))); | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" scoped> | ||||
| .hsalcinq { | ||||
| <style lang="scss" module> | ||||
| .root { | ||||
| 	display: block; | ||||
| 	height: 100%; | ||||
| } | ||||
|  | ||||
| 	> circle { | ||||
| 		transform-origin: center; | ||||
| 		transform: rotate(-90deg); | ||||
| 		transition: stroke-dashoffset 0.5s ease; | ||||
| 	} | ||||
| .circle { | ||||
| 	transform-origin: center; | ||||
| 	transform: rotate(-90deg); | ||||
| 	transition: stroke-dashoffset 0.5s ease; | ||||
| } | ||||
|  | ||||
| 	> text { | ||||
| 		font-size: 0.15px; | ||||
| 		fill: currentColor; | ||||
| 	} | ||||
| .text { | ||||
| 	font-size: 0.15px; | ||||
| 	fill: currentColor; | ||||
| } | ||||
| </style> | ||||
|   | ||||
| @@ -6,6 +6,7 @@ import { type UserConfig, defineConfig } from 'vite'; | ||||
| import ReactivityTransform from '@vue-macros/reactivity-transform/vite'; | ||||
|  | ||||
| import locales from '../../locales'; | ||||
| import generateDTS from '../../locales/generateDTS'; | ||||
| import meta from '../../package.json'; | ||||
| import pluginJson5 from './vite.json5'; | ||||
|  | ||||
| @@ -64,6 +65,10 @@ export function getConfig(): UserConfig { | ||||
| 					}), | ||||
| 				] | ||||
| 				: [], | ||||
| 			{ | ||||
| 				name: 'locale:generateDTS', | ||||
| 				buildStart: generateDTS, | ||||
| 			}, | ||||
| 		], | ||||
|  | ||||
| 		resolve: { | ||||
|   | ||||
							
								
								
									
										1303
									
								
								pnpm-lock.yaml
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										1303
									
								
								pnpm-lock.yaml
									
									
									
										generated
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
		Reference in New Issue
	
	Block a user