Compare commits
	
		
			163 Commits
		
	
	
		
			2.1.1+36
			...
			3818328afe
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 3818328afe | |||
| 11627e2455 | |||
| 3f82c06ff8 | |||
| 2350f59131 | |||
| 9fe7c9530a | |||
| 52f1826e91 | |||
| 28a4c86dbf | |||
| 85e48ce03b | |||
| efef61a8ea | |||
| 10ead95af9 | |||
| 838ee4d55d | |||
| 13e42429a9 | |||
| c6ce3fe2b7 | |||
| ae9a7eb0fd | |||
| 5d6fb2442f | |||
| 5a85985534 | |||
| c80499db03 | |||
| b8dcdb2315 | |||
| b7b921f1f4 | |||
| 319d5c7d7f | |||
| 4b5b001739 | |||
| db8871a455 | |||
| 38dcaa6066 | |||
| 03275b46ca | |||
| cf3b482fef | |||
| aa4c04d4ef | |||
| 73b82f65e4 | |||
| 9471fe40fe | |||
| 0d1e18735e | |||
| 8bb62b5992 | |||
| 1e8a6dea5b | |||
| 5c2804cc4d | |||
| 0dbb8f132a | |||
| 3395f3dbd0 | |||
| d258ba776e | |||
| 0dcfcaad56 | |||
| 687e720956 | |||
| 180876949e | |||
| 9718965809 | |||
| 5377161fb0 | |||
| 963e538ae5 | |||
| a355e3bf90 | |||
| cb4a2598c8 | |||
| 950612dc07 | |||
| cbd1eaf1af | |||
| ac41cbd99f | |||
| 9f9c90abc4 | |||
| 87029e3538 | |||
| 127d9adc09 | |||
| c82dc7ad85 | |||
| 36bcff7a7c | |||
| 38201b547a | |||
| ed0334fcda | |||
| fbb486b90b | |||
| 9b34f385d5 | |||
| bb7b731602 | |||
| 19076f8136 | |||
| dc77a936ce | |||
| 7f58710c6f | |||
| 068ddcdcdc | |||
| f4e9252ca0 | |||
| 3b1e918117 | |||
| ed7981fdaf | |||
| 9698ca53e4 | |||
| ddc1dc7daf | |||
| 1625a957f8 | |||
| 2dc50d627e | |||
| 2ffde9a3dd | |||
| 5967a91ae1 | |||
| 32c1effcb5 | |||
| 9d0e19c56f | |||
| acf4e634fe | |||
| 25942c2338 | |||
| a4f81f6ba1 | |||
| c1b9090e51 | |||
| f494f70003 | |||
| fb2a55a909 | |||
| 4edfa7fd50 | |||
| d699cac9b1 | |||
| c0428e12c1 | |||
| 55f434ff05 | |||
| f2b3bdda2d | |||
| 1f6bf33b0e | |||
| e2027b1a32 | |||
| 2b3a58b55e | |||
| 6ac536412a | |||
| 52f8ffe4e4 | |||
| aca81431aa | |||
| 1fadd850b7 | |||
| ed2a9a21b6 | |||
| 57279eb3e4 | |||
| c403a2914a | |||
| bcb176344c | |||
| ecf362cffc | |||
| f4ab7671d8 | |||
| a2a3018917 | |||
| 0bdb664000 | |||
| 9c3b61ce57 | |||
| d06df3d278 | |||
| 547ba19e61 | |||
| cb05ff2e9e | |||
| f614da7918 | |||
| a3c8dafff9 | |||
| fa978a7cd1 | |||
| aaa0a562b4 | |||
| 590a4ce2a6 | |||
| f26edce071 | |||
| 603799ea32 | |||
| a32baf7798 | |||
| 498c9af663 | |||
| 202dbff6d3 | |||
| 96fd64d85d | |||
| e236b7f98b | |||
| 5c7929e618 | |||
| 7ba5260246 | |||
| a6d4947a23 | |||
| 7fbd4e9647 | |||
| 95d926b29f | |||
| f6cf6d0440 | |||
| e503c3f02f | |||
| d4fbdd397e | |||
| 03943a7138 | |||
| 44f2c5fe0e | |||
| bb66d5b684 | |||
| 1fca36293d | |||
| 2c7dc8c2ea | |||
| cf0df91d8c | |||
| 91c85e8a58 | |||
| 2851780dda | |||
| 00fd58fb97 | |||
| ee7d0ddd25 | |||
| 7656c08832 | |||
| 619c90cdd9 | |||
| 168d51c9fe | |||
| d4b831f98e | |||
| 4d96a15c31 | |||
| 06dd3e092a | |||
| 82fe9e287a | |||
| dc1c285de1 | |||
| 5a3313e94f | |||
| 61032c84f1 | |||
| 36a5b8fb39 | |||
| 3eda464e03 | |||
| 7a3ab6fd7d | |||
| 3d15c0b9f9 | |||
| 67a29b4305 | |||
| 594f57e0d3 | |||
| d1eb51c596 | |||
| 85d2eff7f8 | |||
| 2375c46852 | |||
| fd2eb5cda6 | |||
| 1256f440bd | |||
| 5b05ca67b6 | |||
| 95af7140cd | |||
| 77e9994204 | |||
| 3f6c186c13 | |||
| 9ac4a940dd | |||
| ec050ab712 | |||
| 77e3ce8bcc | |||
| f5dcf71e10 | |||
| 7fc18b40db | |||
| 8c8ab24c9e | |||
| a319bd7f8c | 
| @@ -1,12 +1,12 @@ | ||||
| { | ||||
|   "sync": { | ||||
|     "region": "solian-next", | ||||
|     "region": "solian", | ||||
|     "configPath": "roadsign.toml" | ||||
|   }, | ||||
|   "deployments": [ | ||||
|     { | ||||
|       "region": "solian-next", | ||||
|       "site": "solian-next-web", | ||||
|       "region": "solian", | ||||
|       "site": "solian-web", | ||||
|       "path": "build/web" | ||||
|     } | ||||
|   ] | ||||
|   | ||||
| @@ -15,6 +15,7 @@ analyzer: | ||||
|     - "**/*.freezed.dart" | ||||
|   errors: | ||||
|     invalid_annotation_target: ignore # Due to freezed + json_serializable issue, ref https://github.com/rrousselGit/freezed/issues/488#issuecomment-894358980 | ||||
|     deprecated_member_use: ignore | ||||
|  | ||||
| linter: | ||||
|   # The lint rules applied to this project can be customized in the | ||||
|   | ||||
| @@ -10,8 +10,9 @@ plugins { | ||||
| } | ||||
|  | ||||
| dependencies { | ||||
|     implementation "androidx.glance:glance:1.1.1" | ||||
|     implementation "androidx.glance:glance-appwidget:1.1.1" | ||||
|     implementation 'com.google.android.material:material:1.12.0' | ||||
|     implementation 'androidx.glance:glance:1.1.1' | ||||
|     implementation 'androidx.glance:glance-appwidget:1.1.1' | ||||
|     implementation 'androidx.compose.foundation:foundation-layout-android:1.7.6' | ||||
|     implementation 'com.google.code.gson:gson:2.10.1' | ||||
|     implementation 'com.squareup.okhttp3:okhttp:4.12.0' | ||||
| @@ -19,6 +20,12 @@ dependencies { | ||||
|     implementation 'io.coil-kt.coil3:coil-network-okhttp:3.0.4' | ||||
| } | ||||
|  | ||||
| def keystoreProperties = new Properties() | ||||
| def keystorePropertiesFile = rootProject.file('key.properties') | ||||
| if (keystorePropertiesFile.exists()) { | ||||
|     keystoreProperties.load(new FileInputStream(keystorePropertiesFile)) | ||||
| } | ||||
|  | ||||
| android { | ||||
|     buildFeatures { | ||||
|         compose true | ||||
| @@ -49,6 +56,15 @@ android { | ||||
|         versionName = flutter.versionName | ||||
|     } | ||||
|  | ||||
|     signingConfigs { | ||||
|         release { | ||||
|             keyAlias = keystoreProperties['keyAlias'] | ||||
|             keyPassword = keystoreProperties['keyPassword'] | ||||
|             storeFile = keystoreProperties['storeFile'] ? file(keystoreProperties['storeFile']) : null | ||||
|             storePassword = keystoreProperties['storePassword'] | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     buildTypes { | ||||
|         debug { | ||||
|             debuggable true | ||||
| @@ -56,9 +72,7 @@ android { | ||||
|             proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro' | ||||
|         } | ||||
|         release { | ||||
|             // TODO: Add your own signing config for the release build. | ||||
|             // Signing with the debug keys for now, so `flutter run --release` works. | ||||
|             signingConfig = signingConfigs.debug | ||||
|             signingConfig = signingConfigs.release | ||||
|  | ||||
|             proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro' | ||||
|         } | ||||
|   | ||||
| @@ -17,11 +17,16 @@ | ||||
|         android:label="Solian" | ||||
|         android:name="${applicationName}" | ||||
|         android:icon="@mipmap/ic_launcher" | ||||
|         android:enableOnBackInvokedCallback="true" | ||||
|         android:requestLegacyExternalStorage="true"> | ||||
|         <meta-data | ||||
|             android:name="flutterEmbedding" | ||||
|             android:value="2" /> | ||||
|  | ||||
|         <activity | ||||
|             android:name=".MainActivity" | ||||
|             android:exported="true" | ||||
|             android:launchMode="singleTask" | ||||
|             android:launchMode="singleInstance" | ||||
|             android:taskAffinity="" | ||||
|             android:theme="@style/LaunchTheme" | ||||
|             android:configChanges="orientation|keyboardHidden|keyboard|screenSize|smallestScreenSize|locale|layoutDirection|fontScale|screenLayout|density|uiMode" | ||||
|   | ||||
| Before Width: | Height: | Size: 1.1 KiB After Width: | Height: | Size: 537 B | 
| Before Width: | Height: | Size: 717 B After Width: | Height: | Size: 372 B | 
| Before Width: | Height: | Size: 1.6 KiB After Width: | Height: | Size: 736 B | 
| Before Width: | Height: | Size: 2.4 KiB After Width: | Height: | Size: 1.1 KiB | 
| Before Width: | Height: | Size: 3.3 KiB After Width: | Height: | Size: 1.5 KiB | 
| @@ -1,4 +1,5 @@ | ||||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <resources> | ||||
|   <color name="ic_launcher_background">#FFFFFFFF</color> | ||||
|   <color name="ic_notification_background">#00000000</color> | ||||
| </resources> | ||||
| @@ -1,7 +1,7 @@ | ||||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <resources> | ||||
|     <!-- Theme applied to the Android Window while the process is starting when the OS's Dark Mode setting is off --> | ||||
|     <style name="LaunchTheme" parent="@android:style/Theme.Light.NoTitleBar"> | ||||
|     <style name="LaunchTheme" parent="Theme.MaterialComponents.Light.NoActionBar"> | ||||
|         <!-- Show a splash screen on the activity. Automatically removed when | ||||
|              the Flutter engine draws its first frame --> | ||||
|         <item name="android:windowBackground">@drawable/launch_background</item> | ||||
| @@ -16,7 +16,7 @@ | ||||
|          running. | ||||
|  | ||||
|          This Theme is only used starting with V2 of Flutter's Android embedding. --> | ||||
|     <style name="NormalTheme" parent="@android:style/Theme.Light.NoTitleBar"> | ||||
|     <style name="NormalTheme" parent="Theme.MaterialComponents.DayNight.NoActionBar"> | ||||
|         <item name="android:windowBackground">?android:colorBackground</item> | ||||
|     </style> | ||||
| </resources> | ||||
|   | ||||
							
								
								
									
										26
									
								
								api/Paperclip/Activate Boost.bru
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,26 @@ | ||||
| meta { | ||||
|   name: Activate Boost | ||||
|   type: http | ||||
|   seq: 1 | ||||
| } | ||||
|  | ||||
| post { | ||||
|   url: {{endpoint}}/cgi/uc/boosts/1/activate | ||||
|   body: none | ||||
|   auth: inherit | ||||
| } | ||||
|  | ||||
| body:json { | ||||
|   { | ||||
|     "client_id": "{{third_client_id}}", | ||||
|     "client_secret":"{{third_client_tk}}", | ||||
|     "type": "general", | ||||
|     "subject": "Merry Christmas!", | ||||
|     "subtitle": "一条来自 Solar Network 团队的信息", | ||||
|     "content": "今天是 12 月 25 日 (UTC+8),小羊祝您圣诞快乐 🎄", | ||||
|     "metadata": { | ||||
|       "image": "6EqsYQwmFRCkbmhR" | ||||
|     }, | ||||
|     "priority": 10 | ||||
|   } | ||||
| } | ||||
							
								
								
									
										19
									
								
								api/Paperclip/Stickers/Create Sticker Pack.bru
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,19 @@ | ||||
| meta { | ||||
|   name: Create Sticker Pack | ||||
|   type: http | ||||
|   seq: 1 | ||||
| } | ||||
|  | ||||
| post { | ||||
|   url: {{endpoint}}/cgi/uc/stickers/packs | ||||
|   body: json | ||||
|   auth: inherit | ||||
| } | ||||
|  | ||||
| body:json { | ||||
|   { | ||||
|     "prefix": "cat", | ||||
|     "name": "Solar Network full of Cats!", | ||||
|     "description": "The sticker packs is full of stickers which related with cats!" | ||||
|   } | ||||
| } | ||||
							
								
								
									
										20
									
								
								api/Paperclip/Stickers/Create Sticker.bru
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,20 @@ | ||||
| meta { | ||||
|   name: Create Sticker | ||||
|   type: http | ||||
|   seq: 2 | ||||
| } | ||||
|  | ||||
| post { | ||||
|   url: {{endpoint}}/cgi/uc/stickers | ||||
|   body: json | ||||
|   auth: inherit | ||||
| } | ||||
|  | ||||
| body:json { | ||||
|   { | ||||
|     "alias": "Meltdown", | ||||
|     "name": "Meltdown", | ||||
|     "attachment_id": "IpDPHEbWDDCbBofX", | ||||
|     "pack_id": 4 | ||||
|   } | ||||
| } | ||||
							
								
								
									
										11
									
								
								api/Paperclip/Stickers/Get Sticker Packs.bru
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,11 @@ | ||||
| meta { | ||||
|   name: Get Sticker Packs | ||||
|   type: http | ||||
|   seq: 3 | ||||
| } | ||||
|  | ||||
| get { | ||||
|   url: {{endpoint}}/cgi/uc/stickers/packs | ||||
|   body: none | ||||
|   auth: none | ||||
| } | ||||
							
								
								
									
										15
									
								
								api/Paperclip/Stickers/Get Stickers.bru
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,15 @@ | ||||
| meta { | ||||
|   name: Get Stickers | ||||
|   type: http | ||||
|   seq: 4 | ||||
| } | ||||
|  | ||||
| get { | ||||
|   url: {{endpoint}}/cgi/uc/stickers?take=10 | ||||
|   body: none | ||||
|   auth: none | ||||
| } | ||||
|  | ||||
| params:query { | ||||
|   take: 10 | ||||
| } | ||||
							
								
								
									
										26
									
								
								api/Passport/Developer Notify All Users.bru
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,26 @@ | ||||
| meta { | ||||
|   name: Developer Notify All Users | ||||
|   type: http | ||||
|   seq: 1 | ||||
| } | ||||
|  | ||||
| post { | ||||
|   url: {{endpoint}}/cgi/id/dev/notify/all | ||||
|   body: json | ||||
|   auth: inherit | ||||
| } | ||||
|  | ||||
| body:json { | ||||
|   { | ||||
|     "client_id": "{{third_client_id}}", | ||||
|     "client_secret":"{{third_client_tk}}", | ||||
|     "type": "general", | ||||
|     "subject": "新年快乐!", | ||||
|     "subtitle": "一条来自 Solar Network 团队的信息", | ||||
|     "content": "今天是农历正月初一,小羊祝您新年快乐 🎉", | ||||
|     "metadata": { | ||||
|       "image": "D2EDbcrsTugs3xk5" | ||||
|     }, | ||||
|     "priority": 10 | ||||
|   } | ||||
| } | ||||
							
								
								
									
										26
									
								
								api/Passport/Developer Notify One User.bru
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,26 @@ | ||||
| meta { | ||||
|   name: Developer Notify One User | ||||
|   type: http | ||||
|   seq: 2 | ||||
| } | ||||
|  | ||||
| post { | ||||
|   url: {{endpoint}}/cgi/id/dev/notify/1 | ||||
|   body: json | ||||
|   auth: inherit | ||||
| } | ||||
|  | ||||
| body:json { | ||||
|   { | ||||
|     "client_id": "{{third_client_id}}", | ||||
|     "client_secret":"{{third_client_tk}}", | ||||
|     "type": "general", | ||||
|     "subject": "测试", | ||||
|     "subtitle": "Alphabot です", | ||||
|     "content": "全新通知动画", | ||||
|     "metadata": { | ||||
|       "image": "D2EDbcrsTugs3xk5" | ||||
|     }, | ||||
|     "priority": 10 | ||||
|   } | ||||
| } | ||||
							
								
								
									
										11
									
								
								api/Reader/List News Sources.bru
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,11 @@ | ||||
| meta { | ||||
|   name: List News Sources | ||||
|   type: http | ||||
|   seq: 3 | ||||
| } | ||||
|  | ||||
| get { | ||||
|   url: {{endpoint}}/cgi/re/well-known/sources | ||||
|   body: none | ||||
|   auth: none | ||||
| } | ||||
							
								
								
									
										17
									
								
								api/Reader/List News.bru
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,17 @@ | ||||
| meta { | ||||
|   name: List News | ||||
|   type: http | ||||
|   seq: 2 | ||||
| } | ||||
|  | ||||
| get { | ||||
|   url: {{endpoint}}/cgi/re/news?take=10&offset=0&source=shadiao | ||||
|   body: none | ||||
|   auth: none | ||||
| } | ||||
|  | ||||
| params:query { | ||||
|   take: 10 | ||||
|   offset: 0 | ||||
|   source: shadiao | ||||
| } | ||||
							
								
								
									
										18
									
								
								api/Reader/Trigger Scan News.bru
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,18 @@ | ||||
| meta { | ||||
|   name: Trigger Scan News | ||||
|   type: http | ||||
|   seq: 1 | ||||
| } | ||||
|  | ||||
| post { | ||||
|   url: {{endpoint}}/cgi/re/admin/scan | ||||
|   body: json | ||||
|   auth: inherit | ||||
| } | ||||
|  | ||||
| body:json { | ||||
|   { | ||||
|     "sources": ["taiwan-ltn"], | ||||
|     "eager": true | ||||
|   } | ||||
| } | ||||
							
								
								
									
										20
									
								
								api/Wallet/Create Order.bru
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,20 @@ | ||||
| meta { | ||||
|   name: Create Order | ||||
|   type: http | ||||
|   seq: 1 | ||||
| } | ||||
|  | ||||
| post { | ||||
|   url: {{endpoint}}/cgi/wa/orders | ||||
|   body: json | ||||
|   auth: none | ||||
| } | ||||
|  | ||||
| body:json { | ||||
|   { | ||||
|     "client_id": "highland-mc", | ||||
|     "client_secret": "(3^DLAvo3v", | ||||
|     "remark": "我是秦始皇,现在被困香港,现在 SN 转我 500 源点,帮助我回到咸阳,到时候封你为太监一职。", | ||||
|     "amount": 500 | ||||
|   } | ||||
| } | ||||
							
								
								
									
										21
									
								
								api/Wallet/Create Transaction.bru
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,21 @@ | ||||
| meta { | ||||
|   name: Create Transaction | ||||
|   type: http | ||||
|   seq: 3 | ||||
| } | ||||
|  | ||||
| post { | ||||
|   url: {{endpoint}}/cgi/wa/transactions | ||||
|   body: json | ||||
|   auth: none | ||||
| } | ||||
|  | ||||
| body:json { | ||||
|   { | ||||
|     "client_id": "alphabot", | ||||
|     "client_secret": "_uR0sVnHTh", | ||||
|     "remark": "新年红包", | ||||
|     "amount": 9705, | ||||
|     "payee_id": 2 | ||||
|   } | ||||
| } | ||||
							
								
								
									
										20
									
								
								api/Wallet/Get Order.bru
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,20 @@ | ||||
| meta { | ||||
|   name: Get Order | ||||
|   type: http | ||||
|   seq: 2 | ||||
| } | ||||
|  | ||||
| get { | ||||
|   url: {{endpoint}}/cgi/wa/orders/4 | ||||
|   body: none | ||||
|   auth: none | ||||
| } | ||||
|  | ||||
| body:json { | ||||
|   { | ||||
|     "client_id": "highland-mc", | ||||
|     "client_secret": "(3^DLAvo3v", | ||||
|     "remark": "我是秦始皇,现在被困香港,现在 SN 转我 500 源点,帮助我回到咸阳,到时候封你为太监一职。", | ||||
|     "amount": 500 | ||||
|   } | ||||
| } | ||||
							
								
								
									
										20
									
								
								api/Wallet/Get Transaction.bru
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,20 @@ | ||||
| meta { | ||||
|   name: Get Transaction | ||||
|   type: http | ||||
|   seq: 4 | ||||
| } | ||||
|  | ||||
| get { | ||||
|   url: {{endpoint}}/cgi/wa/transactions/67 | ||||
|   body: none | ||||
|   auth: inherit | ||||
| } | ||||
|  | ||||
| body:json { | ||||
|   { | ||||
|     "client_id": "highland-mc", | ||||
|     "client_secret": "(3^DLAvo3v", | ||||
|     "remark": "我是秦始皇,现在被困香港,现在 SN 转我 500 源点,帮助我回到咸阳,到时候封你为太监一职。", | ||||
|     "amount": 500 | ||||
|   } | ||||
| } | ||||
							
								
								
									
										11
									
								
								api/WatchTower/Run Database Maintenance.bru
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,11 @@ | ||||
| meta { | ||||
|   name: Run Database Maintenance | ||||
|   type: http | ||||
|   seq: 1 | ||||
| } | ||||
|  | ||||
| post { | ||||
|   url: {{endpoint}}/wt/maintenance/database | ||||
|   body: none | ||||
|   auth: inherit | ||||
| } | ||||
							
								
								
									
										9
									
								
								api/bruno.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,9 @@ | ||||
| { | ||||
|   "version": "1", | ||||
|   "name": "Solar Network", | ||||
|   "type": "collection", | ||||
|   "ignore": [ | ||||
|     "node_modules", | ||||
|     ".git" | ||||
|   ] | ||||
| } | ||||
							
								
								
									
										7
									
								
								api/collection.bru
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,7 @@ | ||||
| auth { | ||||
|   mode: bearer | ||||
| } | ||||
|  | ||||
| auth:bearer { | ||||
|   token: {{atk}} | ||||
| } | ||||
							
								
								
									
										8
									
								
								api/environments/Prod.bru
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,8 @@ | ||||
| vars { | ||||
|   endpoint: https://api.sn.solsynth.dev | ||||
|   third_client_id: alphabot | ||||
| } | ||||
| vars:secret [ | ||||
|   atk, | ||||
|   third_client_tk | ||||
| ] | ||||
							
								
								
									
										
											BIN
										
									
								
								assets/icon/tray-icon.ico
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 16 KiB | 
							
								
								
									
										
											BIN
										
									
								
								assets/icon/tray-icon.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 228 KiB | 
| @@ -17,6 +17,10 @@ | ||||
|   "screenAccountProfileEdit": "Edit Profile", | ||||
|   "screenAbuseReport": "Abuse Reports", | ||||
|   "screenSettings": "Settings", | ||||
|   "screenAccountSettings": "Account Settings", | ||||
|   "screenFactorSettings": "Auth Factors", | ||||
|   "screenAccountWallet": "Wallet", | ||||
|   "screenNews": "News", | ||||
|   "screenAlbum": "Album", | ||||
|   "screenChat": "Chat", | ||||
|   "screenChatManage": "Edit Channel", | ||||
| @@ -57,7 +61,7 @@ | ||||
|   "reply": "Reply", | ||||
|   "unset": "Unset", | ||||
|   "untitled": "Untitled", | ||||
|   "postDetail": "Post detail", | ||||
|   "postDetail": "Post Detail", | ||||
|   "postNoun": "Post", | ||||
|   "postReadMore": "Read more", | ||||
|   "postReadEstimate": "Est read time {}", | ||||
| @@ -103,8 +107,18 @@ | ||||
|   }, | ||||
|   "loginEnterPassword": "Enter the code", | ||||
|   "loginSuccess": "Logged in as {}", | ||||
|   "authFactorDelete": "Delete Auth Factor", | ||||
|   "authFactorDeleteDescription": "Are you sure you want delete auth factor {}?", | ||||
|   "authFactorPassword": "Password", | ||||
|   "authFactorPasswordDescription": "The password you set when you registered.", | ||||
|   "authFactorEmail": "Email verification code", | ||||
|   "authFactorEmailDescription": "An one-time code sent to the email address you set when you registered.", | ||||
|   "authFactorTOTP": "Time-based OTP", | ||||
|   "authFactorTOTPDescription": "A one-time code generated by a TOTP authenticator such as Google Authenticator or Authy.", | ||||
|   "authFactorInAppNotify": "In-app notification", | ||||
|   "authFactorInAppNotifyDescription": "A one-time code sent via in-app notification.", | ||||
|   "authFactorAdd": "Add a factor", | ||||
|   "authFactorAddSubtitle": "Provide another way to login your account.", | ||||
|   "accountIntroTitle": "Hello there!", | ||||
|   "accountIntroSubtitle": "Pick an option below to get started.", | ||||
|   "accountLogout": "Logout", | ||||
| @@ -113,8 +127,14 @@ | ||||
|   "accountLogoutConfirm": "You will need to re-enter your account password, even if you have already done so. This is required to login again.", | ||||
|   "accountPublishers": "Your publishers", | ||||
|   "accountPublishersSubtitle": "Manage your publish identities.", | ||||
|   "accountSettings": "Account Settings", | ||||
|   "accountSettingsSubtitle": "Manage your account and make it yours.", | ||||
|   "accountProfileEdit": "Edit your profile", | ||||
|   "accountProfileEditSubtitle": "Make your Solarpass account more looks like you.", | ||||
|   "accountWallet": "Wallet", | ||||
|   "accountWalletSubtitle": "View your balance and transactions.", | ||||
|   "factorSettings": "Auth Factors", | ||||
|   "factorSettingsSubtitle": "Manage your authentication factors.", | ||||
|   "accountProfileEditApplied": "Profile modification applied.", | ||||
|   "publishersNew": "New Publisher", | ||||
|   "publisherNewSubtitle": "Create a new publisher identity.", | ||||
| @@ -139,6 +159,9 @@ | ||||
|   "fieldPostTitle": "Title", | ||||
|   "fieldPostDescription": "Description", | ||||
|   "fieldPostTags": "Tags", | ||||
|   "fieldPostCategories": "Categories", | ||||
|   "fieldPostAlias": "Alias", | ||||
|   "fieldPostAliasHint": "Optional, used to represent the post in URL, should follow URL-Safe.", | ||||
|   "postPublish": "Publish", | ||||
|   "postPosted": "Post has been posted.", | ||||
|   "postPublishedAt": "Published At", | ||||
| @@ -176,12 +199,30 @@ | ||||
|     "other": "{} comments" | ||||
|   }, | ||||
|   "settingsAppearance": "Appearance", | ||||
|   "settingsDisplayLanguage": "Display Language", | ||||
|   "settingsDisplayLanguageDescription": "Set the application language.", | ||||
|   "settingsDisplayLanguageSystem": "Follow System", | ||||
|   "settingsAppBarTransparent": "Transparent App Bar", | ||||
|   "settingsAppBarTransparentDescription": "Enable transparent effect for the app bar.", | ||||
|   "settingsDrawerPreferCollapse": "Prefer Drawer Collapse", | ||||
|   "settingsDrawerPreferCollapseDescription": "Make the drawer to collapse even when the screen is wide enough.", | ||||
|   "settingsBackgroundImage": "Background Image", | ||||
|   "settingsBackgroundImageDescription": "Set the background image that will be applied globally.", | ||||
|   "settingsBackgroundImageClear": "Clear Existing Background Image", | ||||
|   "settingsBackgroundImageClearDescription": "Reset the background image to blank.", | ||||
|   "settingsThemeMaterial3": "Use Material You Design", | ||||
|   "settingsThemeMaterial3Description": "Set the application theme to Material 3 Design.", | ||||
|   "settingsColorScheme": "Color Scheme", | ||||
|   "settingsColorSchemeDescription": "Set the application primary color.", | ||||
|   "settingsColorSeed": "Color Seed", | ||||
|   "settingsColorSeedDescription": "Select one of the present color schemes.", | ||||
|   "settingsFeatures": "Features", | ||||
|   "settingsNotifyWithHaptic": "Haptic when Notified", | ||||
|   "settingsNotifyWithHapticDescription": "Vibrate lightly when a new notification appears in the foreground.", | ||||
|   "settingsExpandPostLink": "Expand Post Link", | ||||
|   "settingsExpandPostLinkDescription": "Expand the post link in the post list.", | ||||
|   "settingsExpandChatLink": "Expand Chat Link", | ||||
|   "settingsExpandChatLinkDescription": "Expand the chat link in the chat list.", | ||||
|   "settingsNetwork": "Network", | ||||
|   "settingsNetworkServer": "HyperNet Server", | ||||
|   "settingsNetworkServerDescription": "Set the HyperNet server address, choose ours or build your own.", | ||||
| @@ -200,12 +241,15 @@ | ||||
|   "settingsMisc": "Misc", | ||||
|   "settingsMiscAbout": "About", | ||||
|   "settingsMiscAboutDescription": "View the version information of Solian.", | ||||
|   "settingsAccountLanguage": "Account Language", | ||||
|   "settingsAccountLanguageDescription": "Set the language for email, notification, and other account-related content.", | ||||
|   "sensitiveContent": "Sensitive Content", | ||||
|   "sensitiveContentCollapsed": "Sensitive content has been collapsed.", | ||||
|   "sensitiveContentDescription": "This content has been marked as sensitive, and may not be suitable for all viewers.", | ||||
|   "sensitiveContentReveal": "Reveal", | ||||
|   "serverConnecting": "Connecting to server...", | ||||
|   "serverDisconnected": "Lost connection from server", | ||||
|   "serverConnecting": "Connecting...", | ||||
|   "serverDisconnected": "Connection Lost", | ||||
|   "serverConnected": "Connected", | ||||
|   "fieldChatAlias": "Channel Alias", | ||||
|   "fieldChatAliasHint": "The unique channel alias within the site, used to represent the channel in URL, leave blank to auto generate. Should be URL-Safe.", | ||||
|   "fieldChatName": "Name", | ||||
| @@ -272,16 +316,50 @@ | ||||
|     "one": "{} attachment", | ||||
|     "other": "{} attachments" | ||||
|   }, | ||||
|   "messageTyping": { | ||||
|     "one": "{} is typing...", | ||||
|     "other": "{} are typing..." | ||||
|   }, | ||||
|   "fieldAttachmentRandomId": "Random ID", | ||||
|   "fieldAttachmentAlt": "Alternative text", | ||||
|   "addAttachmentFromAlbum": "Add from album", | ||||
|   "addAttachmentFromClipboard": "Paste file", | ||||
|   "addAttachmentFromCameraPhoto": "Take photo", | ||||
|   "addAttachmentFromCameraVideo": "Take video", | ||||
|   "addAttachmentFromRandomId": "Link via RID", | ||||
|   "attachmentDetailInfo": "Attachment details", | ||||
|   "attachmentPastedImage": "Pasted Image", | ||||
|   "attachmentInsertLink": "Insert Link", | ||||
|   "attachmentSetAsPostThumbnail": "Set as post thumbnail", | ||||
|   "attachmentUnsetAsPostThumbnail": "Unset as post thumbnail", | ||||
|   "attachmentCompressVideo": "Re-encode video", | ||||
|   "attachmentSetThumbnail": "Set thumbnail", | ||||
|   "attachmentSetAlt": "Set alternative text", | ||||
|   "attachmentCopyRandomId": "Copy RID", | ||||
|   "attachmentUpload": "Upload", | ||||
|   "attachmentInputDialog": "Upload attachments", | ||||
|   "attachmentInputUseRandomId": "Use Random ID", | ||||
|   "attachmentInputNew": "New Upload", | ||||
|   "waitingForUpload": "Waiting for upload", | ||||
|   "attachmentVideoCompressHint": "Compress a copy of this video", | ||||
|   "attachmentVideoCompressHintDescription": "Do you want to upload a compress copy of video {}? It will help your audience to preview this video faster and they still can watch the original video. It will take some while to process the video on your device, so please be patient.", | ||||
|   "attachmentCompressQuality": "Compress quality", | ||||
|   "attachmentCompressQualityHighest": "Highest", | ||||
|   "attachmentCompressQualityDefault": "Default", | ||||
|   "attachmentCompressQualityMedium": "Medium", | ||||
|   "attachmentCompressQualityLow": "Low", | ||||
|   "attachmentCompressQualityHint": "Solar Network doesn't prevent you from uploading large files, high resolution, high bitrate videos. But for your network conditions, we suggest you choose a suitable compression quality.", | ||||
|   "attachmentUploaded": "Uploaded", | ||||
|   "attachmentPending": "Pending", | ||||
|   "attachmentCopyCompressed": "Copy compressed", | ||||
|   "attachmentGotBoosted": "Boosted", | ||||
|   "attachmentBoost": "Boost", | ||||
|   "attachmentCreateBoost": "Create Boost", | ||||
|   "attachmentBoostHint": "Boost is a feature that allows you to upload attachments to a server closer to your audience or a faster content network. This feature is currently in beta and is subject to change. It's all free for now, you can feel free to try, you will get notified when the pricing plan changed.", | ||||
|   "attachmentDestinationRegion": "Destination Region", | ||||
|   "attachmentDestinationRegionAPAC": "Asia Pacific", | ||||
|   "attachmentDestinationRegionNGB": "Ning Bo, China, Zhejiang", | ||||
|   "attachmentDestinationRegionHKG": "Hong Kong", | ||||
|   "notification": "Notification", | ||||
|   "notificationUnreadCount": { | ||||
|     "zero": "All notifications read", | ||||
| @@ -369,9 +447,32 @@ | ||||
|   "dailyCheckNegativeHint5Description": "Lost connection at a crucial moment", | ||||
|   "dailyCheckNegativeHint6": "Going out", | ||||
|   "dailyCheckNegativeHint6Description": "Forgot your umbrella and got caught in the rain", | ||||
|   "happyBirthday": "Happy birthday, {}!", | ||||
|   "celebrateBirthday": "Happy birthday, {}!", | ||||
|   "celebrateMerryXmas": "Merry christmas, {}!", | ||||
|   "celebrateNewYear": "Happy new year, {}!", | ||||
|   "celebrateLunarNewYear": "Happy lunar new year, {}!", | ||||
|   "celebrateMidAutumn": "Happy mid-autumn festival, {}!", | ||||
|   "celebrateDragonBoat": "Happy dragon boat festival, {}!", | ||||
|   "celebrateValentineDay": "Today is valentine's day, {}!", | ||||
|   "celebrateLaborDay": "Today is labor day, {}.", | ||||
|   "celebrateMotherDay": "Today is mother's day, {}.", | ||||
|   "celebrateChildrenDay": "Today is children's day, {}!", | ||||
|   "celebrateFatherDay": "Today is father's day, {}.", | ||||
|   "celebrateHalloween": "Happy halloween, {}!", | ||||
|   "celebrateThanksgiving": "Today is thanksgiving day, {}!", | ||||
|   "pendingBirthday": "Birthday in {}", | ||||
|   "pendingMerryXmas": "Christmas in {}", | ||||
|   "pendingLunarNewYear": "Lunar new year in {}", | ||||
|   "pendingMidAutumn": "Mid-autumn festival in {}", | ||||
|   "pendingDragonBoat": "Dragon boat festival in {}", | ||||
|   "pendingNewYear": "New year in {}", | ||||
|   "pendingValentineDay": "Valentine's day in {}", | ||||
|   "pendingLaborDay": "Labor day in {}", | ||||
|   "pendingMotherDay": "Mother's day in {}", | ||||
|   "pendingChildrenDay": "Children's day in {}", | ||||
|   "pendingFatherDay": "Father's day in {}", | ||||
|   "pendingHalloween": "Halloween in {}", | ||||
|   "pendingThanksgiving": "Thanksgiving day in {}", | ||||
|   "friendNew": "Add Friend", | ||||
|   "friendRequests": "Friend Requests", | ||||
|   "friendRequestsDescription": { | ||||
| @@ -405,6 +506,7 @@ | ||||
|   "accountJoinedAt": "Joined at {}", | ||||
|   "accountBirthday": "Born on {}", | ||||
|   "accountBadge": "Badge", | ||||
|   "accountCheckInNoRecords": "No check-in records", | ||||
|   "badgeCompanyStaff": "Solsynth Staff", | ||||
|   "badgeSiteMigration": "Solar Network Native", | ||||
|   "accountStatus": "Status", | ||||
| @@ -413,6 +515,7 @@ | ||||
|   "accountStatusLastSeen": "Last seen at {}", | ||||
|   "postArticle": "Article on the Solar Network", | ||||
|   "postStory": "Story on the Solar Network", | ||||
|   "postLocalDraftRestored": "Restored from device", | ||||
|   "articleWrittenAt": "Written at {}", | ||||
|   "articleEditedAt": "Edited at {}", | ||||
|   "attachmentSaved": "Saved to album", | ||||
| @@ -448,16 +551,64 @@ | ||||
|   "publisherBlockHintDescription": "You are going to block this publisher's maintainer, this will also block publishers that run by the same user.", | ||||
|   "userUnblocked": "{} has been unblocked.", | ||||
|   "userBlocked": "{} has been blocked.", | ||||
|   "postSharingViaPicture": "Capturing post as picture, please stand by...", | ||||
|   "postSharingViaPicture": "Capturing post as picture, please wait...", | ||||
|   "postImageShareReadMore": "Scan the QR code to read full post", | ||||
|   "postImageShareAds": "Explore posts on the Solar Network", | ||||
|   "postShare": "Share", | ||||
|   "postShareImage": "Share via Image", | ||||
|   "postGetInsight": "Get Insight", | ||||
|   "postGetInsightTitle": "AI Insight", | ||||
|   "postGetInsightDescription": "AI may make mistakes, check important information.", | ||||
|   "appInitializing": "Initializing", | ||||
|   "poweredBy": "Powered by {}", | ||||
|   "shareIntent": "Share", | ||||
|   "shareIntentDescription":  "What do you want to do with the content you are sharing?", | ||||
|   "shareIntentDescription": "What do you want to do with the content you are sharing?", | ||||
|   "shareIntentPostStory": "Post a Story", | ||||
|   "shareIntentSendChannel": "Share to Channel", | ||||
|   "updateAvailable": "Update Available", | ||||
|   "updateOngoing": "正在更新,请稍后..." | ||||
|   "updateOngoing": "Updating, please wait...", | ||||
|   "custom": "Custom", | ||||
|   "colorSchemeIndigo": "Indigo", | ||||
|   "colorSchemeBlue": "Blue", | ||||
|   "colorSchemeGreen": "Green", | ||||
|   "colorSchemeYellow": "Yellow", | ||||
|   "colorSchemeOrange": "Orange", | ||||
|   "colorSchemeRed": "Red", | ||||
|   "colorSchemeWhite": "White", | ||||
|   "colorSchemeBlack": "Black", | ||||
|   "colorSchemeApplied": "Color scheme has been applied, may need restart the app to take effect.", | ||||
|   "postFeaturedComment": "Featured Comment", | ||||
|   "postCategoryTechnology": "Technology", | ||||
|   "postCategoryGaming": "Gaming", | ||||
|   "postCategoryLife": "Life", | ||||
|   "postCategoryArts": "Arts", | ||||
|   "postCategorySports": "Sports", | ||||
|   "postCategoryMusic": "Music", | ||||
|   "postCategoryNews": "News", | ||||
|   "postCategoryKnowledge": "Knowledge", | ||||
|   "postCategoryLiterature": "Literature", | ||||
|   "postCategoryFunny": "Funny", | ||||
|   "postCategoryUncategorized": "Uncategorized", | ||||
|   "newsAllSources": "All News", | ||||
|   "newsReadingProviderSwap": "Swap", | ||||
|   "newsReadingFromReader": "You're reading from HyperNet.Reader", | ||||
|   "newsReadingFromOriginal": "You're reading the original article", | ||||
|   "newsDisclaimer": "This article is fetched from the Internet, we do not guarantee its authenticity, please judge for yourself. All content in this article belongs to the original author.", | ||||
|   "newsToday": "Today's News", | ||||
|   "totpPostSetup": "One More Thing", | ||||
|   "totpPostSetupDescription": "Scan the QR Code below with Google Authenticator, Microsoft Authenticator, 1Password, Authy, Bitwarden or any of kind of authenticator app which supports TOTP.", | ||||
|   "totpNeverShare": "Never share this QR Code", | ||||
|   "needHelp": "Need Help?", | ||||
|   "needHelpLaunch": "Check out our Goatpedia!", | ||||
|   "walletCreate": "Create a Wallet", | ||||
|   "walletCreateSubtitle": "Create a wallet to start using Source Points", | ||||
|   "walletCreatePassword": "Set a payment password for your new wallet below", | ||||
|   "walletCurrencyShort": "SRC", | ||||
|   "walletCurrency": { | ||||
|     "one": "{} Source Point", | ||||
|     "other": "{} Source Points" | ||||
|   }, | ||||
|   "aiThinkingProcess": "AI Thinking Process", | ||||
|   "accountSettingsApplied": "Account settings have been applied.", | ||||
|   "trayMenuExit": "Exit" | ||||
| } | ||||
|   | ||||
| @@ -15,6 +15,10 @@ | ||||
|   "screenAccountProfileEdit": "编辑资料", | ||||
|   "screenAbuseReport": "滥用检举", | ||||
|   "screenSettings": "设置", | ||||
|   "screenAccountSettings": "账号设置", | ||||
|   "screenFactorSettings": "验证因子", | ||||
|   "screenAccountWallet": "钱包", | ||||
|   "screenNews": "新闻", | ||||
|   "screenAlbum": "相册", | ||||
|   "screenChat": "聊天", | ||||
|   "screenChatManage": "编辑聊天频道", | ||||
| @@ -87,8 +91,18 @@ | ||||
|   }, | ||||
|   "loginEnterPassword": "验证代码", | ||||
|   "loginSuccess": "登录为 {}", | ||||
|   "authFactorDelete": "删除验证因子", | ||||
|   "authFactorDeleteDescription": "你确定要删除 {} 验证因子吗?", | ||||
|   "authFactorPassword": "密码", | ||||
|   "authFactorPasswordDescription": "注册时选择设置的密码。", | ||||
|   "authFactorEmail": "电邮一次性验证码", | ||||
|   "authFactorEmailDescription": "由我们生成并发送到绑定的的电子邮箱的一次性验证码。", | ||||
|   "authFactorTOTP": "时序验证码", | ||||
|   "authFactorTOTPDescription": "使用 Google Authenticator 或 Authy 等验证器生成的一次性验证码。", | ||||
|   "authFactorInAppNotify": "应用内通知验证码", | ||||
|   "authFactorInAppNotifyDescription": "通过站内通知推送的一次性验证码。", | ||||
|   "authFactorAdd": "添加新验证因子", | ||||
|   "authFactorAddSubtitle": "给你的帐户登陆时提供另一个方案。", | ||||
|   "accountIntroTitle": "喜欢您来!", | ||||
|   "accountIntroSubtitle": "登陆以探索更广大的世界。", | ||||
|   "accountLogout": "退出登录", | ||||
| @@ -97,8 +111,14 @@ | ||||
|   "accountLogoutConfirm": "您需要重新输入账号密码,甚至可能需要多步验证来再次登陆。", | ||||
|   "accountPublishers": "你的发布者", | ||||
|   "accountPublishersSubtitle": "管理你的公共形象。", | ||||
|   "accountSettings": "帐号设置", | ||||
|   "accountSettingsSubtitle": "管理你的帐号并让它更好的服务你。", | ||||
|   "accountProfileEdit": "编辑资料", | ||||
|   "accountProfileEditSubtitle": "使你的 Solarpass 账户更像你。", | ||||
|   "accountWallet": "钱包", | ||||
|   "accountWalletSubtitle": "查看你的余额和交易记录。", | ||||
|   "factorSettings": "验证因子", | ||||
|   "factorSettingsSubtitle": "管理你的登陆验证方式。", | ||||
|   "accountProfileEditApplied": "个人资料修改已被应用。", | ||||
|   "publishersNew": "新发布者", | ||||
|   "publisherNewSubtitle": "创建一个新的公共身份。", | ||||
| @@ -123,6 +143,9 @@ | ||||
|   "fieldPostTitle": "标题", | ||||
|   "fieldPostDescription": "描述", | ||||
|   "fieldPostTags": "标签", | ||||
|   "fieldPostCategories": "分类", | ||||
|   "fieldPostAlias": "别名", | ||||
|   "fieldPostAliasHint": "可选项,用于在 URL 中表示该帖子,应遵循 URL-Safe 的原则。", | ||||
|   "postPublish": "发布", | ||||
|   "postPublishedAt": "发布于", | ||||
|   "postPublishedUntil": "取消发布于", | ||||
| @@ -174,12 +197,30 @@ | ||||
|     "other": "{} 条评论" | ||||
|   }, | ||||
|   "settingsAppearance": "外观", | ||||
|   "settingsDisplayLanguage": "显示语言", | ||||
|   "settingsDisplayLanguageDescription": "设置应用程序使用的语言", | ||||
|   "settingsDisplayLanguageSystem": "跟随系统", | ||||
|   "settingsBackgroundImage": "背景图片", | ||||
|   "settingsBackgroundImageDescription": "设置应用全局生效的的背景图片。", | ||||
|   "settingsBackgroundImageClear": "清除现存背景图", | ||||
|   "settingsBackgroundImageClearDescription": "将应用背景图重置为空白。", | ||||
|   "settingsThemeMaterial3": "使用 Material You 设计范式", | ||||
|   "settingsThemeMaterial3Description": "将应用主题设置为 Material 3 设计范式的主题。", | ||||
|   "settingsAppBarTransparent": "透明顶栏", | ||||
|   "settingsAppBarTransparentDescription": "为顶栏启用透明效果。", | ||||
|   "settingsDrawerPreferCollapse": "侧边栏偏好折叠", | ||||
|   "settingsDrawerPreferCollapseDescription": "将侧边栏优先折叠,即使屏幕宽度足够大去放下整个侧边栏。", | ||||
|   "settingsColorScheme": "主题色", | ||||
|   "settingsColorSchemeDescription": "设置应用主题色。", | ||||
|   "settingsColorSeed": "预设色彩主题", | ||||
|   "settingsColorSeedDescription": "选择一个预设色彩主题。", | ||||
|   "settingsFeatures": "功能", | ||||
|   "settingsNotifyWithHaptic": "新通知时振动", | ||||
|   "settingsNotifyWithHapticDescription": "在应用在前台时收到新通知出现时出发轻量的振动。", | ||||
|   "settingsExpandPostLink": "展开帖子链接", | ||||
|   "settingsExpandPostLinkDescription": "在帖子列表中展开显示帖子中的链接。", | ||||
|   "settingsExpandChatLink": "展开聊天链接", | ||||
|   "settingsExpandChatLinkDescription": "在聊天信息中展开显示内容中的链接。", | ||||
|   "settingsNetwork": "网络", | ||||
|   "settingsNetworkServer": "HyperNet 服务器", | ||||
|   "settingsNetworkServerDescription": "设置 HyperNet 服务器地址,选择我们提供的,或者自己搭建。", | ||||
| @@ -198,12 +239,15 @@ | ||||
|   "settingsMisc": "杂项", | ||||
|   "settingsMiscAbout": "关于", | ||||
|   "settingsMiscAboutDescription": "查看 Solian 的版本信息。", | ||||
|   "settingsAccountLanguage": "帐号偏好语言", | ||||
|   "settingsAccountLanguageDescription": "设置邮件、通知和其他帐号相关内容的语言。", | ||||
|   "sensitiveContent": "敏感内容", | ||||
|   "sensitiveContentCollapsed": "敏感内容已折叠。", | ||||
|   "sensitiveContentDescription": "此内容已被标记,可能不适合所有人查看。", | ||||
|   "sensitiveContentReveal": "显示内容", | ||||
|   "serverConnecting": "正在连接服务器…", | ||||
|   "serverDisconnected": "已与服务器断开连接", | ||||
|   "serverConnecting": "正在连接…", | ||||
|   "serverDisconnected": "已断开连接", | ||||
|   "serverConnected": "已连接", | ||||
|   "fieldChatAlias": "频道别名", | ||||
|   "fieldChatAliasHint": "全站范围内唯一的频道别名,用于在 URL 中表示该频道,留空则自动生成。应遵循 URL-Safe 的原则。", | ||||
|   "fieldChatName": "名称", | ||||
| @@ -270,16 +314,50 @@ | ||||
|     "one": "{} 个附件", | ||||
|     "other": "{} 个附件" | ||||
|   }, | ||||
|   "messageTyping": { | ||||
|     "one": "{} 正在输入", | ||||
|     "other": "{} 正在输入" | ||||
|   }, | ||||
|   "fieldAttachmentRandomId": "访问 ID", | ||||
|   "fieldAttachmentAlt": "概述文字", | ||||
|   "addAttachmentFromAlbum": "从相册中添加附件", | ||||
|   "addAttachmentFromClipboard": "粘贴附件", | ||||
|   "addAttachmentFromCameraPhoto": "拍摄照片", | ||||
|   "addAttachmentFromCameraVideo": "拍摄视频", | ||||
|   "addAttachmentFromRandomId": "通过访问 ID 链接", | ||||
|   "attachmentDetailInfo": "附件详细信息", | ||||
|   "attachmentPastedImage": "粘贴的图片", | ||||
|   "attachmentInsertLink": "插入连接", | ||||
|   "attachmentSetAsPostThumbnail": "设置为帖子缩略图", | ||||
|   "attachmentUnsetAsPostThumbnail": "取消设置为帖子缩略图", | ||||
|   "attachmentCompressVideo": "重新编码视频", | ||||
|   "attachmentSetThumbnail": "设置缩略图", | ||||
|   "attachmentSetAlt": "设置概述文字", | ||||
|   "attachmentCopyRandomId": "复制访问 ID", | ||||
|   "attachmentUpload": "上传", | ||||
|   "attachmentInputDialog": "上传附件", | ||||
|   "attachmentInputUseRandomId": "使用访问 ID", | ||||
|   "attachmentInputNew": "新上传附件", | ||||
|   "waitingForUpload": "等待上传", | ||||
|   "attachmentVideoCompressHint": "压缩一份视频的副本", | ||||
|   "attachmentVideoCompressHintDescription": "你想上传压缩视频 {} 的副本吗?它将帮助你的观众快速预览视频,并且他们仍然可以观看原始视频。这将会在在你的设备上处理视频,所以需要一些时间,所以请耐心等待。", | ||||
|   "attachmentCompressQuality": "压缩质量", | ||||
|   "attachmentCompressQualityHighest": "最高", | ||||
|   "attachmentCompressQualityDefault": "默认", | ||||
|   "attachmentCompressQualityMedium": "中等", | ||||
|   "attachmentCompressQualityLow": "低", | ||||
|   "attachmentCompressQualityHint": "Solar Network 并没有阻止你上传大文件、高分辨率、高码率的视频,但是为了你的网络情况观众考虑,我们建议你选择一个合适的压缩质量。", | ||||
|   "attachmentUploaded": "已上传", | ||||
|   "attachmentPending": "未上传", | ||||
|   "attachmentCopyCompressed": "有压缩副本", | ||||
|   "attachmentGotBoosted": "有加速传递", | ||||
|   "attachmentBoost": "加速包", | ||||
|   "attachmentCreateBoost": "加速传递", | ||||
|   "attachmentBoostHint": "加速传递允许您将附件上传到更近的受众或更快的内容网络。该功能目前处于 Beta 阶段。该功能限时免费,当有价格计划更改时,您将会被通知。", | ||||
|   "attachmentDestinationRegion": "目标节点", | ||||
|   "attachmentDestinationRegionAPAC": "亚太地区", | ||||
|   "attachmentDestinationRegionNGB": "中国 · 浙江 · 宁波", | ||||
|   "attachmentDestinationRegionHKG": "香港", | ||||
|   "notification": "通知", | ||||
|   "notificationUnreadCount": { | ||||
|     "zero": "无未读通知", | ||||
| @@ -367,9 +445,32 @@ | ||||
|   "dailyCheckNegativeHint5Description": "关键时刻断网", | ||||
|   "dailyCheckNegativeHint6": "出门", | ||||
|   "dailyCheckNegativeHint6Description": "忘带伞遇上大雨", | ||||
|   "happyBirthday": "生日快乐,{}!", | ||||
|   "celebrateBirthday": "生日快乐,{}!", | ||||
|   "celebrateLunarNewYear": "春节快乐,{}!", | ||||
|   "celebrateMidAutumn": "中秋节快乐,{}!", | ||||
|   "celebrateDragonBoat": "端午节快乐,{}!", | ||||
|   "celebrateMerryXmas": "圣诞快乐,{}!", | ||||
|   "celebrateNewYear": "新年快乐,{}!", | ||||
|   "celebrateValentineDay": "今天是情人节,{}!", | ||||
|   "celebrateLaborDay": "今天是劳动节,{}。", | ||||
|   "celebrateMotherDay": "今天是母亲节,{}。", | ||||
|   "celebrateChildrenDay": "今天是儿童节,{}!", | ||||
|   "celebrateFatherDay": "今天是父亲节,{}。", | ||||
|   "celebrateHalloween": "快乐在圣诞节,{}!", | ||||
|   "celebrateThanksgiving": "今天是感恩节,{}!", | ||||
|   "pendingLunarNewYear": "{} 过春节", | ||||
|   "pendingMidAutumn": "{} 过中秋节", | ||||
|   "pendingDragonBoat": "{} 过端午节", | ||||
|   "pendingBirthday": "{} 过生日", | ||||
|   "pendingMerryXmas": "{} 过圣诞节", | ||||
|   "pendingNewYear": "{} 跨年", | ||||
|   "pendingValentineDay": "{} 过情人节", | ||||
|   "pendingLaborDay": "{} 过劳动节", | ||||
|   "pendingMotherDay": "{} 过母亲节", | ||||
|   "pendingChildrenDay": "{} 过儿童节", | ||||
|   "pendingFatherDay": "{} 过父亲节", | ||||
|   "pendingHalloween": "{} 过圣诞节", | ||||
|   "pendingThanksgiving": "{} 过感恩节", | ||||
|   "friendNew": "添加好友", | ||||
|   "friendRequests": "好友请求", | ||||
|   "friendRequestsDescription": { | ||||
| @@ -403,14 +504,16 @@ | ||||
|   "accountJoinedAt": "加入于 {}", | ||||
|   "accountBirthday": "出生于 {}", | ||||
|   "accountBadge": "徽章", | ||||
|   "accountCheckInNoRecords": "暂无运势记录", | ||||
|   "badgeCompanyStaff": "索尔辛茨士大夫 · 员工", | ||||
|   "badgeSiteMigration": "Solar Network 原住民", | ||||
|   "accountStatus": "状态", | ||||
|   "accountStatusOnline": "在线", | ||||
|   "accountStatusOffline": "离线", | ||||
|   "accountStatusLastSeen": "最后一次在 {} 上线", | ||||
|   "accountStatusLastSeen": "最后一次上线于 {}", | ||||
|   "postArticle": "Solar Network 上的文章", | ||||
|   "postStory": "Solar Network 上的故事", | ||||
|   "postLocalDraftRestored": "从本地恢复草稿", | ||||
|   "articleWrittenAt": "发表于 {}", | ||||
|   "articleEditedAt": "编辑于 {}", | ||||
|   "attachmentSaved": "已保存到相册", | ||||
| @@ -451,11 +554,59 @@ | ||||
|   "postImageShareAds": "来 Solar Network 探索更多有趣帖子", | ||||
|   "postShare": "分享", | ||||
|   "postShareImage": "分享帖图", | ||||
|   "postGetInsight": "获取见解", | ||||
|   "postGetInsightTitle": "AI 见解", | ||||
|   "postGetInsightDescription": "AI 可能会出错,检查信息真实性。", | ||||
|   "appInitializing": "正在初始化", | ||||
|   "poweredBy": "由 {} 提供支持", | ||||
|   "shareIntent": "分享", | ||||
|   "shareIntentDescription": "您想对您分享的内容做些什么?", | ||||
|   "shareIntentPostStory": "发布动态", | ||||
|   "shareIntentSendChannel": "分享到聊天频道", | ||||
|   "updateAvailable": "检测到更新可用", | ||||
|   "updateOngoing": "正在更新,请稍后……" | ||||
|   "updateOngoing": "正在更新,请稍后……", | ||||
|   "custom": "自定义", | ||||
|   "colorSchemeIndigo": "靛蓝", | ||||
|   "colorSchemeBlue": "蓝色", | ||||
|   "colorSchemeGreen": "绿色", | ||||
|   "colorSchemeYellow": "黄色", | ||||
|   "colorSchemeOrange": "橙色", | ||||
|   "colorSchemeRed": "红色", | ||||
|   "colorSchemeWhite": "白色", | ||||
|   "colorSchemeBlack": "黑色", | ||||
|   "colorSchemeApplied": "主题色已应用,可能需要重启来生效。", | ||||
|   "postFeaturedComment": "精选评论", | ||||
|   "postCategoryTechnology": "技术", | ||||
|   "postCategoryGaming": "游戏", | ||||
|   "postCategoryLife": "生活", | ||||
|   "postCategoryArts": "艺术", | ||||
|   "postCategorySports": "体育", | ||||
|   "postCategoryMusic": "音乐", | ||||
|   "postCategoryNews": "新闻", | ||||
|   "postCategoryKnowledge": "知识", | ||||
|   "postCategoryLiterature": "文学", | ||||
|   "postCategoryFunny": "搞笑", | ||||
|   "postCategoryUncategorized": "未分类", | ||||
|   "newsAllSources": "所有新闻", | ||||
|   "newsReadingProviderSwap": "切换", | ||||
|   "newsReadingFromReader": "你正在从 HyperNet.Reader 阅读文章", | ||||
|   "newsReadingFromOriginal": "你正在阅读原始文章", | ||||
|   "newsDisclaimer": "本文由 HyperNet.Reader 从互联网上获取,我们不担保其内容的真实性,请自行判断。本文章的所有内容版权归原作者所有。", | ||||
|   "newsToday": "快讯", | ||||
|   "totpPostSetup": "还有一件事", | ||||
|   "totpPostSetupDescription": "使用 Google Authenticator, Microsoft Authenticator, 1Password, Authy, Bitwarden 或其他支持 TOTP 的验证器扫描本 QR Code 来添加。", | ||||
|   "totpNeverShare": "永远不要分享这个 QR Code", | ||||
|   "needHelp": "需要帮助?", | ||||
|   "needHelpLaunch": "查看我们的山羊维基!", | ||||
|   "walletCreate": "创建钱包", | ||||
|   "walletCreateSubtitle": "创建于一个钱包来开始使用源点。", | ||||
|   "walletCreatePassword": "在下方设置你的付款密码", | ||||
|   "walletCurrencyShort": "源点", | ||||
|   "walletCurrency": { | ||||
|     "one": "{} 源点", | ||||
|     "other": "{} 源点" | ||||
|   }, | ||||
|   "aiThinkingProcess": "AI 思考过程", | ||||
|   "accountSettingsApplied": "帐号设置已应用。", | ||||
|   "trayMenuExit": "退出" | ||||
| } | ||||
|   | ||||
| @@ -15,6 +15,10 @@ | ||||
|   "screenAccountProfileEdit": "編輯資料", | ||||
|   "screenAbuseReport": "濫用檢舉", | ||||
|   "screenSettings": "設置", | ||||
|   "screenAccountSettings": "賬號設置", | ||||
|   "screenFactorSettings": "驗證因子", | ||||
|   "screenAccountWallet": "錢包", | ||||
|   "screenNews": "新聞", | ||||
|   "screenAlbum": "相冊", | ||||
|   "screenChat": "聊天", | ||||
|   "screenChatManage": "編輯聊天頻道", | ||||
| @@ -87,8 +91,18 @@ | ||||
|   }, | ||||
|   "loginEnterPassword": "驗證代碼", | ||||
|   "loginSuccess": "登錄為 {}", | ||||
|   "authFactorDelete": "刪除驗證因子", | ||||
|   "authFactorDeleteDescription": "你確定要刪除 {} 驗證因子嗎?", | ||||
|   "authFactorPassword": "密碼", | ||||
|   "authFactorPasswordDescription": "註冊時選擇設置的密碼。", | ||||
|   "authFactorEmail": "電郵一次性驗證碼", | ||||
|   "authFactorEmailDescription": "由我們生成併發送到綁定的的電子郵箱的一次性驗證碼。", | ||||
|   "authFactorTOTP": "時序驗證碼", | ||||
|   "authFactorTOTPDescription": "使用 Google Authenticator 或 Authy 等驗證器生成的一次性驗證碼。", | ||||
|   "authFactorInAppNotify": "應用內通知驗證碼", | ||||
|   "authFactorInAppNotifyDescription": "通過站內通知推送的一次性驗證碼。", | ||||
|   "authFactorAdd": "添加新驗證因子", | ||||
|   "authFactorAddSubtitle": "給你的帳户登陸時提供另一個方案。", | ||||
|   "accountIntroTitle": "喜歡您來!", | ||||
|   "accountIntroSubtitle": "登陸以探索更廣大的世界。", | ||||
|   "accountLogout": "退出登錄", | ||||
| @@ -97,8 +111,14 @@ | ||||
|   "accountLogoutConfirm": "您需要重新輸入賬號密碼,甚至可能需要多步驗證來再次登陸。", | ||||
|   "accountPublishers": "你的發佈者", | ||||
|   "accountPublishersSubtitle": "管理你的公共形象。", | ||||
|   "accountSettings": "帳號設置", | ||||
|   "accountSettingsSubtitle": "管理你的帳號並讓它更好的服務你。", | ||||
|   "accountProfileEdit": "編輯資料", | ||||
|   "accountProfileEditSubtitle": "使你的 Solarpass 賬户更像你。", | ||||
|   "accountWallet": "錢包", | ||||
|   "accountWalletSubtitle": "查看你的餘額和交易記錄。", | ||||
|   "factorSettings": "驗證因子", | ||||
|   "factorSettingsSubtitle": "管理你的登陸驗證方式。", | ||||
|   "accountProfileEditApplied": "個人資料修改已被應用。", | ||||
|   "publishersNew": "新發布者", | ||||
|   "publisherNewSubtitle": "創建一個新的公共身份。", | ||||
| @@ -123,6 +143,9 @@ | ||||
|   "fieldPostTitle": "標題", | ||||
|   "fieldPostDescription": "描述", | ||||
|   "fieldPostTags": "標籤", | ||||
|   "fieldPostCategories": "分類", | ||||
|   "fieldPostAlias": "別名", | ||||
|   "fieldPostAliasHint": "可選項,用於在 URL 中表示該帖子,應遵循 URL-Safe 的原則。", | ||||
|   "postPublish": "發佈", | ||||
|   "postPublishedAt": "發佈於", | ||||
|   "postPublishedUntil": "取消發佈於", | ||||
| @@ -174,12 +197,30 @@ | ||||
|     "other": "{} 條評論" | ||||
|   }, | ||||
|   "settingsAppearance": "外觀", | ||||
|   "settingsDisplayLanguage": "顯示語言", | ||||
|   "settingsDisplayLanguageDescription": "設置應用程序使用的語言", | ||||
|   "settingsDisplayLanguageSystem": "跟隨系統", | ||||
|   "settingsBackgroundImage": "背景圖片", | ||||
|   "settingsBackgroundImageDescription": "設置應用全局生效的的背景圖片。", | ||||
|   "settingsBackgroundImageClear": "清除現存背景圖", | ||||
|   "settingsBackgroundImageClearDescription": "將應用背景圖重置為空白。", | ||||
|   "settingsThemeMaterial3": "使用 Material You 設計範式", | ||||
|   "settingsThemeMaterial3Description": "將應用主題設置為 Material 3 設計範式的主題。", | ||||
|   "settingsAppBarTransparent": "透明頂欄", | ||||
|   "settingsAppBarTransparentDescription": "為頂欄啓用透明效果。", | ||||
|   "settingsDrawerPreferCollapse": "側邊欄偏好摺疊", | ||||
|   "settingsDrawerPreferCollapseDescription": "將側邊欄優先摺疊,即使屏幕寬度足夠大去放下整個側邊欄。", | ||||
|   "settingsColorScheme": "主題色", | ||||
|   "settingsColorSchemeDescription": "設置應用主題色。", | ||||
|   "settingsColorSeed": "預設色彩主題", | ||||
|   "settingsColorSeedDescription": "選擇一個預設色彩主題。", | ||||
|   "settingsFeatures": "功能", | ||||
|   "settingsNotifyWithHaptic": "新通知時振動", | ||||
|   "settingsNotifyWithHapticDescription": "在應用在前台時收到新通知出現時出發輕量的振動。", | ||||
|   "settingsExpandPostLink": "展開帖子鏈接", | ||||
|   "settingsExpandPostLinkDescription": "在帖子列表中展開顯示帖子中的鏈接。", | ||||
|   "settingsExpandChatLink": "展開聊天鏈接", | ||||
|   "settingsExpandChatLinkDescription": "在聊天信息中展開顯示內容中的鏈接。", | ||||
|   "settingsNetwork": "網絡", | ||||
|   "settingsNetworkServer": "HyperNet 服務器", | ||||
|   "settingsNetworkServerDescription": "設置 HyperNet 服務器地址,選擇我們提供的,或者自己搭建。", | ||||
| @@ -198,12 +239,15 @@ | ||||
|   "settingsMisc": "雜項", | ||||
|   "settingsMiscAbout": "關於", | ||||
|   "settingsMiscAboutDescription": "查看 Solian 的版本信息。", | ||||
|   "settingsAccountLanguage": "帳號偏好語言", | ||||
|   "settingsAccountLanguageDescription": "設置郵件、通知和其他帳號相關內容的語言。", | ||||
|   "sensitiveContent": "敏感內容", | ||||
|   "sensitiveContentCollapsed": "敏感內容已摺疊。", | ||||
|   "sensitiveContentDescription": "此內容已被標記,可能不適合所有人查看。", | ||||
|   "sensitiveContentReveal": "顯示內容", | ||||
|   "serverConnecting": "正在連接服務器…", | ||||
|   "serverDisconnected": "已與服務器斷開連接", | ||||
|   "serverConnecting": "正在連接…", | ||||
|   "serverDisconnected": "已斷開連接", | ||||
|   "serverConnected": "已連接", | ||||
|   "fieldChatAlias": "頻道別名", | ||||
|   "fieldChatAliasHint": "全站範圍內唯一的頻道別名,用於在 URL 中表示該頻道,留空則自動生成。應遵循 URL-Safe 的原則。", | ||||
|   "fieldChatName": "名稱", | ||||
| @@ -270,16 +314,50 @@ | ||||
|     "one": "{} 個附件", | ||||
|     "other": "{} 個附件" | ||||
|   }, | ||||
|   "messageTyping": { | ||||
|     "one": "{} 正在輸入", | ||||
|     "other": "{} 正在輸入" | ||||
|   }, | ||||
|   "fieldAttachmentRandomId": "訪問 ID", | ||||
|   "fieldAttachmentAlt": "概述文字", | ||||
|   "addAttachmentFromAlbum": "從相冊中添加附件", | ||||
|   "addAttachmentFromClipboard": "粘貼附件", | ||||
|   "addAttachmentFromCameraPhoto": "拍攝照片", | ||||
|   "addAttachmentFromCameraVideo": "拍攝視頻", | ||||
|   "addAttachmentFromRandomId": "通過訪問 ID 鏈接", | ||||
|   "attachmentDetailInfo": "附件詳細信息", | ||||
|   "attachmentPastedImage": "粘貼的圖片", | ||||
|   "attachmentInsertLink": "插入連接", | ||||
|   "attachmentSetAsPostThumbnail": "設置為帖子縮略圖", | ||||
|   "attachmentUnsetAsPostThumbnail": "取消設置為帖子縮略圖", | ||||
|   "attachmentCompressVideo": "重新編碼視頻", | ||||
|   "attachmentSetThumbnail": "設置縮略圖", | ||||
|   "attachmentSetAlt": "設置概述文字", | ||||
|   "attachmentCopyRandomId": "複製訪問 ID", | ||||
|   "attachmentUpload": "上傳", | ||||
|   "attachmentInputDialog": "上傳附件", | ||||
|   "attachmentInputUseRandomId": "使用訪問 ID", | ||||
|   "attachmentInputNew": "新上傳附件", | ||||
|   "waitingForUpload": "等待上傳", | ||||
|   "attachmentVideoCompressHint": "壓縮一份視頻的副本", | ||||
|   "attachmentVideoCompressHintDescription": "你想上傳壓縮視頻 {} 的副本嗎?它將幫助你的觀眾快速預覽視頻,並且他們仍然可以觀看原始視頻。這將會在在你的設備上處理視頻,所以需要一些時間,所以請耐心等待。", | ||||
|   "attachmentCompressQuality": "壓縮質量", | ||||
|   "attachmentCompressQualityHighest": "最高", | ||||
|   "attachmentCompressQualityDefault": "默認", | ||||
|   "attachmentCompressQualityMedium": "中等", | ||||
|   "attachmentCompressQualityLow": "低", | ||||
|   "attachmentCompressQualityHint": "Solar Network 並沒有阻止你上傳大文件、高分辨率、高碼率的視頻,但是為了你的網絡情況觀眾考慮,我們建議你選擇一個合適的壓縮質量。", | ||||
|   "attachmentUploaded": "已上傳", | ||||
|   "attachmentPending": "未上傳", | ||||
|   "attachmentCopyCompressed": "有壓縮副本", | ||||
|   "attachmentGotBoosted": "有加速傳遞", | ||||
|   "attachmentBoost": "加速包", | ||||
|   "attachmentCreateBoost": "加速傳遞", | ||||
|   "attachmentBoostHint": "加速傳遞允許您將附件上傳到更近的受眾或更快的內容網絡。該功能目前處於 Beta 階段。該功能限時免費,當有價格計劃更改時,您將會被通知。", | ||||
|   "attachmentDestinationRegion": "目標節點", | ||||
|   "attachmentDestinationRegionAPAC": "亞太地區", | ||||
|   "attachmentDestinationRegionNGB": "中國 · 浙江 · 寧波", | ||||
|   "attachmentDestinationRegionHKG": "香港", | ||||
|   "notification": "通知", | ||||
|   "notificationUnreadCount": { | ||||
|     "zero": "無未讀通知", | ||||
| @@ -367,7 +445,32 @@ | ||||
|   "dailyCheckNegativeHint5Description": "關鍵時刻斷網", | ||||
|   "dailyCheckNegativeHint6": "出門", | ||||
|   "dailyCheckNegativeHint6Description": "忘帶傘遇上大雨", | ||||
|   "happyBirthday": "生日快樂,{}!", | ||||
|   "celebrateBirthday": "生日快樂,{}!", | ||||
|   "celebrateLunarNewYear": "春節快樂,{}!", | ||||
|   "celebrateMidAutumn": "中秋節快樂,{}!", | ||||
|   "celebrateDragonBoat": "端午節快樂,{}!", | ||||
|   "celebrateMerryXmas": "聖誕快樂,{}!", | ||||
|   "celebrateNewYear": "新年快樂,{}!", | ||||
|   "celebrateValentineDay": "今天是情人節,{}!", | ||||
|   "celebrateLaborDay": "今天是勞動節,{}。", | ||||
|   "celebrateMotherDay": "今天是母親節,{}。", | ||||
|   "celebrateChildrenDay": "今天是兒童節,{}!", | ||||
|   "celebrateFatherDay": "今天是父親節,{}。", | ||||
|   "celebrateHalloween": "快樂在聖誕節,{}!", | ||||
|   "celebrateThanksgiving": "今天是感恩節,{}!", | ||||
|   "pendingLunarNewYear": "{} 過春節", | ||||
|   "pendingMidAutumn": "{} 過中秋節", | ||||
|   "pendingDragonBoat": "{} 過端午節", | ||||
|   "pendingBirthday": "{} 過生日", | ||||
|   "pendingMerryXmas": "{} 過聖誕節", | ||||
|   "pendingNewYear": "{} 跨年", | ||||
|   "pendingValentineDay": "{} 過情人節", | ||||
|   "pendingLaborDay": "{} 過勞動節", | ||||
|   "pendingMotherDay": "{} 過母親節", | ||||
|   "pendingChildrenDay": "{} 過兒童節", | ||||
|   "pendingFatherDay": "{} 過父親節", | ||||
|   "pendingHalloween": "{} 過聖誕節", | ||||
|   "pendingThanksgiving": "{} 過感恩節", | ||||
|   "friendNew": "添加好友", | ||||
|   "friendRequests": "好友請求", | ||||
|   "friendRequestsDescription": { | ||||
| @@ -401,14 +504,16 @@ | ||||
|   "accountJoinedAt": "加入於 {}", | ||||
|   "accountBirthday": "出生於 {}", | ||||
|   "accountBadge": "徽章", | ||||
|   "accountCheckInNoRecords": "暫無運勢記錄", | ||||
|   "badgeCompanyStaff": "索爾辛茨士大夫 · 員工", | ||||
|   "badgeSiteMigration": "Solar Network 原住民", | ||||
|   "accountStatus": "狀態", | ||||
|   "accountStatusOnline": "在線", | ||||
|   "accountStatusOffline": "離線", | ||||
|   "accountStatusLastSeen": "最後一次在 {} 上線", | ||||
|   "accountStatusLastSeen": "最後一次上線於 {}", | ||||
|   "postArticle": "Solar Network 上的文章", | ||||
|   "postStory": "Solar Network 上的故事", | ||||
|   "postLocalDraftRestored": "從本地恢復草稿", | ||||
|   "articleWrittenAt": "發表於 {}", | ||||
|   "articleEditedAt": "編輯於 {}", | ||||
|   "attachmentSaved": "已保存到相冊", | ||||
| @@ -449,9 +554,58 @@ | ||||
|   "postImageShareAds": "來 Solar Network 探索更多有趣帖子", | ||||
|   "postShare": "分享", | ||||
|   "postShareImage": "分享帖圖", | ||||
|   "postGetInsight": "獲取見解", | ||||
|   "postGetInsightTitle": "AI 見解", | ||||
|   "postGetInsightDescription": "AI 可能會出錯,檢查信息真實性。", | ||||
|   "appInitializing": "正在初始化", | ||||
|   "poweredBy": "由 {} 提供支持", | ||||
|   "shareIntent": "分享", | ||||
|   "shareIntentDescription": "您想對您分享的內容做些什麼?", | ||||
|   "shareIntentPostStory": "發佈動態" | ||||
|   "shareIntentPostStory": "發佈動態", | ||||
|   "shareIntentSendChannel": "分享到聊天頻道", | ||||
|   "updateAvailable": "檢測到更新可用", | ||||
|   "updateOngoing": "正在更新,請稍後……", | ||||
|   "custom": "自定義", | ||||
|   "colorSchemeIndigo": "靛藍", | ||||
|   "colorSchemeBlue": "藍色", | ||||
|   "colorSchemeGreen": "綠色", | ||||
|   "colorSchemeYellow": "黃色", | ||||
|   "colorSchemeOrange": "橙色", | ||||
|   "colorSchemeRed": "紅色", | ||||
|   "colorSchemeWhite": "白色", | ||||
|   "colorSchemeBlack": "黑色", | ||||
|   "colorSchemeApplied": "主題色已應用,可能需要重啓來生效。", | ||||
|   "postFeaturedComment": "精選評論", | ||||
|   "postCategoryTechnology": "技術", | ||||
|   "postCategoryGaming": "遊戲", | ||||
|   "postCategoryLife": "生活", | ||||
|   "postCategoryArts": "藝術", | ||||
|   "postCategorySports": "體育", | ||||
|   "postCategoryMusic": "音樂", | ||||
|   "postCategoryNews": "新聞", | ||||
|   "postCategoryKnowledge": "知識", | ||||
|   "postCategoryLiterature": "文學", | ||||
|   "postCategoryFunny": "搞笑", | ||||
|   "postCategoryUncategorized": "未分類", | ||||
|   "newsAllSources": "所有新聞", | ||||
|   "newsReadingProviderSwap": "切換", | ||||
|   "newsReadingFromReader": "你正在從 HyperNet.Reader 閲讀文章", | ||||
|   "newsReadingFromOriginal": "你正在閲讀原始文章", | ||||
|   "newsDisclaimer": "本文由 HyperNet.Reader 從互聯網上獲取,我們不擔保其內容的真實性,請自行判斷。本文章的所有內容版權歸原作者所有。", | ||||
|   "newsToday": "快訊", | ||||
|   "totpPostSetup": "還有一件事", | ||||
|   "totpPostSetupDescription": "使用 Google Authenticator, Microsoft Authenticator, 1Password, Authy, Bitwarden 或其他支持 TOTP 的驗證器掃描本 QR Code 來添加。", | ||||
|   "totpNeverShare": "永遠不要分享這個 QR Code", | ||||
|   "needHelp": "需要幫助?", | ||||
|   "needHelpLaunch": "查看我們的山羊維基!", | ||||
|   "walletCreate": "創建錢包", | ||||
|   "walletCreateSubtitle": "創建於一個錢包來開始使用源點。", | ||||
|   "walletCreatePassword": "在下方設置你的付款密碼", | ||||
|   "walletCurrencyShort": "源點", | ||||
|   "walletCurrency": { | ||||
|     "one": "{} 源點", | ||||
|     "other": "{} 源點" | ||||
|   }, | ||||
|   "aiThinkingProcess": "AI 思考過程", | ||||
|   "accountSettingsApplied": "帳號設置已應用。" | ||||
| } | ||||
|   | ||||
| @@ -7,15 +7,19 @@ | ||||
|   "screenAuthLogin": "登陸", | ||||
|   "screenAuthLoginSubtitle": "使用 Solarpass 登陸 Solar Network", | ||||
|   "screenAuthLoginGreeting": "歡迎回來", | ||||
|   "screenAuthRegister": "建立賬號", | ||||
|   "screenAuthRegisterSubtitle": "建立一個 Solarpass 賬號", | ||||
|   "screenAccountPublishers": "釋出者", | ||||
|   "screenAccountPublisherNew": "新建釋出者", | ||||
|   "screenAccountPublisherEdit": "編輯釋出者", | ||||
|   "screenAuthRegister": "創建賬號", | ||||
|   "screenAuthRegisterSubtitle": "創建一個 Solarpass 賬號", | ||||
|   "screenAccountPublishers": "發佈者", | ||||
|   "screenAccountPublisherNew": "新建發佈者", | ||||
|   "screenAccountPublisherEdit": "編輯發佈者", | ||||
|   "screenAccountProfileEdit": "編輯資料", | ||||
|   "screenAbuseReport": "濫用檢舉", | ||||
|   "screenSettings": "設定", | ||||
|   "screenAlbum": "相簿", | ||||
|   "screenSettings": "設置", | ||||
|   "screenAccountSettings": "賬號設置", | ||||
|   "screenFactorSettings": "驗證因子", | ||||
|   "screenAccountWallet": "錢包", | ||||
|   "screenNews": "新聞", | ||||
|   "screenAlbum": "相冊", | ||||
|   "screenChat": "聊天", | ||||
|   "screenChatManage": "編輯聊天頻道", | ||||
|   "screenChatNew": "新建聊天頻道", | ||||
| @@ -23,37 +27,37 @@ | ||||
|   "screenRealmManage": "編輯領域", | ||||
|   "screenRealmNew": "新建領域", | ||||
|   "screenNotification": "通知", | ||||
|   "screenPostSearch": "搜尋帖子", | ||||
|   "screenPostSearch": "搜索帖子", | ||||
|   "screenFriend": "好友", | ||||
|   "dialogOkay": "好的", | ||||
|   "dialogCancel": "取消", | ||||
|   "dialogConfirm": "確認", | ||||
|   "dialogDismiss": "忽略", | ||||
|   "dialogError": "出了點問題", | ||||
|   "errorRequestBad": "伺服器拒絕了您的請求,請檢查您的輸入。", | ||||
|   "errorRequestUnauthorized": "未授權的請求,請登入或者嘗試重新登陸。", | ||||
|   "errorRequestForbidden": "被禁止的請求,您沒有足夠的許可權去做那件事。", | ||||
|   "errorRequestNotFound": "您正查詢的資源無法被找到。", | ||||
|   "errorRequestConnection": "網路連線錯誤,請檢查您的網路狀態或者檢查我們的服務狀態。", | ||||
|   "errorRequestUnknown": "未知請求錯誤,您可能想將此對話方塊截圖併發送給我們。", | ||||
|   "errorRequestBad": "服務器拒絕了您的請求,請檢查您的輸入。", | ||||
|   "errorRequestUnauthorized": "未授權的請求,請登錄或者嘗試重新登陸。", | ||||
|   "errorRequestForbidden": "被禁止的請求,您沒有足夠的權限去做那件事。", | ||||
|   "errorRequestNotFound": "您正查找的資源無法被找到。", | ||||
|   "errorRequestConnection": "網絡連接錯誤,請檢查您的網絡狀態或者檢查我們的服務狀態。", | ||||
|   "errorRequestUnknown": "未知請求錯誤,您可能想將此對話框截圖併發送給我們。", | ||||
|   "unknown": "未知", | ||||
|   "loading": "載入中…", | ||||
|   "loading": "加載中…", | ||||
|   "prev": "上一步", | ||||
|   "next": "下一步", | ||||
|   "edit": "編輯", | ||||
|   "apply": "應用", | ||||
|   "cancel": "取消", | ||||
|   "create": "建立", | ||||
|   "create": "創建", | ||||
|   "preview": "預覽", | ||||
|   "delete": "刪除", | ||||
|   "unlink": "解除連結", | ||||
|   "unlink": "解除鏈接", | ||||
|   "crop": "裁剪", | ||||
|   "compress": "壓縮", | ||||
|   "report": "檢舉", | ||||
|   "repost": "轉帖", | ||||
|   "replyPost": "回貼", | ||||
|   "reply": "回覆", | ||||
|   "unset": "未設定", | ||||
|   "unset": "未設置", | ||||
|   "untitled": "無題", | ||||
|   "postDetail": "帖子詳情", | ||||
|   "postNoun": "帖子", | ||||
| @@ -64,20 +68,20 @@ | ||||
|     "one": "總計 {} 字", | ||||
|     "other": "總計 {} 字" | ||||
|   }, | ||||
|   "fieldUsername": "使用者名稱", | ||||
|   "fieldUsername": "用戶名", | ||||
|   "fieldNickname": "顯示名", | ||||
|   "fieldEmail": "電子郵箱地址", | ||||
|   "fieldPassword": "密碼", | ||||
|   "fieldUsernameAlphanumOnly": "使用者名稱只能包含英文大小寫字母和數字。", | ||||
|   "fieldUsernameLengthLimit": "使用者名稱必須在 {} 和 {} 之間。", | ||||
|   "fieldUsernameCannotEditHint": "使用者名稱在建立後無法修改", | ||||
|   "fieldUsernameLookupHint": "支援使用者名稱、電話號碼或郵箱地址", | ||||
|   "fieldUsernameAlphanumOnly": "用戶名只能包含英文大小寫字母和數字。", | ||||
|   "fieldUsernameLengthLimit": "用戶名必須在 {} 和 {} 之間。", | ||||
|   "fieldUsernameCannotEditHint": "用戶名在創建後無法修改", | ||||
|   "fieldUsernameLookupHint": "支持用戶名、電話號碼或郵箱地址", | ||||
|   "fieldNicknameLengthLimit": "暱稱必須在 {} 和 {} 之間。", | ||||
|   "fieldEmailAddressMustBeValid": "電子郵箱地址必須是一個電子郵箱地址。", | ||||
|   "fieldFirstName": "名", | ||||
|   "fieldLastName": "姓", | ||||
|   "fieldBirthday": "生日", | ||||
|   "fieldImageHint": "你可以點選這些個人頭像來編輯它們。", | ||||
|   "fieldImageHint": "你可以點擊這些個人頭像來編輯它們。", | ||||
|   "fieldDescription": "簡介", | ||||
|   "forgotPassword": "忘記密碼", | ||||
|   "loginPickFactor": "選擇方式驗證", | ||||
| @@ -85,24 +89,40 @@ | ||||
|     "one": "{} 步驗證", | ||||
|     "other": "{} 步驗證" | ||||
|   }, | ||||
|   "loginEnterPassword": "驗證程式碼", | ||||
|   "loginSuccess": "登入為 {}", | ||||
|   "loginEnterPassword": "驗證代碼", | ||||
|   "loginSuccess": "登錄為 {}", | ||||
|   "authFactorDelete": "刪除驗證因子", | ||||
|   "authFactorDeleteDescription": "你確定要刪除 {} 驗證因子嗎?", | ||||
|   "authFactorPassword": "密碼", | ||||
|   "authFactorPasswordDescription": "註冊時選擇設置的密碼。", | ||||
|   "authFactorEmail": "電郵一次性驗證碼", | ||||
|   "authFactorEmailDescription": "由我們生成併發送到綁定的的電子郵箱的一次性驗證碼。", | ||||
|   "authFactorTOTP": "時序驗證碼", | ||||
|   "authFactorTOTPDescription": "使用 Google Authenticator 或 Authy 等驗證器生成的一次性驗證碼。", | ||||
|   "authFactorInAppNotify": "應用內通知驗證碼", | ||||
|   "authFactorInAppNotifyDescription": "通過站內通知推送的一次性驗證碼。", | ||||
|   "authFactorAdd": "添加新驗證因子", | ||||
|   "authFactorAddSubtitle": "給你的帳戶登陸時提供另一個方案。", | ||||
|   "accountIntroTitle": "喜歡您來!", | ||||
|   "accountIntroSubtitle": "登陸以探索更廣大的世界。", | ||||
|   "accountLogout": "退出登入", | ||||
|   "accountLogoutSubtitle": "登出當前賬戶的登陸狀態。", | ||||
|   "accountLogoutConfirmTitle": "您確定要退出登入嗎?", | ||||
|   "accountLogout": "退出登錄", | ||||
|   "accountLogoutSubtitle": "註銷當前賬戶的登陸狀態。", | ||||
|   "accountLogoutConfirmTitle": "您確定要退出登錄嗎?", | ||||
|   "accountLogoutConfirm": "您需要重新輸入賬號密碼,甚至可能需要多步驗證來再次登陸。", | ||||
|   "accountPublishers": "你的釋出者", | ||||
|   "accountPublishers": "你的發佈者", | ||||
|   "accountPublishersSubtitle": "管理你的公共形象。", | ||||
|   "accountSettings": "帳號設置", | ||||
|   "accountSettingsSubtitle": "管理你的帳號並讓它更好的服務你。", | ||||
|   "accountProfileEdit": "編輯資料", | ||||
|   "accountProfileEditSubtitle": "使你的 Solarpass 賬戶更像你。", | ||||
|   "accountWallet": "錢包", | ||||
|   "accountWalletSubtitle": "查看你的餘額和交易記錄。", | ||||
|   "factorSettings": "驗證因子", | ||||
|   "factorSettingsSubtitle": "管理你的登陸驗證方式。", | ||||
|   "accountProfileEditApplied": "個人資料修改已被應用。", | ||||
|   "publishersNew": "新發布者", | ||||
|   "publisherNewSubtitle": "建立一個新的公共身份。", | ||||
|   "publisherSyncWithAccount": "同步賬戶資訊", | ||||
|   "publisherNewSubtitle": "創建一個新的公共身份。", | ||||
|   "publisherSyncWithAccount": "同步賬戶信息", | ||||
|   "publisherTotalUpvote": "總頂數", | ||||
|   "publisherTotalDownvote": "總踩數", | ||||
|   "publisherSocialPoint": "社會信用點", | ||||
| @@ -115,34 +135,37 @@ | ||||
|   "publisherAffiliatedBy": "隸屬於 {}", | ||||
|   "publisherRunBy": "由 {} 管理", | ||||
|   "fieldPublisherBelongToRealm": "所屬領域", | ||||
|   "fieldPublisherBelongToRealmUnset": "未設定釋出者所屬領域", | ||||
|   "fieldPublisherBelongToRealmUnset": "未設置發佈者所屬領域", | ||||
|   "writePostTypeStory": "發動態", | ||||
|   "writePostTypeArticle": "寫文章", | ||||
|   "fieldPostPublisher": "帖子釋出者", | ||||
|   "fieldPostPublisher": "帖子發佈者", | ||||
|   "fieldPostContent": "發生什麼事了?!", | ||||
|   "fieldPostTitle": "標題", | ||||
|   "fieldPostDescription": "描述", | ||||
|   "fieldPostTags": "標籤", | ||||
|   "postPublish": "釋出", | ||||
|   "postPublishedAt": "釋出於", | ||||
|   "postPublishedUntil": "取消釋出於", | ||||
|   "fieldPostCategories": "分類", | ||||
|   "fieldPostAlias": "別名", | ||||
|   "fieldPostAliasHint": "可選項,用於在 URL 中表示該帖子,應遵循 URL-Safe 的原則。", | ||||
|   "postPublish": "發佈", | ||||
|   "postPublishedAt": "發佈於", | ||||
|   "postPublishedUntil": "取消發佈於", | ||||
|   "postVisibility": "可見性", | ||||
|   "postVisibilityDescription": "帖子可見性決定了誰能檢視該篇帖子。", | ||||
|   "postVisibilityDescription": "帖子可見性決定了誰能查看該篇帖子。", | ||||
|   "postVisibilityAll": "所有人可見", | ||||
|   "postVisibilityFriends": "僅限好友可見", | ||||
|   "postVisibilitySelected": "選定的使用者可見", | ||||
|   "postVisibilityFiltered": "選定使用者不可見", | ||||
|   "postVisibilitySelected": "選定的用戶可見", | ||||
|   "postVisibilityFiltered": "選定用戶不可見", | ||||
|   "postVisibilityNone": "僅自己可見", | ||||
|   "postVisibleUsers": "可見的使用者", | ||||
|   "postInvisibleUsers": "不可見的使用者", | ||||
|   "postVisibleUsers": "可見的用戶", | ||||
|   "postInvisibleUsers": "不可見的用戶", | ||||
|   "postSelectedUsers": { | ||||
|     "zero": "未選擇使用者", | ||||
|     "one": "選擇了 {} 個使用者", | ||||
|     "other": "選擇了 {} 個使用者" | ||||
|     "zero": "未選擇用戶", | ||||
|     "one": "選擇了 {} 個用戶", | ||||
|     "other": "選擇了 {} 個用戶" | ||||
|   }, | ||||
|   "postEditingNotice": "你正在修改由 {} 釋出的帖子。", | ||||
|   "postReplyingNotice": "你正在回覆由 {} 釋出的帖子。", | ||||
|   "postRepostingNotice": "你正在轉發由 {} 釋出的帖子。", | ||||
|   "postEditingNotice": "你正在修改由 {} 發佈的帖子。", | ||||
|   "postReplyingNotice": "你正在回覆由 {} 發佈的帖子。", | ||||
|   "postRepostingNotice": "你正在轉發由 {} 發佈的帖子。", | ||||
|   "postReact": "反應", | ||||
|   "postPosted": "帖子已經發表。", | ||||
|   "postReactions": "帖子的反應", | ||||
| @@ -161,7 +184,7 @@ | ||||
|     "one": "{} 點社會信用點變更", | ||||
|     "other": "{} 點社會信用點變更" | ||||
|   }, | ||||
|   "postReactCompleted": "反應已被新增。", | ||||
|   "postReactCompleted": "反應已被添加。", | ||||
|   "postReactUncompleted": "反應已被移除。", | ||||
|   "postComments": { | ||||
|     "zero": "評論", | ||||
| @@ -174,71 +197,92 @@ | ||||
|     "other": "{} 條評論" | ||||
|   }, | ||||
|   "settingsAppearance": "外觀", | ||||
|   "settingsDisplayLanguage": "顯示語言", | ||||
|   "settingsDisplayLanguageDescription": "設置應用程序使用的語言", | ||||
|   "settingsDisplayLanguageSystem": "跟隨系統", | ||||
|   "settingsBackgroundImage": "背景圖片", | ||||
|   "settingsBackgroundImageDescription": "設定應用全域性生效的的背景圖片。", | ||||
|   "settingsBackgroundImageDescription": "設置應用全局生效的的背景圖片。", | ||||
|   "settingsBackgroundImageClear": "清除現存背景圖", | ||||
|   "settingsBackgroundImageClearDescription": "將應用背景圖重置為空白。", | ||||
|   "settingsThemeMaterial3": "使用 Material You 設計正規化", | ||||
|   "settingsThemeMaterial3Description": "將應用主題設定為 Material 3 設計正規化的主題。", | ||||
|   "settingsNetwork": "網路", | ||||
|   "settingsNetworkServer": "HyperNet 伺服器", | ||||
|   "settingsNetworkServerDescription": "設定 HyperNet 伺服器地址,選擇我們提供的,或者自己搭建。", | ||||
|   "settingsNetworkServerReset": "重設為官方伺服器", | ||||
|   "settingsNetworkServerResetDescription": "重設為 Solar Network 的伺服器地址。", | ||||
|   "settingsNetworkServerPreset": "預設的 HyperNet 伺服器", | ||||
|   "settingsNetworkServerPresetDescription": "你可以在旁邊的列表中選擇我們提供的預設 HyperNet 伺服器地址。", | ||||
|   "settingsNetworkServerSaved": "伺服器地址已儲存。", | ||||
|   "settingsPerformance": "效能", | ||||
|   "settingsThemeMaterial3": "使用 Material You 設計範式", | ||||
|   "settingsThemeMaterial3Description": "將應用主題設置為 Material 3 設計範式的主題。", | ||||
|   "settingsAppBarTransparent": "透明頂欄", | ||||
|   "settingsAppBarTransparentDescription": "為頂欄啟用透明效果。", | ||||
|   "settingsDrawerPreferCollapse": "側邊欄偏好摺疊", | ||||
|   "settingsDrawerPreferCollapseDescription": "將側邊欄優先摺疊,即使屏幕寬度足夠大去放下整個側邊欄。", | ||||
|   "settingsColorScheme": "主題色", | ||||
|   "settingsColorSchemeDescription": "設置應用主題色。", | ||||
|   "settingsColorSeed": "預設色彩主題", | ||||
|   "settingsColorSeedDescription": "選擇一個預設色彩主題。", | ||||
|   "settingsFeatures": "功能", | ||||
|   "settingsNotifyWithHaptic": "新通知時振動", | ||||
|   "settingsNotifyWithHapticDescription": "在應用在前臺時收到新通知出現時出發輕量的振動。", | ||||
|   "settingsExpandPostLink": "展開帖子鏈接", | ||||
|   "settingsExpandPostLinkDescription": "在帖子列表中展開顯示帖子中的鏈接。", | ||||
|   "settingsExpandChatLink": "展開聊天鏈接", | ||||
|   "settingsExpandChatLinkDescription": "在聊天信息中展開顯示內容中的鏈接。", | ||||
|   "settingsNetwork": "網絡", | ||||
|   "settingsNetworkServer": "HyperNet 服務器", | ||||
|   "settingsNetworkServerDescription": "設置 HyperNet 服務器地址,選擇我們提供的,或者自己搭建。", | ||||
|   "settingsNetworkServerReset": "重設為官方服務器", | ||||
|   "settingsNetworkServerResetDescription": "重設為 Solar Network 的服務器地址。", | ||||
|   "settingsNetworkServerPreset": "預設的 HyperNet 服務器", | ||||
|   "settingsNetworkServerPresetDescription": "你可以在旁邊的列表中選擇我們提供的預設 HyperNet 服務器地址。", | ||||
|   "settingsNetworkServerSaved": "服務器地址已保存。", | ||||
|   "settingsPerformance": "性能", | ||||
|   "settingsImageQuality": "圖片預覽質量", | ||||
|   "settingsImageQualityDescription": "設定圖片預覽質量,會影響圖片解碼速度。", | ||||
|   "settingsImageQualityDescription": "設置圖片預覽質量,會影響圖片解碼速度。", | ||||
|   "settingsImageQualityLowest": "極低", | ||||
|   "settingsImageQualityLow": "低", | ||||
|   "settingsImageQualityMedium": "中", | ||||
|   "settingsImageQualityHigh": "高", | ||||
|   "settingsMisc": "雜項", | ||||
|   "settingsMiscAbout": "關於", | ||||
|   "settingsMiscAboutDescription": "檢視 Solian 的版本資訊。", | ||||
|   "settingsMiscAboutDescription": "查看 Solian 的版本信息。", | ||||
|   "settingsAccountLanguage": "帳號偏好語言", | ||||
|   "settingsAccountLanguageDescription": "設置郵件、通知和其他帳號相關內容的語言。", | ||||
|   "sensitiveContent": "敏感內容", | ||||
|   "sensitiveContentCollapsed": "敏感內容已摺疊。", | ||||
|   "sensitiveContentDescription": "此內容已被標記,可能不適合所有人檢視。", | ||||
|   "sensitiveContentDescription": "此內容已被標記,可能不適合所有人查看。", | ||||
|   "sensitiveContentReveal": "顯示內容", | ||||
|   "serverConnecting": "正在連線伺服器…", | ||||
|   "serverDisconnected": "已與伺服器斷開連線", | ||||
|   "serverConnecting": "正在連接…", | ||||
|   "serverDisconnected": "已斷開連接", | ||||
|   "serverConnected": "已連接", | ||||
|   "fieldChatAlias": "頻道別名", | ||||
|   "fieldChatAliasHint": "全站範圍內唯一的頻道別名,用於在 URL 中表示該頻道,留空則自動生成。應遵循 URL-Safe 的原則。", | ||||
|   "fieldChatName": "名稱", | ||||
|   "fieldChatDescription": "描述", | ||||
|   "fieldChatBelongToRealm": "所屬領域", | ||||
|   "fieldChatBelongToRealmUnset": "未設定頻道所屬領域", | ||||
|   "fieldChatBelongToRealmUnset": "未設置頻道所屬領域", | ||||
|   "channelEditingNotice": "您正在編輯頻道 {}", | ||||
|   "channelDeleted": "聊天頻道 {} 已被刪除", | ||||
|   "channelDelete": "刪除聊天頻道 {}", | ||||
|   "channelDeleteDescription": "你確定要刪除這個聊天頻道嗎?該操作不可撤銷,其頻道內的所有訊息將被永久刪除。", | ||||
|   "channelDeleteDescription": "你確定要刪除這個聊天頻道嗎?該操作不可撤銷,其頻道內的所有消息將被永久刪除。", | ||||
|   "channelDetailPersonalRegion": "個人區域", | ||||
|   "channelDetailMemberRegion": "成員管理", | ||||
|   "channelMemberManage": "管理成員", | ||||
|   "channelMemberManageDescription": "管理頻道內現有成員。", | ||||
|   "channelMemberAdd": "新增成員", | ||||
|   "channelMemberAddDescription": "給當前頻道新增新成員。", | ||||
|   "channelMemberAdded": "頻道成員已新增。", | ||||
|   "channelMemberAdd": "添加成員", | ||||
|   "channelMemberAddDescription": "給當前頻道添加新成員。", | ||||
|   "channelMemberAdded": "頻道成員已添加。", | ||||
|   "fieldMemberRelatedName": "成員名 / 賬戶 ID", | ||||
|   "channelDetailAdminRegion": "管理區域", | ||||
|   "channelEditProfile": "更改頻道身份", | ||||
|   "channelEdit": "編輯頻道", | ||||
|   "channelEditDescription": "更改頻道基本資訊,元資料等。", | ||||
|   "channelEditDescription": "更改頻道基本信息,元數據等。", | ||||
|   "channelProfileEdit": "編輯頻道身份", | ||||
|   "channelActionDelete": "刪除頻道", | ||||
|   "channelActionDeleteDescription": "刪除整個頻道,並且刪除頻道里的所有資訊。", | ||||
|   "channelActionDeleteDescription": "刪除整個頻道,並且刪除頻道里的所有信息。", | ||||
|   "channelLeave": "退出頻道 {}", | ||||
|   "channelLeaveDescription": "退出該頻道,但是你頻道內的資訊不會被移除。", | ||||
|   "channelLeaveDescription": "退出該頻道,但是你頻道內的信息不會被移除。", | ||||
|   "channelActionLeave": "退出頻道", | ||||
|   "channelActionLeaveDescription": "刪除你在這個頻道的身份。", | ||||
|   "channelNotifyLevel": "通知級別", | ||||
|   "channelNotifyLevelDescription": "有您決定要接受多少來自這個頻道的訊息。", | ||||
|   "channelNotifyLevelDescription": "有您決定要接受多少來自這個頻道的消息。", | ||||
|   "channelNotifyLevelAll": "全部通知", | ||||
|   "channelNotifyLevelMentioned": "僅提及", | ||||
|   "channelNotifyLevelNone": "全部靜音", | ||||
|   "channelNotifyLevelApplied": "已經儲存並應用頻道通知級別配置。", | ||||
|   "channelNotifyLevelApplied": "已經保存並應用頻道通知級別配置。", | ||||
|   "fieldChannelProfileNick": "頻道內顯示名", | ||||
|   "fieldChannelProfileNickHint": "在頻道內顯示的暱稱,留空則使用賬號顯示名。", | ||||
|   "fieldRealmAlias": "領域別名", | ||||
| @@ -248,38 +292,72 @@ | ||||
|   "realmEditingNotice": "您正在編輯領域 {}", | ||||
|   "realmDeleted": "領域 {} 已被刪除", | ||||
|   "realmDelete": "刪除領域 {}", | ||||
|   "realmDeleteDescription": "你確定要刪除這個領域嗎?該操作不可撤銷,其隸屬於該領域的所有資源(帖子、聊天頻道、釋出者、製品等)都將被永久刪除。三思而後行!", | ||||
|   "realmDeleteDescription": "你確定要刪除這個領域嗎?該操作不可撤銷,其隸屬於該領域的所有資源(帖子、聊天頻道、發佈者、製品等)都將被永久刪除。三思而後行!", | ||||
|   "realmActionDelete": "刪除領域", | ||||
|   "realmActionDeleteDescription": "刪除整個領域及其附屬的資源。", | ||||
|   "realmEdit": "編輯領域", | ||||
|   "realmEditDescription": "更改領域基本資訊,元資料等。", | ||||
|   "realmMemberAdd": "新增成員", | ||||
|   "realmMemberAddDescription": "給當前領域新增新成員。", | ||||
|   "realmMemberAdded": "領域成員已新增。", | ||||
|   "fieldChatMessage": "在 {} 中發訊息", | ||||
|   "fieldChatMessageDirect": "給 {} 發訊息", | ||||
|   "eventResourceTag": "訊息 {}", | ||||
|   "messageDelete": "刪除訊息 {}", | ||||
|   "messageDeleteDescription": "你確定要刪除這個訊息嗎?該操作不可撤銷。同時您將留下一條刪除訊息的記錄。", | ||||
|   "messageDeleted": "訊息 {} 已被刪除", | ||||
|   "messageEdited": "訊息 {} 已被編輯", | ||||
|   "realmEditDescription": "更改領域基本信息,元數據等。", | ||||
|   "realmMemberAdd": "添加成員", | ||||
|   "realmMemberAddDescription": "給當前領域添加新成員。", | ||||
|   "realmMemberAdded": "領域成員已添加。", | ||||
|   "fieldChatMessage": "在 {} 中發消息", | ||||
|   "fieldChatMessageDirect": "給 {} 發消息", | ||||
|   "eventResourceTag": "消息 {}", | ||||
|   "messageDelete": "刪除消息 {}", | ||||
|   "messageDeleteDescription": "你確定要刪除這個消息嗎?該操作不可撤銷。同時您將留下一條刪除消息的記錄。", | ||||
|   "messageDeleted": "消息 {} 已被刪除", | ||||
|   "messageEdited": "消息 {} 已被編輯", | ||||
|   "messageEditedHint": "已編輯", | ||||
|   "messageUnsupported": "不支援的訊息 {}", | ||||
|   "messageUnsupported": "不支持的消息 {}", | ||||
|   "messageFileHint": { | ||||
|     "zero": "沒有附件", | ||||
|     "one": "{} 個附件", | ||||
|     "other": "{} 個附件" | ||||
|   }, | ||||
|   "addAttachmentFromAlbum": "從相簿中新增附件", | ||||
|   "addAttachmentFromClipboard": "貼上附件", | ||||
|   "messageTyping": { | ||||
|     "one": "{} 正在輸入", | ||||
|     "other": "{} 正在輸入" | ||||
|   }, | ||||
|   "fieldAttachmentRandomId": "訪問 ID", | ||||
|   "fieldAttachmentAlt": "概述文字", | ||||
|   "addAttachmentFromAlbum": "從相冊中添加附件", | ||||
|   "addAttachmentFromClipboard": "粘貼附件", | ||||
|   "addAttachmentFromCameraPhoto": "拍攝照片", | ||||
|   "addAttachmentFromCameraVideo": "拍攝影片", | ||||
|   "attachmentPastedImage": "貼上的圖片", | ||||
|   "attachmentInsertLink": "插入連線", | ||||
|   "attachmentSetAsPostThumbnail": "設定為帖子縮圖", | ||||
|   "attachmentUnsetAsPostThumbnail": "取消設定為帖子縮圖", | ||||
|   "attachmentSetThumbnail": "設定縮圖", | ||||
|   "addAttachmentFromCameraVideo": "拍攝視頻", | ||||
|   "addAttachmentFromRandomId": "通過訪問 ID 鏈接", | ||||
|   "attachmentDetailInfo": "附件詳細信息", | ||||
|   "attachmentPastedImage": "粘貼的圖片", | ||||
|   "attachmentInsertLink": "插入連接", | ||||
|   "attachmentSetAsPostThumbnail": "設置為帖子縮略圖", | ||||
|   "attachmentUnsetAsPostThumbnail": "取消設置為帖子縮略圖", | ||||
|   "attachmentCompressVideo": "重新編碼視頻", | ||||
|   "attachmentSetThumbnail": "設置縮略圖", | ||||
|   "attachmentSetAlt": "設置概述文字", | ||||
|   "attachmentCopyRandomId": "複製訪問 ID", | ||||
|   "attachmentUpload": "上傳", | ||||
|   "attachmentInputDialog": "上傳附件", | ||||
|   "attachmentInputUseRandomId": "使用訪問 ID", | ||||
|   "attachmentInputNew": "新上傳附件", | ||||
|   "waitingForUpload": "等待上傳", | ||||
|   "attachmentVideoCompressHint": "壓縮一份視頻的副本", | ||||
|   "attachmentVideoCompressHintDescription": "你想上傳壓縮視頻 {} 的副本嗎?它將幫助你的觀眾快速預覽視頻,並且他們仍然可以觀看原始視頻。這將會在在你的設備上處理視頻,所以需要一些時間,所以請耐心等待。", | ||||
|   "attachmentCompressQuality": "壓縮質量", | ||||
|   "attachmentCompressQualityHighest": "最高", | ||||
|   "attachmentCompressQualityDefault": "默認", | ||||
|   "attachmentCompressQualityMedium": "中等", | ||||
|   "attachmentCompressQualityLow": "低", | ||||
|   "attachmentCompressQualityHint": "Solar Network 並沒有阻止你上傳大文件、高分辨率、高碼率的視頻,但是為了你的網絡情況觀眾考慮,我們建議你選擇一個合適的壓縮質量。", | ||||
|   "attachmentUploaded": "已上傳", | ||||
|   "attachmentPending": "未上傳", | ||||
|   "attachmentCopyCompressed": "有壓縮副本", | ||||
|   "attachmentGotBoosted": "有加速傳遞", | ||||
|   "attachmentBoost": "加速包", | ||||
|   "attachmentCreateBoost": "加速傳遞", | ||||
|   "attachmentBoostHint": "加速傳遞允許您將附件上傳到更近的受眾或更快的內容網絡。該功能目前處於 Beta 階段。該功能限時免費,當有價格計劃更改時,您將會被通知。", | ||||
|   "attachmentDestinationRegion": "目標節點", | ||||
|   "attachmentDestinationRegionAPAC": "亞太地區", | ||||
|   "attachmentDestinationRegionNGB": "中國 · 浙江 · 寧波", | ||||
|   "attachmentDestinationRegionHKG": "香港", | ||||
|   "notification": "通知", | ||||
|   "notificationUnreadCount": { | ||||
|     "zero": "無未讀通知", | ||||
| @@ -289,18 +367,18 @@ | ||||
|   "notificationUnread": "未讀", | ||||
|   "notificationRead": "已讀", | ||||
|   "notificationMarkAllRead": "已讀所有通知", | ||||
|   "notificationMarkAllReadDescription": "您確定要將所有通知設定為已讀嗎?該操作不可撤銷。", | ||||
|   "notificationMarkAllReadDescription": "您確定要將所有通知設置為已讀嗎?該操作不可撤銷。", | ||||
|   "notificationMarkAllReadPrompt": { | ||||
|     "zero": "已將 0 個通知標記為已讀。", | ||||
|     "one": "已將 {} 個通知標記為已讀。", | ||||
|     "other": "已將 {} 個通知標記為已讀。" | ||||
|   }, | ||||
|   "notificationMarkOneReadPrompt": "已將通知 {} 標記為已讀。", | ||||
|   "search": "搜尋", | ||||
|   "search": "搜索", | ||||
|   "postSearchResult": { | ||||
|     "zero": "沒有搜尋到結果", | ||||
|     "one": "搜尋到 {} 個結果", | ||||
|     "other": "搜尋到 {} 個結果" | ||||
|     "zero": "沒有搜索到結果", | ||||
|     "one": "搜索到 {} 個結果", | ||||
|     "other": "搜索到 {} 個結果" | ||||
|   }, | ||||
|   "postSearchTook": "耗時 {}", | ||||
|   "postDelete": "刪除帖子 {}", | ||||
| @@ -312,26 +390,26 @@ | ||||
|   "callResume": "恢復", | ||||
|   "callMicrophone": "麥克風", | ||||
|   "callCamera": "攝像頭", | ||||
|   "callMicrophoneDisabled": "麥克風已停用", | ||||
|   "callMicrophoneDisabled": "麥克風已禁用", | ||||
|   "callMicrophoneSelect": "選擇麥克風", | ||||
|   "callCameraDisabled": "攝像頭已停用", | ||||
|   "callCameraDisabled": "攝像頭已禁用", | ||||
|   "callCameraSelect": "選擇攝像頭", | ||||
|   "callDisconnected": "通話已斷開", | ||||
|   "callEnded": "通話已結束", | ||||
|   "callStatusConnected": "已連線", | ||||
|   "callStatusDisconnected": "未連線", | ||||
|   "callStatusConnecting": "正在連線", | ||||
|   "callStatusConnected": "已連接", | ||||
|   "callStatusDisconnected": "未連接", | ||||
|   "callStatusConnecting": "正在連接", | ||||
|   "callStatusReconnecting": "正在重連", | ||||
|   "callDisconnect": "斷開連線", | ||||
|   "callDisconnectDescription": "您確定要與通話斷開連線嗎?", | ||||
|   "callDisconnect": "斷開連接", | ||||
|   "callDisconnectDescription": "您確定要與通話斷開連接嗎?", | ||||
|   "callMicrophoneOff": "關閉麥克風", | ||||
|   "callMicrophoneOn": "開啟麥克風", | ||||
|   "callMicrophoneOn": "打開麥克風", | ||||
|   "callCameraOff": "關閉攝像頭", | ||||
|   "callCameraOn": "開啟攝像頭", | ||||
|   "callVideoFlip": "映象畫面", | ||||
|   "callCameraOn": "打開攝像頭", | ||||
|   "callVideoFlip": "鏡像畫面", | ||||
|   "callSpeakerphoneToggle": "切換揚聲器", | ||||
|   "callScreenOff": "關閉螢幕共享", | ||||
|   "callScreenOn": "開啟螢幕共享", | ||||
|   "callScreenOff": "關閉屏幕共享", | ||||
|   "callScreenOn": "開啟屏幕共享", | ||||
|   "callMessageEnded": "通話持續了 {}", | ||||
|   "callMessageStarted": "通話開始了", | ||||
|   "dailyCheckIn": "每日簽到", | ||||
| @@ -367,28 +445,53 @@ | ||||
|   "dailyCheckNegativeHint5Description": "關鍵時刻斷網", | ||||
|   "dailyCheckNegativeHint6": "出門", | ||||
|   "dailyCheckNegativeHint6Description": "忘帶傘遇上大雨", | ||||
|   "happyBirthday": "生日快樂,{}!", | ||||
|   "friendNew": "新增好友", | ||||
|   "celebrateBirthday": "生日快樂,{}!", | ||||
|   "celebrateLunarNewYear": "春節快樂,{}!", | ||||
|   "celebrateMidAutumn": "中秋節快樂,{}!", | ||||
|   "celebrateDragonBoat": "端午節快樂,{}!", | ||||
|   "celebrateMerryXmas": "聖誕快樂,{}!", | ||||
|   "celebrateNewYear": "新年快樂,{}!", | ||||
|   "celebrateValentineDay": "今天是情人節,{}!", | ||||
|   "celebrateLaborDay": "今天是勞動節,{}。", | ||||
|   "celebrateMotherDay": "今天是母親節,{}。", | ||||
|   "celebrateChildrenDay": "今天是兒童節,{}!", | ||||
|   "celebrateFatherDay": "今天是父親節,{}。", | ||||
|   "celebrateHalloween": "快樂在聖誕節,{}!", | ||||
|   "celebrateThanksgiving": "今天是感恩節,{}!", | ||||
|   "pendingLunarNewYear": "{} 過春節", | ||||
|   "pendingMidAutumn": "{} 過中秋節", | ||||
|   "pendingDragonBoat": "{} 過端午節", | ||||
|   "pendingBirthday": "{} 過生日", | ||||
|   "pendingMerryXmas": "{} 過聖誕節", | ||||
|   "pendingNewYear": "{} 跨年", | ||||
|   "pendingValentineDay": "{} 過情人節", | ||||
|   "pendingLaborDay": "{} 過勞動節", | ||||
|   "pendingMotherDay": "{} 過母親節", | ||||
|   "pendingChildrenDay": "{} 過兒童節", | ||||
|   "pendingFatherDay": "{} 過父親節", | ||||
|   "pendingHalloween": "{} 過聖誕節", | ||||
|   "pendingThanksgiving": "{} 過感恩節", | ||||
|   "friendNew": "添加好友", | ||||
|   "friendRequests": "好友請求", | ||||
|   "friendRequestsDescription": { | ||||
|     "zero": "你沒有好友請求", | ||||
|     "one": "你有 {} 個好友請求", | ||||
|     "other": "你有 {} 個好友請求" | ||||
|   }, | ||||
|   "friendBlocklist": "遮蔽列表", | ||||
|   "friendBlocklist": "屏蔽列表", | ||||
|   "friendBlocklistDescription": { | ||||
|     "zero": "你沒有遮蔽任何人", | ||||
|     "one": "你遮蔽了 {} 個使用者", | ||||
|     "other": "你遮蔽了 {} 個使用者" | ||||
|     "zero": "你沒有屏蔽任何人", | ||||
|     "one": "你屏蔽了 {} 個用戶", | ||||
|     "other": "你屏蔽了 {} 個用戶" | ||||
|   }, | ||||
|   "friendStatusPending": "待處理", | ||||
|   "friendStatusWaiting": "等待中", | ||||
|   "friendStatusActive": "正活躍", | ||||
|   "friendStatusBlocked": "已遮蔽", | ||||
|   "friendRequestSent": "好友請求已傳送。", | ||||
|   "friendStatusBlocked": "已屏蔽", | ||||
|   "friendRequestSent": "好友請求已發送。", | ||||
|   "fieldFriendRelatedName": "好友名 / 賬戶 ID", | ||||
|   "friendBlock": "遮蔽", | ||||
|   "friendUnblock": "解除遮蔽", | ||||
|   "friendBlock": "屏蔽", | ||||
|   "friendUnblock": "解除屏蔽", | ||||
|   "friendDeleteAction": "遺忘", | ||||
|   "friendDelete": "遺忘跟 {} 的關係", | ||||
|   "friendDeleteDescription": "你確定要遺忘跟 {} 的關係嗎?這個操作無法撤銷。", | ||||
| @@ -401,23 +504,25 @@ | ||||
|   "accountJoinedAt": "加入於 {}", | ||||
|   "accountBirthday": "出生於 {}", | ||||
|   "accountBadge": "徽章", | ||||
|   "accountCheckInNoRecords": "暫無運勢記錄", | ||||
|   "badgeCompanyStaff": "索爾辛茨士大夫 · 員工", | ||||
|   "badgeSiteMigration": "Solar Network 原住民", | ||||
|   "accountStatus": "狀態", | ||||
|   "accountStatusOnline": "線上", | ||||
|   "accountStatusOnline": "在線", | ||||
|   "accountStatusOffline": "離線", | ||||
|   "accountStatusLastSeen": "最後一次在 {} 上線", | ||||
|   "accountStatusLastSeen": "最後一次上線於 {}", | ||||
|   "postArticle": "Solar Network 上的文章", | ||||
|   "postStory": "Solar Network 上的故事", | ||||
|   "postLocalDraftRestored": "從本地恢復草稿", | ||||
|   "articleWrittenAt": "發表於 {}", | ||||
|   "articleEditedAt": "編輯於 {}", | ||||
|   "attachmentSaved": "已儲存到相簿", | ||||
|   "attachmentSavedDesktop": "已儲存到下載目錄", | ||||
|   "openInAlbum": "在相簿中開啟", | ||||
|   "attachmentSaved": "已保存到相冊", | ||||
|   "attachmentSavedDesktop": "已保存到下載目錄", | ||||
|   "openInAlbum": "在相冊中打開", | ||||
|   "postAbuseReport": "檢舉帖子", | ||||
|   "postAbuseReportDescription": "檢舉不符合我們使用者協議以及社群準則的帖子,來幫助我們更好的維護 Solar Network 上的內容。請在下面描述該帖子如何違反我麼的相關規定。請勿填寫任何敏感資訊。我們將會在 24 小時內處理您的檢舉。", | ||||
|   "postAbuseReportDescription": "檢舉不符合我們用戶協議以及社區準則的帖子,來幫助我們更好的維護 Solar Network 上的內容。請在下面描述該帖子如何違反我麼的相關規定。請勿填寫任何敏感信息。我們將會在 24 小時內處理您的檢舉。", | ||||
|   "abuseReport": "檢舉", | ||||
|   "abuseReportDescription": "檢舉不符合我們使用者協議以及社群準則的任何資源,來幫助我們更好的維護 Solar Network 上的內容。請在下面描述資源的位置(提供資源 ID 為佳)以及如何違反我麼的相關規定。請勿填寫任何敏感資訊。我們將會在 24 小時內處理您的檢舉。", | ||||
|   "abuseReportDescription": "檢舉不符合我們用戶協議以及社區準則的任何資源,來幫助我們更好的維護 Solar Network 上的內容。請在下面描述資源的位置(提供資源 ID 為佳)以及如何違反我麼的相關規定。請勿填寫任何敏感信息。我們將會在 24 小時內處理您的檢舉。", | ||||
|   "abuseReportAction": "提交檢舉", | ||||
|   "abuseReportActionDescription": "檢舉不合規行為。", | ||||
|   "abuseReportResource": "資源位置 / ID", | ||||
| @@ -425,33 +530,82 @@ | ||||
|   "abuseReportSubmitted": "檢舉已提交,感謝你的貢獻。", | ||||
|   "submit": "提交", | ||||
|   "accountDeletion": "刪除帳戶", | ||||
|   "accountDeletionDescription": "你確定要刪除這個帳戶嗎?該操作不可撤銷,其隸屬於該帳戶的所有資源(帖子、聊天頻道、釋出者、製品等)都將被永久刪除。三思而後行!", | ||||
|   "accountDeletionDescription": "你確定要刪除這個帳戶嗎?該操作不可撤銷,其隸屬於該帳戶的所有資源(帖子、聊天頻道、發佈者、製品等)都將被永久刪除。三思而後行!", | ||||
|   "accountDeletionActionDescription": "刪除你的 Solarpass 帳戶。", | ||||
|   "accountDeletionSubmitted": "帳戶刪除申請已發出,你可以檢查你的收件箱並根據郵件內的指示完成刪除操作。", | ||||
|   "channelNewChannel": "新建頻道", | ||||
|   "channelNewDirectMessage": "發起私信", | ||||
|   "channelDirectMessageDescription": "與 {} 的私聊", | ||||
|   "fieldCannotBeEmpty": "此欄位不能為空。", | ||||
|   "fieldCannotBeEmpty": "此字段不能為空。", | ||||
|   "termAcceptLink": "瀏覽條款", | ||||
|   "termAcceptNextWithAgree": "點選 “下一步”,即表示你同意我們的各項條款,包括其之後的更新。", | ||||
|   "termAcceptNextWithAgree": "點擊 “下一步”,即表示你同意我們的各項條款,包括其之後的更新。", | ||||
|   "unauthorized": "未登陸", | ||||
|   "unauthorizedDescription": "登陸以探索整個 Solar Network。", | ||||
|   "serviceStatus": "服務狀態", | ||||
|   "termRelated": "相關條款", | ||||
|   "appDetails": "應用程式詳情", | ||||
|   "appDetails": "應用程序詳情", | ||||
|   "postRecommendation": "推薦帖子", | ||||
|   "publisherBlockHint": "遮蔽 {}", | ||||
|   "publisherBlockHintDescription": "你正要遮蔽此釋出者的運營者,該操作也將遮蔽由同一使用者運營的釋出者。", | ||||
|   "userUnblocked": "已解除遮蔽使用者 {}", | ||||
|   "userBlocked": "已遮蔽使用者 {}", | ||||
|   "publisherBlockHint": "屏蔽 {}", | ||||
|   "publisherBlockHintDescription": "你正要屏蔽此發佈者的運營者,該操作也將屏蔽由同一用戶運營的發佈者。", | ||||
|   "userUnblocked": "已解除屏蔽用戶 {}", | ||||
|   "userBlocked": "已屏蔽用戶 {}", | ||||
|   "postSharingViaPicture": "正在生成帖子截圖,請稍等片刻……", | ||||
|   "postImageShareReadMore": "掃描右側 QRCode 檢視全文", | ||||
|   "postImageShareReadMore": "掃描右側 QRCode 查看全文", | ||||
|   "postImageShareAds": "來 Solar Network 探索更多有趣帖子", | ||||
|   "postShare": "分享", | ||||
|   "postShareImage": "分享帖圖", | ||||
|   "postGetInsight": "獲取見解", | ||||
|   "postGetInsightTitle": "AI 見解", | ||||
|   "postGetInsightDescription": "AI 可能會出錯,檢查信息真實性。", | ||||
|   "appInitializing": "正在初始化", | ||||
|   "poweredBy": "由 {} 提供支援", | ||||
|   "poweredBy": "由 {} 提供支持", | ||||
|   "shareIntent": "分享", | ||||
|   "shareIntentDescription": "您想對您分享的內容做些什麼?", | ||||
|   "shareIntentPostStory": "釋出動態" | ||||
|   "shareIntentPostStory": "發佈動態", | ||||
|   "shareIntentSendChannel": "分享到聊天頻道", | ||||
|   "updateAvailable": "檢測到更新可用", | ||||
|   "updateOngoing": "正在更新,請稍後……", | ||||
|   "custom": "自定義", | ||||
|   "colorSchemeIndigo": "靛藍", | ||||
|   "colorSchemeBlue": "藍色", | ||||
|   "colorSchemeGreen": "綠色", | ||||
|   "colorSchemeYellow": "黃色", | ||||
|   "colorSchemeOrange": "橙色", | ||||
|   "colorSchemeRed": "紅色", | ||||
|   "colorSchemeWhite": "白色", | ||||
|   "colorSchemeBlack": "黑色", | ||||
|   "colorSchemeApplied": "主題色已應用,可能需要重啟來生效。", | ||||
|   "postFeaturedComment": "精選評論", | ||||
|   "postCategoryTechnology": "技術", | ||||
|   "postCategoryGaming": "遊戲", | ||||
|   "postCategoryLife": "生活", | ||||
|   "postCategoryArts": "藝術", | ||||
|   "postCategorySports": "體育", | ||||
|   "postCategoryMusic": "音樂", | ||||
|   "postCategoryNews": "新聞", | ||||
|   "postCategoryKnowledge": "知識", | ||||
|   "postCategoryLiterature": "文學", | ||||
|   "postCategoryFunny": "搞笑", | ||||
|   "postCategoryUncategorized": "未分類", | ||||
|   "newsAllSources": "所有新聞", | ||||
|   "newsReadingProviderSwap": "切換", | ||||
|   "newsReadingFromReader": "你正在從 HyperNet.Reader 閱讀文章", | ||||
|   "newsReadingFromOriginal": "你正在閱讀原始文章", | ||||
|   "newsDisclaimer": "本文由 HyperNet.Reader 從互聯網上獲取,我們不擔保其內容的真實性,請自行判斷。本文章的所有內容版權歸原作者所有。", | ||||
|   "newsToday": "快訊", | ||||
|   "totpPostSetup": "還有一件事", | ||||
|   "totpPostSetupDescription": "使用 Google Authenticator, Microsoft Authenticator, 1Password, Authy, Bitwarden 或其他支持 TOTP 的驗證器掃描本 QR Code 來添加。", | ||||
|   "totpNeverShare": "永遠不要分享這個 QR Code", | ||||
|   "needHelp": "需要幫助?", | ||||
|   "needHelpLaunch": "查看我們的山羊維基!", | ||||
|   "walletCreate": "創建錢包", | ||||
|   "walletCreateSubtitle": "創建於一個錢包來開始使用源點。", | ||||
|   "walletCreatePassword": "在下方設置你的付款密碼", | ||||
|   "walletCurrencyShort": "源點", | ||||
|   "walletCurrency": { | ||||
|     "one": "{} 源點", | ||||
|     "other": "{} 源點" | ||||
|   }, | ||||
|   "aiThinkingProcess": "AI 思考過程", | ||||
|   "accountSettingsApplied": "帳號設置已應用。" | ||||
| } | ||||
|   | ||||
							
								
								
									
										118
									
								
								ios/Podfile.lock
									
									
									
									
									
								
							
							
						
						| @@ -43,58 +43,58 @@ PODS: | ||||
|     - Flutter | ||||
|   - file_saver (0.0.1): | ||||
|     - Flutter | ||||
|   - Firebase/Analytics (11.4.0): | ||||
|   - Firebase/Analytics (11.7.0): | ||||
|     - Firebase/Core | ||||
|   - Firebase/Core (11.4.0): | ||||
|   - Firebase/Core (11.7.0): | ||||
|     - Firebase/CoreOnly | ||||
|     - FirebaseAnalytics (~> 11.4.0) | ||||
|   - Firebase/CoreOnly (11.4.0): | ||||
|     - FirebaseCore (= 11.4.0) | ||||
|   - Firebase/Messaging (11.4.0): | ||||
|     - FirebaseAnalytics (~> 11.7.0) | ||||
|   - Firebase/CoreOnly (11.7.0): | ||||
|     - FirebaseCore (~> 11.7.0) | ||||
|   - Firebase/Messaging (11.7.0): | ||||
|     - Firebase/CoreOnly | ||||
|     - FirebaseMessaging (~> 11.4.0) | ||||
|   - firebase_analytics (11.3.6): | ||||
|     - Firebase/Analytics (= 11.4.0) | ||||
|     - FirebaseMessaging (~> 11.7.0) | ||||
|   - firebase_analytics (11.4.2): | ||||
|     - Firebase/Analytics (= 11.7.0) | ||||
|     - firebase_core | ||||
|     - Flutter | ||||
|   - firebase_core (3.9.0): | ||||
|     - Firebase/CoreOnly (= 11.4.0) | ||||
|   - firebase_core (3.11.0): | ||||
|     - Firebase/CoreOnly (= 11.7.0) | ||||
|     - Flutter | ||||
|   - firebase_messaging (15.1.6): | ||||
|     - Firebase/Messaging (= 11.4.0) | ||||
|   - firebase_messaging (15.2.2): | ||||
|     - Firebase/Messaging (= 11.7.0) | ||||
|     - firebase_core | ||||
|     - Flutter | ||||
|   - FirebaseAnalytics (11.4.0): | ||||
|     - FirebaseAnalytics/AdIdSupport (= 11.4.0) | ||||
|     - FirebaseCore (~> 11.0) | ||||
|   - FirebaseAnalytics (11.7.0): | ||||
|     - FirebaseAnalytics/AdIdSupport (= 11.7.0) | ||||
|     - FirebaseCore (~> 11.7.0) | ||||
|     - FirebaseInstallations (~> 11.0) | ||||
|     - GoogleUtilities/AppDelegateSwizzler (~> 8.0) | ||||
|     - GoogleUtilities/MethodSwizzler (~> 8.0) | ||||
|     - GoogleUtilities/Network (~> 8.0) | ||||
|     - "GoogleUtilities/NSData+zlib (~> 8.0)" | ||||
|     - nanopb (~> 3.30910.0) | ||||
|   - FirebaseAnalytics/AdIdSupport (11.4.0): | ||||
|     - FirebaseCore (~> 11.0) | ||||
|   - FirebaseAnalytics/AdIdSupport (11.7.0): | ||||
|     - FirebaseCore (~> 11.7.0) | ||||
|     - FirebaseInstallations (~> 11.0) | ||||
|     - GoogleAppMeasurement (= 11.4.0) | ||||
|     - GoogleAppMeasurement (= 11.7.0) | ||||
|     - GoogleUtilities/AppDelegateSwizzler (~> 8.0) | ||||
|     - GoogleUtilities/MethodSwizzler (~> 8.0) | ||||
|     - GoogleUtilities/Network (~> 8.0) | ||||
|     - "GoogleUtilities/NSData+zlib (~> 8.0)" | ||||
|     - nanopb (~> 3.30910.0) | ||||
|   - FirebaseCore (11.4.0): | ||||
|     - FirebaseCoreInternal (~> 11.0) | ||||
|   - FirebaseCore (11.7.0): | ||||
|     - FirebaseCoreInternal (~> 11.7.0) | ||||
|     - GoogleUtilities/Environment (~> 8.0) | ||||
|     - GoogleUtilities/Logger (~> 8.0) | ||||
|   - FirebaseCoreInternal (11.6.0): | ||||
|   - FirebaseCoreInternal (11.7.0): | ||||
|     - "GoogleUtilities/NSData+zlib (~> 8.0)" | ||||
|   - FirebaseInstallations (11.4.0): | ||||
|     - FirebaseCore (~> 11.0) | ||||
|   - FirebaseInstallations (11.7.0): | ||||
|     - FirebaseCore (~> 11.7.0) | ||||
|     - GoogleUtilities/Environment (~> 8.0) | ||||
|     - GoogleUtilities/UserDefaults (~> 8.0) | ||||
|     - PromisesObjC (~> 2.4) | ||||
|   - FirebaseMessaging (11.4.0): | ||||
|     - FirebaseCore (~> 11.0) | ||||
|   - FirebaseMessaging (11.7.0): | ||||
|     - FirebaseCore (~> 11.7.0) | ||||
|     - FirebaseInstallations (~> 11.0) | ||||
|     - GoogleDataTransport (~> 10.0) | ||||
|     - GoogleUtilities/AppDelegateSwizzler (~> 8.0) | ||||
| @@ -105,32 +105,39 @@ PODS: | ||||
|   - Flutter (1.0.0) | ||||
|   - flutter_app_update (0.0.1): | ||||
|     - Flutter | ||||
|   - flutter_inappwebview_ios (0.0.1): | ||||
|     - Flutter | ||||
|     - flutter_inappwebview_ios/Core (= 0.0.1) | ||||
|     - OrderedSet (~> 6.0.3) | ||||
|   - flutter_inappwebview_ios/Core (0.0.1): | ||||
|     - Flutter | ||||
|     - OrderedSet (~> 6.0.3) | ||||
|   - flutter_native_splash (2.4.3): | ||||
|     - Flutter | ||||
|   - flutter_udid (0.0.1): | ||||
|     - Flutter | ||||
|     - SAMKeychain | ||||
|   - flutter_webrtc (0.12.2): | ||||
|   - flutter_webrtc (0.12.6): | ||||
|     - Flutter | ||||
|     - WebRTC-SDK (= 125.6422.06) | ||||
|   - gal (1.0.0): | ||||
|     - Flutter | ||||
|     - FlutterMacOS | ||||
|   - GoogleAppMeasurement (11.4.0): | ||||
|     - GoogleAppMeasurement/AdIdSupport (= 11.4.0) | ||||
|   - GoogleAppMeasurement (11.7.0): | ||||
|     - GoogleAppMeasurement/AdIdSupport (= 11.7.0) | ||||
|     - GoogleUtilities/AppDelegateSwizzler (~> 8.0) | ||||
|     - GoogleUtilities/MethodSwizzler (~> 8.0) | ||||
|     - GoogleUtilities/Network (~> 8.0) | ||||
|     - "GoogleUtilities/NSData+zlib (~> 8.0)" | ||||
|     - nanopb (~> 3.30910.0) | ||||
|   - GoogleAppMeasurement/AdIdSupport (11.4.0): | ||||
|     - GoogleAppMeasurement/WithoutAdIdSupport (= 11.4.0) | ||||
|   - GoogleAppMeasurement/AdIdSupport (11.7.0): | ||||
|     - GoogleAppMeasurement/WithoutAdIdSupport (= 11.7.0) | ||||
|     - GoogleUtilities/AppDelegateSwizzler (~> 8.0) | ||||
|     - GoogleUtilities/MethodSwizzler (~> 8.0) | ||||
|     - GoogleUtilities/Network (~> 8.0) | ||||
|     - "GoogleUtilities/NSData+zlib (~> 8.0)" | ||||
|     - nanopb (~> 3.30910.0) | ||||
|   - GoogleAppMeasurement/WithoutAdIdSupport (11.4.0): | ||||
|   - GoogleAppMeasurement/WithoutAdIdSupport (11.7.0): | ||||
|     - GoogleUtilities/AppDelegateSwizzler (~> 8.0) | ||||
|     - GoogleUtilities/MethodSwizzler (~> 8.0) | ||||
|     - GoogleUtilities/Network (~> 8.0) | ||||
| @@ -172,8 +179,8 @@ PODS: | ||||
|     - Flutter | ||||
|   - in_app_review (2.0.0): | ||||
|     - Flutter | ||||
|   - Kingfisher (8.1.3) | ||||
|   - livekit_client (2.3.2): | ||||
|   - Kingfisher (8.2.0) | ||||
|   - livekit_client (2.3.5): | ||||
|     - Flutter | ||||
|     - flutter_webrtc | ||||
|     - WebRTC-SDK (= 125.6422.06) | ||||
| @@ -188,6 +195,7 @@ PODS: | ||||
|     - nanopb/encode (= 3.30910.0) | ||||
|   - nanopb/decode (3.30910.0) | ||||
|   - nanopb/encode (3.30910.0) | ||||
|   - OrderedSet (6.0.3) | ||||
|   - package_info_plus (0.4.5): | ||||
|     - Flutter | ||||
|   - pasteboard (0.0.1): | ||||
| @@ -217,6 +225,8 @@ PODS: | ||||
|   - SwiftyGif (5.4.5) | ||||
|   - url_launcher_ios (0.0.1): | ||||
|     - Flutter | ||||
|   - video_compress (0.3.0): | ||||
|     - Flutter | ||||
|   - volume_controller (0.0.1): | ||||
|     - Flutter | ||||
|   - wakelock_plus (0.0.1): | ||||
| @@ -237,6 +247,7 @@ DEPENDENCIES: | ||||
|   - firebase_messaging (from `.symlinks/plugins/firebase_messaging/ios`) | ||||
|   - Flutter (from `Flutter`) | ||||
|   - flutter_app_update (from `.symlinks/plugins/flutter_app_update/ios`) | ||||
|   - flutter_inappwebview_ios (from `.symlinks/plugins/flutter_inappwebview_ios/ios`) | ||||
|   - flutter_native_splash (from `.symlinks/plugins/flutter_native_splash/ios`) | ||||
|   - flutter_udid (from `.symlinks/plugins/flutter_udid/ios`) | ||||
|   - flutter_webrtc (from `.symlinks/plugins/flutter_webrtc/ios`) | ||||
| @@ -259,6 +270,7 @@ DEPENDENCIES: | ||||
|   - shared_preferences_foundation (from `.symlinks/plugins/shared_preferences_foundation/darwin`) | ||||
|   - sqflite_darwin (from `.symlinks/plugins/sqflite_darwin/darwin`) | ||||
|   - url_launcher_ios (from `.symlinks/plugins/url_launcher_ios/ios`) | ||||
|   - video_compress (from `.symlinks/plugins/video_compress/ios`) | ||||
|   - volume_controller (from `.symlinks/plugins/volume_controller/ios`) | ||||
|   - wakelock_plus (from `.symlinks/plugins/wakelock_plus/ios`) | ||||
|   - workmanager (from `.symlinks/plugins/workmanager/ios`) | ||||
| @@ -279,6 +291,7 @@ SPEC REPOS: | ||||
|     - GoogleUtilities | ||||
|     - Kingfisher | ||||
|     - nanopb | ||||
|     - OrderedSet | ||||
|     - PromisesObjC | ||||
|     - SAMKeychain | ||||
|     - SDWebImage | ||||
| @@ -306,6 +319,8 @@ EXTERNAL SOURCES: | ||||
|     :path: Flutter | ||||
|   flutter_app_update: | ||||
|     :path: ".symlinks/plugins/flutter_app_update/ios" | ||||
|   flutter_inappwebview_ios: | ||||
|     :path: ".symlinks/plugins/flutter_inappwebview_ios/ios" | ||||
|   flutter_native_splash: | ||||
|     :path: ".symlinks/plugins/flutter_native_splash/ios" | ||||
|   flutter_udid: | ||||
| @@ -348,6 +363,8 @@ EXTERNAL SOURCES: | ||||
|     :path: ".symlinks/plugins/sqflite_darwin/darwin" | ||||
|   url_launcher_ios: | ||||
|     :path: ".symlinks/plugins/url_launcher_ios/ios" | ||||
|   video_compress: | ||||
|     :path: ".symlinks/plugins/video_compress/ios" | ||||
|   volume_controller: | ||||
|     :path: ".symlinks/plugins/volume_controller/ios" | ||||
|   wakelock_plus: | ||||
| @@ -362,35 +379,37 @@ SPEC CHECKSUMS: | ||||
|   device_info_plus: bf2e3232933866d73fe290f2942f2156cdd10342 | ||||
|   DKImagePickerController: 946cec48c7873164274ecc4624d19e3da4c1ef3c | ||||
|   DKPhotoGallery: b3834fecb755ee09a593d7c9e389d8b5d6deed60 | ||||
|   file_picker: 09aa5ec1ab24135ccd7a1621c46c84134bfd6655 | ||||
|   file_picker: b159e0c068aef54932bb15dc9fd1571818edaf49 | ||||
|   file_saver: 503e386464dbe118f630e17b4c2e1190fa0cf808 | ||||
|   Firebase: cf1b19f21410b029b6786a54e9764a0cacad3c99 | ||||
|   firebase_analytics: 2815af29d49c1a994652abd37a5b001a88bc7b75 | ||||
|   firebase_core: b62a5080210edad3f2934314a8b2c6f5124e8e10 | ||||
|   firebase_messaging: 98619a0572d82cfb3668e78859ba9f1110e268c9 | ||||
|   FirebaseAnalytics: 3feef9ae8733c567866342a1000691baaa7cad49 | ||||
|   FirebaseCore: e0510f1523bc0eb21653cac00792e1e2bd6f1771 | ||||
|   FirebaseCoreInternal: d98ab91e2d80a56d7b246856a8885443b302c0c2 | ||||
|   FirebaseInstallations: 6ef4a1c7eb2a61ee1f74727d7f6ce2e72acf1414 | ||||
|   FirebaseMessaging: f8a160d99c2c2e5babbbcc90c4a3e15db036aee2 | ||||
|   Firebase: a64bf6a8546e6eab54f1c715cd6151f39d2329f4 | ||||
|   firebase_analytics: 7236e6115c1b4e62c2270faa29c052a317e31107 | ||||
|   firebase_core: aa979ae726f00b3ef4ccf59dfb96170af84efbd4 | ||||
|   firebase_messaging: 3af84b6a90aeac4d7a67fbf4c43a91e7083bea1f | ||||
|   FirebaseAnalytics: bc9e565af9044ba8d6c6e4157e4edca8e5fdf7ec | ||||
|   FirebaseCore: 3227e35f4197a924206fbcdc0349325baf4f5de4 | ||||
|   FirebaseCoreInternal: d6c17dafc8dc33614733a8b52df78fcb4394c881 | ||||
|   FirebaseInstallations: 9347e719c3d52d8d7b9074b2c32407dd027305e9 | ||||
|   FirebaseMessaging: 00ece041b71ddb52a2862ffdee73fb6e9824bd0c | ||||
|   Flutter: e0871f40cf51350855a761d2e70bf5af5b9b5de7 | ||||
|   flutter_app_update: 65f61da626cb111d1b24674abc4b01728d7723bc | ||||
|   flutter_native_splash: e8a1e01082d97a8099d973f919f57904c925008a | ||||
|   flutter_inappwebview_ios: 6f63631e2c62a7c350263b13fa5427aedefe81d4 | ||||
|   flutter_native_splash: f71420956eb811e6d310720fee915f1d42852e7a | ||||
|   flutter_udid: b2417673f287ee62817a1de3d1643f47b9f508ab | ||||
|   flutter_webrtc: 1a53bd24f97bcfeff512f13699e721897f261563 | ||||
|   flutter_webrtc: 90260f83024b1b96d239a575ea4e3708e79344d1 | ||||
|   gal: 6a522c75909f1244732d4596d11d6a2f86ff37a5 | ||||
|   GoogleAppMeasurement: 987769c4ca6b968f2479fbcc9fe3ce34af454b8e | ||||
|   GoogleAppMeasurement: 0471a5b5bff51f3a91b1e76df22c952d04c63967 | ||||
|   GoogleDataTransport: aae35b7ea0c09004c3797d53c8c41f66f219d6a7 | ||||
|   GoogleUtilities: 26a3abef001b6533cf678d3eb38fd3f614b7872d | ||||
|   home_widget: 0434835a4c9a75704264feff6be17ea40e0f0d57 | ||||
|   image_picker_ios: c560581cceedb403a6ff17f2f816d7fea1421fc1 | ||||
|   in_app_review: a31b5257259646ea78e0e35fc914979b0031d011 | ||||
|   Kingfisher: f2af9028b16baf9dc6c07c570072bc41cbf009ef | ||||
|   livekit_client: 6108dad8b77db3142bafd4c630f471d0a54335cd | ||||
|   Kingfisher: 323e5c4ec7983aaace12af655a7b51a7f88a599d | ||||
|   livekit_client: dcc5fd47ba69c98fc6baeb12e862c9d43807d976 | ||||
|   media_kit_libs_ios_video: a5fe24bc7875ccd6378a0978c13185e1344651c1 | ||||
|   media_kit_native_event_loop: e6b2ab20cf0746eb1c33be961fcf79667304fa2a | ||||
|   media_kit_video: 5da63f157170e5bf303bf85453b7ef6971218a2e | ||||
|   nanopb: fad817b59e0457d11a5dfbde799381cd727c1275 | ||||
|   OrderedSet: e539b66b644ff081c73a262d24ad552a69be3a94 | ||||
|   package_info_plus: c0502532a26c7662a62a356cebe2692ec5fe4ec4 | ||||
|   pasteboard: 982969ebaa7c78af3e6cc7761e8f5e77565d9ce0 | ||||
|   path_provider_foundation: 2b6b4c569c0fb62ec74538f866245ac84301af46 | ||||
| @@ -405,6 +424,7 @@ SPEC CHECKSUMS: | ||||
|   sqflite_darwin: 5a7236e3b501866c1c9befc6771dfd73ffb8702d | ||||
|   SwiftyGif: 706c60cf65fa2bc5ee0313beece843c8eb8194d4 | ||||
|   url_launcher_ios: 5334b05cef931de560670eeae103fd3e431ac3fe | ||||
|   video_compress: fce97e4fb1dfd88175aa07d2ffc8a2f297f87fbe | ||||
|   volume_controller: 531ddf792994285c9b17f9d8a7e4dcdd29b3eae9 | ||||
|   wakelock_plus: 373cfe59b235a6dd5837d0fb88791d2f13a90d56 | ||||
|   WebRTC-SDK: 79942c006ea64f6fb48d7da8a4786dfc820bc1db | ||||
|   | ||||
| @@ -80,7 +80,11 @@ class NotificationService: UNNotificationServiceExtension { | ||||
|          | ||||
|         let metadataCopy = metadata as? [String: String] ?? [:] | ||||
|         let avatarUrl = getAttachmentUrl(for: avatarIdentifier) | ||||
|         KingfisherManager.shared.retrieveImage(with: URL(string: avatarUrl)!, completionHandler: { result in | ||||
|          | ||||
|         let targetSize = 640 | ||||
|         let scaleProcessor = ResizingImageProcessor(referenceSize: CGSize(width: targetSize, height: targetSize), mode: .aspectFit) | ||||
|          | ||||
|         KingfisherManager.shared.retrieveImage(with: URL(string: avatarUrl)!, options: [.processor(scaleProcessor)], completionHandler: { result in | ||||
|             var image: Data? | ||||
|             switch result { | ||||
|             case .success(let value): | ||||
|   | ||||
| @@ -1,4 +1,5 @@ | ||||
| import 'dart:async'; | ||||
| import 'dart:convert'; | ||||
| import 'dart:math' as math; | ||||
|  | ||||
| import 'package:collection/collection.dart'; | ||||
| @@ -11,6 +12,7 @@ import 'package:surface/providers/sn_network.dart'; | ||||
| import 'package:surface/providers/user_directory.dart'; | ||||
| import 'package:surface/providers/websocket.dart'; | ||||
| import 'package:surface/types/chat.dart'; | ||||
| import 'package:surface/types/websocket.dart'; | ||||
| import 'package:uuid/uuid.dart'; | ||||
|  | ||||
| class ChatMessageController extends ChangeNotifier { | ||||
| @@ -36,8 +38,7 @@ class ChatMessageController extends ChangeNotifier { | ||||
|  | ||||
|   int? messageTotal; | ||||
|  | ||||
|   bool get isAllLoaded => | ||||
|       messageTotal != null && messages.length >= messageTotal!; | ||||
|   bool get isAllLoaded => messageTotal != null && messages.length >= messageTotal!; | ||||
|  | ||||
|   String? _boxKey; | ||||
|   SnChannel? channel; | ||||
| @@ -50,8 +51,10 @@ class ChatMessageController extends ChangeNotifier { | ||||
|   /// Stored as a list of nonce to provide the loading state | ||||
|   final List<String> unconfirmedMessages = List.empty(growable: true); | ||||
|  | ||||
|   Box<SnChatMessage>? get _box => | ||||
|       (_boxKey == null || isPending) ? null : Hive.box<SnChatMessage>(_boxKey!); | ||||
|   Box<SnChatMessage>? get _box => (_boxKey == null || isPending) ? null : Hive.box<SnChatMessage>(_boxKey!); | ||||
|  | ||||
|   final List<SnChannelMember> typingMembers = List.empty(growable: true); | ||||
|   final Map<int, Timer> typingInactiveTimer = {}; | ||||
|  | ||||
|   Future<void> initialize(SnChannel chan) async { | ||||
|     channel = chan; | ||||
| @@ -71,6 +74,7 @@ class ChatMessageController extends ChangeNotifier { | ||||
|     _wsSubscription = _ws.stream.stream.listen((event) { | ||||
|       switch (event.method) { | ||||
|         case 'events.new': | ||||
|           if (event.payload?['channel_id'] != channel?.id) break; | ||||
|           final payload = SnChatMessage.fromJson(event.payload!); | ||||
|           _addMessage(payload); | ||||
|           break; | ||||
| @@ -78,22 +82,16 @@ class ChatMessageController extends ChangeNotifier { | ||||
|           if (event.payload?['channel_id'] != channel?.id) break; | ||||
|           final member = SnChannelMember.fromJson(event.payload!['member']); | ||||
|           if (member.id == profile?.id) break; | ||||
|         // TODO impl typing users | ||||
|         // if (!_typingUsers.any((x) => x.id == member.id)) { | ||||
|         //   setState(() { | ||||
|         //     _typingUsers.add(member); | ||||
|         //   }); | ||||
|         // } | ||||
|         // _typingInactiveTimer[member.id]?.cancel(); | ||||
|         // _typingInactiveTimer[member.id] = Timer( | ||||
|         //   const Duration(seconds: 3), | ||||
|         //   () { | ||||
|         //     setState(() { | ||||
|         //       _typingUsers.removeWhere((x) => x.id == member.id); | ||||
|         //       _typingInactiveTimer.remove(member.id); | ||||
|         //     }); | ||||
|         //   }, | ||||
|         // ); | ||||
|           if (!typingMembers.any((x) => x.id == member.id)) { | ||||
|             typingMembers.add(member); | ||||
|             notifyListeners(); | ||||
|           } | ||||
|           typingInactiveTimer[member.id]?.cancel(); | ||||
|           typingInactiveTimer[member.id] = Timer(const Duration(seconds: 3), () { | ||||
|             typingMembers.removeWhere((x) => x.id == member.id); | ||||
|             typingInactiveTimer.remove(member.id); | ||||
|             notifyListeners(); | ||||
|           }); | ||||
|       } | ||||
|     }); | ||||
|  | ||||
| @@ -101,6 +99,35 @@ class ChatMessageController extends ChangeNotifier { | ||||
|     notifyListeners(); | ||||
|   } | ||||
|  | ||||
|   Timer? _typingNotifyTimer; | ||||
|   bool _typingStatus = false; | ||||
|  | ||||
|   Future<void> _sendTypingStatusPackage() async { | ||||
|     _ws.conn?.sink.add(jsonEncode( | ||||
|       WebSocketPackage( | ||||
|         method: 'status.typing', | ||||
|         endpoint: 'im', | ||||
|         payload: { | ||||
|           'channel_id': channel!.id, | ||||
|         }, | ||||
|       ).toJson(), | ||||
|     )); | ||||
|   } | ||||
|  | ||||
|   void pingTypingStatus() { | ||||
|     if (!_typingStatus) { | ||||
|       _sendTypingStatusPackage(); | ||||
|       _typingStatus = true; | ||||
|     } | ||||
|  | ||||
|     if (_typingNotifyTimer == null || !_typingNotifyTimer!.isActive) { | ||||
|       _typingNotifyTimer?.cancel(); | ||||
|       _typingNotifyTimer = Timer(const Duration(milliseconds: 1850), () { | ||||
|         _typingStatus = false; | ||||
|       }); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   Future<void> _saveMessageToLocal(Iterable<SnChatMessage> messages) async { | ||||
|     if (_box == null) return; | ||||
|     await _box!.putAll({ | ||||
| @@ -167,8 +194,7 @@ class ChatMessageController extends ChangeNotifier { | ||||
|     switch (message.type) { | ||||
|       case 'messages.edit': | ||||
|         if (message.relatedEventId != null) { | ||||
|           final idx = | ||||
|               messages.indexWhere((x) => x.id == message.relatedEventId); | ||||
|           final idx = messages.indexWhere((x) => x.id == message.relatedEventId); | ||||
|           if (idx != -1) { | ||||
|             final newBody = message.body; | ||||
|             newBody.remove('related_event'); | ||||
| @@ -207,8 +233,7 @@ class ChatMessageController extends ChangeNotifier { | ||||
|       'algorithm': 'plain', | ||||
|       if (quoteId != null) 'quote_event': quoteId, | ||||
|       if (relatedId != null) 'related_event': relatedId, | ||||
|       if (attachments != null && attachments.isNotEmpty) | ||||
|         'attachments': attachments, | ||||
|       if (attachments != null && attachments.isNotEmpty) 'attachments': attachments, | ||||
|     }; | ||||
|  | ||||
|     // Mock the message locally | ||||
| @@ -305,8 +330,7 @@ class ChatMessageController extends ChangeNotifier { | ||||
|  | ||||
|     if (out == null) { | ||||
|       try { | ||||
|         final resp = await _sn.client | ||||
|             .get('/cgi/im/channels/${channel!.keyPath}/events/$id'); | ||||
|         final resp = await _sn.client.get('/cgi/im/channels/${channel!.keyPath}/events/$id'); | ||||
|         out = SnChatMessage.fromJson(resp.data); | ||||
|         _saveMessageToLocal([out]); | ||||
|       } catch (_) { | ||||
| @@ -341,9 +365,7 @@ class ChatMessageController extends ChangeNotifier { | ||||
|     bool forceRemote = false, | ||||
|   }) async { | ||||
|     late List<SnChatMessage> out; | ||||
|     if (_box != null && | ||||
|         (_box!.length >= take + offset || forceLocal) && | ||||
|         !forceRemote) { | ||||
|     if (_box != null && (_box!.length >= take + offset || forceLocal) && !forceRemote) { | ||||
|       out = _box!.keys | ||||
|           .toList() | ||||
|           .cast<int>() | ||||
| @@ -386,8 +408,7 @@ class ChatMessageController extends ChangeNotifier { | ||||
|           quoteEvent: quoteEvent, | ||||
|           attachments: attachments | ||||
|               .where( | ||||
|                 (ele) => | ||||
|                     out[i].body['attachments']?.contains(ele?.rid) ?? false, | ||||
|                 (ele) => out[i].body['attachments']?.contains(ele?.rid) ?? false, | ||||
|               ) | ||||
|               .toList(), | ||||
|         ), | ||||
| @@ -395,10 +416,7 @@ class ChatMessageController extends ChangeNotifier { | ||||
|     } | ||||
|  | ||||
|     // Preload sender accounts | ||||
|     final accountId = out | ||||
|         .where((ele) => ele.sender.accountId >= 0) | ||||
|         .map((ele) => ele.sender.accountId) | ||||
|         .toSet(); | ||||
|     final accountId = out.where((ele) => ele.sender.accountId >= 0).map((ele) => ele.sender.accountId).toSet(); | ||||
|     await _ud.listAccount(accountId); | ||||
|  | ||||
|     return out; | ||||
|   | ||||
| @@ -1,11 +1,17 @@ | ||||
| import 'dart:async'; | ||||
| import 'dart:convert'; | ||||
| import 'dart:developer'; | ||||
| import 'dart:io'; | ||||
| import 'dart:math' as math; | ||||
|  | ||||
| import 'package:dio/dio.dart'; | ||||
| import 'package:easy_localization/easy_localization.dart'; | ||||
| import 'package:flutter/foundation.dart'; | ||||
| import 'package:flutter/material.dart'; | ||||
| import 'package:image_picker/image_picker.dart'; | ||||
| import 'package:mime/mime.dart'; | ||||
| import 'package:provider/provider.dart'; | ||||
| import 'package:shared_preferences/shared_preferences.dart'; | ||||
| import 'package:surface/providers/post.dart'; | ||||
| import 'package:surface/providers/sn_attachment.dart'; | ||||
| import 'package:surface/providers/sn_network.dart'; | ||||
| @@ -13,17 +19,11 @@ import 'package:surface/types/attachment.dart'; | ||||
| import 'package:surface/types/post.dart'; | ||||
| import 'package:surface/widgets/dialog.dart'; | ||||
| import 'package:surface/widgets/universal_image.dart'; | ||||
|  | ||||
| enum PostWriteMediaType { | ||||
|   image, | ||||
|   video, | ||||
|   audio, | ||||
|   file, | ||||
| } | ||||
| import 'package:video_compress/video_compress.dart'; | ||||
|  | ||||
| class PostWriteMedia { | ||||
|   late String name; | ||||
|   late PostWriteMediaType type; | ||||
|   late SnMediaType type; | ||||
|   final SnAttachment? attachment; | ||||
|   final XFile? file; | ||||
|   final Uint8List? raw; | ||||
| @@ -35,16 +35,16 @@ class PostWriteMedia { | ||||
|  | ||||
|     switch (attachment?.mimetype.split('/').firstOrNull) { | ||||
|       case 'image': | ||||
|         type = PostWriteMediaType.image; | ||||
|         type = SnMediaType.image; | ||||
|         break; | ||||
|       case 'video': | ||||
|         type = PostWriteMediaType.video; | ||||
|         type = SnMediaType.video; | ||||
|         break; | ||||
|       case 'audio': | ||||
|         type = PostWriteMediaType.audio; | ||||
|         type = SnMediaType.audio; | ||||
|         break; | ||||
|       default: | ||||
|         type = PostWriteMediaType.file; | ||||
|         type = SnMediaType.file; | ||||
|     } | ||||
|   } | ||||
|  | ||||
| @@ -56,16 +56,16 @@ class PostWriteMedia { | ||||
|  | ||||
|     switch (mimetype?.split('/').firstOrNull) { | ||||
|       case 'image': | ||||
|         type = PostWriteMediaType.image; | ||||
|         type = SnMediaType.image; | ||||
|         break; | ||||
|       case 'video': | ||||
|         type = PostWriteMediaType.video; | ||||
|         type = SnMediaType.video; | ||||
|         break; | ||||
|       case 'audio': | ||||
|         type = PostWriteMediaType.audio; | ||||
|         type = SnMediaType.audio; | ||||
|         break; | ||||
|       default: | ||||
|         type = PostWriteMediaType.file; | ||||
|         type = SnMediaType.file; | ||||
|     } | ||||
|   } | ||||
|  | ||||
| @@ -104,7 +104,7 @@ class PostWriteMedia { | ||||
|     if (attachment != null) { | ||||
|       final sn = context.read<SnNetworkProvider>(); | ||||
|       final ImageProvider provider = UniversalImage.provider(sn.getAttachmentUrl(attachment!.rid)); | ||||
|       if (width != null && height != null) { | ||||
|       if (width != null && height != null && !kIsWeb) { | ||||
|         return ResizeImage( | ||||
|           provider, | ||||
|           width: width, | ||||
| @@ -152,10 +152,24 @@ class PostWriteController extends ChangeNotifier { | ||||
|   final TextEditingController contentController = TextEditingController(); | ||||
|   final TextEditingController titleController = TextEditingController(); | ||||
|   final TextEditingController descriptionController = TextEditingController(); | ||||
|   final TextEditingController aliasController = TextEditingController(); | ||||
|  | ||||
|   PostWriteController() { | ||||
|     titleController.addListener(() => notifyListeners()); | ||||
|     descriptionController.addListener(() => notifyListeners()); | ||||
|   bool _temporarySaveActive = false; | ||||
|  | ||||
|   PostWriteController({bool doLoadFromTemporary = true}) { | ||||
|     _temporarySaveActive = doLoadFromTemporary; | ||||
|     titleController.addListener(() { | ||||
|       _temporaryPlanSave(); | ||||
|       notifyListeners(); | ||||
|     }); | ||||
|     descriptionController.addListener(() { | ||||
|       _temporaryPlanSave(); | ||||
|       notifyListeners(); | ||||
|     }); | ||||
|     contentController.addListener(() { | ||||
|       _temporaryPlanSave(); | ||||
|     }); | ||||
|     if (doLoadFromTemporary) _temporaryLoad(); | ||||
|   } | ||||
|  | ||||
|   String mode = kTitleMap.keys.first; | ||||
| @@ -176,6 +190,7 @@ class PostWriteController extends ChangeNotifier { | ||||
|   List<int> visibleUsers = List.empty(); | ||||
|   List<int> invisibleUsers = List.empty(); | ||||
|   List<String> tags = List.empty(); | ||||
|   List<String> categories = List.empty(); | ||||
|   PostWriteMedia? thumbnail; | ||||
|   List<PostWriteMedia> attachments = List.empty(growable: true); | ||||
|   DateTime? publishedAt, publishedUntil; | ||||
| @@ -198,12 +213,14 @@ class PostWriteController extends ChangeNotifier { | ||||
|         titleController.text = post.body['title'] ?? ''; | ||||
|         descriptionController.text = post.body['description'] ?? ''; | ||||
|         contentController.text = post.body['content'] ?? ''; | ||||
|         aliasController.text = post.alias ?? ''; | ||||
|         publishedAt = post.publishedAt; | ||||
|         publishedUntil = post.publishedUntil; | ||||
|         visibleUsers = List.from(post.visibleUsersList ?? []); | ||||
|         invisibleUsers = List.from(post.invisibleUsersList ?? []); | ||||
|         visibleUsers = List.from(post.visibleUsersList ?? [], growable: true); | ||||
|         invisibleUsers = List.from(post.invisibleUsersList ?? [], growable: true); | ||||
|         visibility = post.visibility; | ||||
|         tags = List.from(post.tags.map((ele) => ele.alias)); | ||||
|         tags = List.from(post.tags.map((ele) => ele.alias), growable: true); | ||||
|         categories = List.from(post.categories.map((ele) => ele.alias), growable: true); | ||||
|         attachments.addAll(post.preload?.attachments?.map((ele) => PostWriteMedia(ele)) ?? []); | ||||
|  | ||||
|         if (post.preload?.thumbnail != null && (post.preload?.thumbnail?.rid.isNotEmpty ?? false)) { | ||||
| @@ -231,7 +248,8 @@ class PostWriteController extends ChangeNotifier { | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   Future<SnAttachment> _uploadAttachment(BuildContext context, PostWriteMedia media) async { | ||||
|   Future<SnAttachment> _uploadAttachment(BuildContext context, PostWriteMedia media, | ||||
|       {bool isCompressed = false}) async { | ||||
|     final attach = context.read<SnAttachmentProvider>(); | ||||
|  | ||||
|     final place = await attach.chunkedUploadInitialize( | ||||
| @@ -239,22 +257,141 @@ class PostWriteController extends ChangeNotifier { | ||||
|       media.name, | ||||
|       'interactive', | ||||
|       null, | ||||
|       mimetype: media.raw != null && media.type == PostWriteMediaType.image ? 'image/png' : null, | ||||
|       mimetype: media.raw != null && media.type == SnMediaType.image ? 'image/png' : null, | ||||
|     ); | ||||
|  | ||||
|     final item = await attach.chunkedUploadParts( | ||||
|     var item = await attach.chunkedUploadParts( | ||||
|       media.toFile()!, | ||||
|       place.$1, | ||||
|       place.$2, | ||||
|       onProgress: (progress) { | ||||
|         progress = progress; | ||||
|       analyzeNow: media.type == SnMediaType.image, | ||||
|       onProgress: (value) { | ||||
|         progress = value; | ||||
|         notifyListeners(); | ||||
|       }, | ||||
|     ); | ||||
|  | ||||
|     if (media.type == SnMediaType.video && !isCompressed && context.mounted) { | ||||
|       try { | ||||
|         final compressedAttachment = await _tryCompressVideoCopy(context, media); | ||||
|         if (compressedAttachment != null) { | ||||
|           item = await attach.updateOne(item, compressedId: compressedAttachment.id); | ||||
|         } | ||||
|       } catch (err) { | ||||
|         if (context.mounted) context.showErrorDialog(err); | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     return item; | ||||
|   } | ||||
|  | ||||
|   Future<SnAttachment?> _tryCompressVideoCopy(BuildContext context, PostWriteMedia media) async { | ||||
|     if (kIsWeb || !(Platform.isAndroid || Platform.isIOS || Platform.isMacOS)) return null; | ||||
|     if (media.type != SnMediaType.video) return null; | ||||
|     if (media.file == null) return null; | ||||
|     if (VideoCompress.isCompressing) return null; | ||||
|  | ||||
|     final confirm = await context.showConfirmDialog( | ||||
|       'attachmentVideoCompressHint'.tr(), | ||||
|       'attachmentVideoCompressHintDescription'.tr(args: [media.file!.name]), | ||||
|     ); | ||||
|     if (!confirm) return null; | ||||
|  | ||||
|     progress = null; | ||||
|     notifyListeners(); | ||||
|  | ||||
|     final mediaInfo = await VideoCompress.compressVideo( | ||||
|       media.file!.path, | ||||
|       quality: VideoQuality.LowQuality, | ||||
|       frameRate: 30, | ||||
|       deleteOrigin: false, | ||||
|     ); | ||||
|     if (mediaInfo == null) return null; | ||||
|     if (!context.mounted) return null; | ||||
|  | ||||
|     final compressedMedia = PostWriteMedia.fromFile(XFile(mediaInfo.path!)); | ||||
|     final compressedAttachment = await _uploadAttachment(context, compressedMedia, isCompressed: true); | ||||
|  | ||||
|     return compressedAttachment; | ||||
|   } | ||||
|  | ||||
|   static const kTemporaryStorageKey = 'int_draft_post'; | ||||
|  | ||||
|   Timer? _temporarySaveTimer; | ||||
|  | ||||
|   void _temporaryPlanSave() { | ||||
|     if (!_temporarySaveActive) return; | ||||
|     _temporarySaveTimer?.cancel(); | ||||
|     _temporarySaveTimer = Timer(const Duration(seconds: 1), () { | ||||
|       _temporarySave(); | ||||
|       log("[PostWriter] Temporary save saved."); | ||||
|     }); | ||||
|   } | ||||
|  | ||||
|   void _temporarySave() { | ||||
|     SharedPreferences.getInstance().then((prefs) { | ||||
|       if (titleController.text.isEmpty && | ||||
|           descriptionController.text.isEmpty && | ||||
|           contentController.text.isEmpty && | ||||
|           thumbnail == null && | ||||
|           attachments.isEmpty) { | ||||
|         prefs.remove(kTemporaryStorageKey); | ||||
|         return; | ||||
|       } | ||||
|  | ||||
|       prefs.setString( | ||||
|         kTemporaryStorageKey, | ||||
|         jsonEncode({ | ||||
|           'publisher': publisher, | ||||
|           'content': contentController.text, | ||||
|           if (aliasController.text.isNotEmpty) 'alias': aliasController.text, | ||||
|           if (titleController.text.isNotEmpty) 'title': titleController.text, | ||||
|           if (descriptionController.text.isNotEmpty) 'description': descriptionController.text, | ||||
|           if (thumbnail != null && thumbnail!.attachment != null) 'thumbnail': thumbnail!.attachment!.toJson(), | ||||
|           'attachments': | ||||
|               attachments.where((e) => e.attachment != null).map((e) => e.attachment!.toJson()).toList(growable: true), | ||||
|           'tags': tags.map((ele) => {'alias': ele}).toList(growable: true), | ||||
|           'categories': categories.map((ele) => {'alias': ele}).toList(growable: true), | ||||
|           'visibility': visibility, | ||||
|           'visible_users_list': visibleUsers, | ||||
|           'invisible_users_list': invisibleUsers, | ||||
|           if (publishedAt != null) 'published_at': publishedAt!.toUtc().toIso8601String(), | ||||
|           if (publishedUntil != null) 'published_until': publishedAt!.toUtc().toIso8601String(), | ||||
|           if (replyingPost != null) 'reply_to': replyingPost!.toJson(), | ||||
|           if (repostingPost != null) 'repost_to': repostingPost!.toJson(), | ||||
|         }), | ||||
|       ); | ||||
|     }); | ||||
|   } | ||||
|  | ||||
|   bool temporaryRestored = false; | ||||
|  | ||||
|   void _temporaryLoad() { | ||||
|     SharedPreferences.getInstance().then((prefs) { | ||||
|       final raw = prefs.getString(kTemporaryStorageKey); | ||||
|       if (raw == null) return; | ||||
|       final data = jsonDecode(raw); | ||||
|       contentController.text = data['content']; | ||||
|       aliasController.text = data['alias'] ?? ''; | ||||
|       titleController.text = data['title'] ?? ''; | ||||
|       descriptionController.text = data['description'] ?? ''; | ||||
|       if (data['thumbnail'] != null) thumbnail = PostWriteMedia(SnAttachment.fromJson(data['thumbnail'])); | ||||
|       attachments | ||||
|           .addAll(data['attachments'].map((ele) => PostWriteMedia(SnAttachment.fromJson(ele))).cast<PostWriteMedia>()); | ||||
|       tags = List.from(data['tags'].map((ele) => ele['alias'])); | ||||
|       categories = List.from(data['categories'].map((ele) => ele['alias'])); | ||||
|       visibility = data['visibility']; | ||||
|       visibleUsers = List.from(data['visible_users_list'] ?? []); | ||||
|       invisibleUsers = List.from(data['invisible_users_list'] ?? []); | ||||
|       if (data['published_at'] != null) publishedAt = DateTime.tryParse(data['published_at'])?.toLocal(); | ||||
|       if (data['published_until'] != null) publishedUntil = DateTime.tryParse(data['published_until'])?.toLocal(); | ||||
|       replyingPost = data['reply_to'] != null ? SnPost.fromJson(data['reply_to']) : null; | ||||
|       repostingPost = data['repost_to'] != null ? SnPost.fromJson(data['repost_to']) : null; | ||||
|       temporaryRestored = true; | ||||
|       notifyListeners(); | ||||
|     }); | ||||
|   } | ||||
|  | ||||
|   Future<void> uploadSingleAttachment(BuildContext context, int idx) async { | ||||
|     if (isBusy) return; | ||||
|  | ||||
| @@ -269,7 +406,7 @@ class PostWriteController extends ChangeNotifier { | ||||
|     notifyListeners(); | ||||
|   } | ||||
|  | ||||
|   Future<void> post(BuildContext context) async { | ||||
|   Future<void> sendPost(BuildContext context) async { | ||||
|     if (isBusy || publisher == null) return; | ||||
|  | ||||
|     final sn = context.read<SnNetworkProvider>(); | ||||
| @@ -296,21 +433,34 @@ class PostWriteController extends ChangeNotifier { | ||||
|           media.name, | ||||
|           'interactive', | ||||
|           null, | ||||
|           mimetype: media.raw != null && media.type == PostWriteMediaType.image ? 'image/png' : null, | ||||
|           mimetype: media.raw != null && media.type == SnMediaType.image ? 'image/png' : null, | ||||
|         ); | ||||
|  | ||||
|         final item = await attach.chunkedUploadParts( | ||||
|         var item = await attach.chunkedUploadParts( | ||||
|           media.toFile()!, | ||||
|           place.$1, | ||||
|           place.$2, | ||||
|           onProgress: (progress) { | ||||
|           onProgress: (value) { | ||||
|             // Calculate overall progress for attachments | ||||
|             progress = ((i + progress) / attachments.length) * kAttachmentProgressWeight; | ||||
|             progress = math.max(((i + value) / attachments.length) * kAttachmentProgressWeight, value); | ||||
|             notifyListeners(); | ||||
|           }, | ||||
|         ); | ||||
|  | ||||
|         try { | ||||
|           if (context.mounted) { | ||||
|             final compressedAttachment = await _tryCompressVideoCopy(context, media); | ||||
|             if (compressedAttachment != null) { | ||||
|               item = await attach.updateOne(item, compressedId: compressedAttachment.id); | ||||
|             } | ||||
|           } | ||||
|         } catch (err) { | ||||
|           if (context.mounted) context.showErrorDialog(err); | ||||
|         } | ||||
|  | ||||
|         progress = (i + 1) / attachments.length * kAttachmentProgressWeight; | ||||
|         attachments[i] = PostWriteMedia(item); | ||||
|         notifyListeners(); | ||||
|       } | ||||
|     } catch (err) { | ||||
|       isBusy = false; | ||||
| @@ -334,11 +484,13 @@ class PostWriteController extends ChangeNotifier { | ||||
|         data: { | ||||
|           'publisher': publisher!.id, | ||||
|           'content': contentController.text, | ||||
|           if (aliasController.text.isNotEmpty) 'alias': aliasController.text, | ||||
|           if (titleController.text.isNotEmpty) 'title': titleController.text, | ||||
|           if (descriptionController.text.isNotEmpty) 'description': descriptionController.text, | ||||
|           if (thumbnail != null && thumbnail!.attachment != null) 'thumbnail': thumbnail!.attachment!.rid, | ||||
|           'attachments': attachments.where((e) => e.attachment != null).map((e) => e.attachment!.rid).toList(), | ||||
|           'tags': tags.map((ele) => {'alias': ele}).toList(), | ||||
|           'categories': categories.map((ele) => {'alias': ele}).toList(), | ||||
|           'visibility': visibility, | ||||
|           'visible_users_list': visibleUsers, | ||||
|           'invisible_users_list': invisibleUsers, | ||||
| @@ -359,6 +511,7 @@ class PostWriteController extends ChangeNotifier { | ||||
|           method: editingPost != null ? 'PUT' : 'POST', | ||||
|         ), | ||||
|       ); | ||||
|       reset(); | ||||
|     } catch (err) { | ||||
|       if (!context.mounted) return; | ||||
|       context.showErrorDialog(err); | ||||
| @@ -407,65 +560,88 @@ class PostWriteController extends ChangeNotifier { | ||||
|  | ||||
|   void setPublisher(SnPublisher? item) { | ||||
|     publisher = item; | ||||
|     _temporaryPlanSave(); | ||||
|     notifyListeners(); | ||||
|   } | ||||
|  | ||||
|   void setPublishedAt(DateTime? value) { | ||||
|     publishedAt = value; | ||||
|     _temporaryPlanSave(); | ||||
|     notifyListeners(); | ||||
|   } | ||||
|  | ||||
|   void setPublishedUntil(DateTime? value) { | ||||
|     publishedUntil = value; | ||||
|     _temporaryPlanSave(); | ||||
|     notifyListeners(); | ||||
|   } | ||||
|  | ||||
|   void setTags(List<String> value) { | ||||
|     tags = value; | ||||
|     _temporaryPlanSave(); | ||||
|     notifyListeners(); | ||||
|   } | ||||
|  | ||||
|   void setCategories(List<String> value) { | ||||
|     categories = value; | ||||
|     _temporaryPlanSave(); | ||||
|     notifyListeners(); | ||||
|   } | ||||
|  | ||||
|   void setVisibility(int value) { | ||||
|     visibility = value; | ||||
|     _temporaryPlanSave(); | ||||
|     notifyListeners(); | ||||
|   } | ||||
|  | ||||
|   void setVisibleUsers(List<int> value) { | ||||
|     visibleUsers = value; | ||||
|     _temporaryPlanSave(); | ||||
|     notifyListeners(); | ||||
|   } | ||||
|  | ||||
|   void setInvisibleUsers(List<int> value) { | ||||
|     invisibleUsers = value; | ||||
|     _temporaryPlanSave(); | ||||
|     notifyListeners(); | ||||
|   } | ||||
|  | ||||
|   void setProgress(double? value) { | ||||
|     progress = value; | ||||
|     _temporaryPlanSave(); | ||||
|     notifyListeners(); | ||||
|   } | ||||
|  | ||||
|   void setIsBusy(bool value) { | ||||
|     isBusy = value; | ||||
|     _temporaryPlanSave(); | ||||
|     notifyListeners(); | ||||
|   } | ||||
|  | ||||
|   void setMode(String value) { | ||||
|     mode = value; | ||||
|     _temporaryPlanSave(); | ||||
|     notifyListeners(); | ||||
|   } | ||||
|  | ||||
|   void reset() { | ||||
|     publishedAt = null; | ||||
|     publishedUntil = null; | ||||
|     thumbnail = null; | ||||
|     visibility = 0; | ||||
|     titleController.clear(); | ||||
|     descriptionController.clear(); | ||||
|     contentController.clear(); | ||||
|     attachments.clear(); | ||||
|     aliasController.clear(); | ||||
|     tags = List.empty(growable: true); | ||||
|     categories = List.empty(growable: true); | ||||
|     attachments = List.empty(growable: true); | ||||
|     editingPost = null; | ||||
|     replyingPost = null; | ||||
|     repostingPost = null; | ||||
|     mode = kTitleMap.keys.first; | ||||
|     temporaryRestored = false; | ||||
|     SharedPreferences.getInstance().then((prefs) => prefs.remove(kTemporaryStorageKey)); | ||||
|     notifyListeners(); | ||||
|   } | ||||
|  | ||||
| @@ -474,6 +650,7 @@ class PostWriteController extends ChangeNotifier { | ||||
|     contentController.dispose(); | ||||
|     titleController.dispose(); | ||||
|     descriptionController.dispose(); | ||||
|     aliasController.dispose(); | ||||
|     super.dispose(); | ||||
|   } | ||||
| } | ||||
|   | ||||
							
								
								
									
										164
									
								
								lib/main.dart
									
									
									
									
									
								
							
							
						
						| @@ -1,6 +1,7 @@ | ||||
| import 'dart:async'; | ||||
| import 'dart:developer'; | ||||
| import 'dart:io'; | ||||
| import 'dart:ui'; | ||||
|  | ||||
| import 'package:bitsdojo_window/bitsdojo_window.dart'; | ||||
| import 'package:croppy/croppy.dart'; | ||||
| @@ -10,15 +11,15 @@ import 'package:easy_localization_loader/easy_localization_loader.dart'; | ||||
| import 'package:firebase_core/firebase_core.dart'; | ||||
| import 'package:flutter/foundation.dart'; | ||||
| import 'package:flutter/material.dart'; | ||||
| import 'package:gap/gap.dart'; | ||||
| import 'package:flutter/services.dart'; | ||||
| import 'package:go_router/go_router.dart'; | ||||
| import 'package:hive_flutter/hive_flutter.dart'; | ||||
| import 'package:hotkey_manager/hotkey_manager.dart'; | ||||
| import 'package:package_info_plus/package_info_plus.dart'; | ||||
| import 'package:provider/provider.dart'; | ||||
| import 'package:relative_time/relative_time.dart'; | ||||
| import 'package:responsive_framework/responsive_framework.dart'; | ||||
| import 'package:shared_preferences/shared_preferences.dart'; | ||||
| import 'package:styled_widget/styled_widget.dart'; | ||||
| import 'package:surface/firebase_options.dart'; | ||||
| import 'package:surface/providers/channel.dart'; | ||||
| import 'package:surface/providers/chat_call.dart'; | ||||
| @@ -30,6 +31,8 @@ import 'package:surface/providers/post.dart'; | ||||
| import 'package:surface/providers/relationship.dart'; | ||||
| import 'package:surface/providers/sn_attachment.dart'; | ||||
| import 'package:surface/providers/sn_network.dart'; | ||||
| import 'package:surface/providers/sn_sticker.dart'; | ||||
| import 'package:surface/providers/special_day.dart'; | ||||
| import 'package:surface/providers/theme.dart'; | ||||
| import 'package:surface/providers/user_directory.dart'; | ||||
| import 'package:surface/providers/userinfo.dart'; | ||||
| @@ -40,7 +43,7 @@ import 'package:surface/types/chat.dart'; | ||||
| import 'package:surface/types/realm.dart'; | ||||
| import 'package:flutter_web_plugins/url_strategy.dart' show usePathUrlStrategy; | ||||
| import 'package:surface/widgets/dialog.dart'; | ||||
| import 'package:surface/widgets/version_label.dart'; | ||||
| import 'package:tray_manager/tray_manager.dart'; | ||||
| import 'package:version/version.dart'; | ||||
| import 'package:workmanager/workmanager.dart'; | ||||
| import 'package:in_app_review/in_app_review.dart'; | ||||
| @@ -143,11 +146,15 @@ class SolianApp extends StatelessWidget { | ||||
|             Provider(create: (ctx) => SnPostContentProvider(ctx)), | ||||
|             Provider(create: (ctx) => SnRelationshipProvider(ctx)), | ||||
|             Provider(create: (ctx) => SnLinkPreviewProvider(ctx)), | ||||
|             Provider(create: (ctx) => SnStickerProvider(ctx)), | ||||
|             ChangeNotifierProvider(create: (ctx) => UserProvider(ctx)), | ||||
|             ChangeNotifierProvider(create: (ctx) => WebSocketProvider(ctx)), | ||||
|             ChangeNotifierProvider(create: (ctx) => NotificationProvider(ctx)), | ||||
|             ChangeNotifierProvider(create: (ctx) => ChatChannelProvider(ctx)), | ||||
|             ChangeNotifierProvider(create: (ctx) => ChatCallProvider(ctx)), | ||||
|  | ||||
|             // Additional helper layer | ||||
|             Provider(create: (ctx) => SpecialDayProvider(ctx)), | ||||
|           ], | ||||
|           child: _AppDelegate(), | ||||
|         ), | ||||
| @@ -203,9 +210,7 @@ class _AppSplashScreen extends StatefulWidget { | ||||
|   State<_AppSplashScreen> createState() => _AppSplashScreenState(); | ||||
| } | ||||
|  | ||||
| class _AppSplashScreenState extends State<_AppSplashScreen> { | ||||
|   bool _isReady = false; | ||||
|  | ||||
| class _AppSplashScreenState extends State<_AppSplashScreen> with TrayListener { | ||||
|   void _tryRequestRating() async { | ||||
|     final prefs = await SharedPreferences.getInstance(); | ||||
|     if (prefs.containsKey('first_boot_time')) { | ||||
| @@ -257,6 +262,10 @@ class _AppSplashScreenState extends State<_AppSplashScreen> { | ||||
|  | ||||
|   Future<void> _initialize() async { | ||||
|     try { | ||||
|       final cfg = context.read<ConfigProvider>(); | ||||
|       WidgetsBinding.instance.addPostFrameCallback((_) { | ||||
|         cfg.calcDrawerSize(context); | ||||
|       }); | ||||
|       final home = context.read<HomeWidgetProvider>(); | ||||
|       await home.initialize(); | ||||
|       if (!mounted) return; | ||||
| @@ -265,6 +274,7 @@ class _AppSplashScreenState extends State<_AppSplashScreen> { | ||||
|       // The Network initialization will also save initialize the Config, so it not need to be initialized again | ||||
|       final sn = context.read<SnNetworkProvider>(); | ||||
|       await sn.initializeUserAgent(); | ||||
|       await sn.setConfigWithNative(); | ||||
|       if (!mounted) return; | ||||
|       final ua = context.read<UserProvider>(); | ||||
|       await ua.initialize(); | ||||
| @@ -273,12 +283,14 @@ class _AppSplashScreenState extends State<_AppSplashScreen> { | ||||
|       await ws.tryConnect(); | ||||
|       if (!mounted) return; | ||||
|       final notify = context.read<NotificationProvider>(); | ||||
|       notify.listen(); | ||||
|       await notify.registerPushNotifications(); | ||||
|       if (!mounted) return; | ||||
|       final sticker = context.read<SnStickerProvider>(); | ||||
|       await sticker.listStickerEagerly(); | ||||
|     } catch (err) { | ||||
|       if (!mounted) return; | ||||
|       await context.showErrorDialog(err); | ||||
|     } finally { | ||||
|       setState(() => _isReady = true); | ||||
|     } | ||||
|   } | ||||
|  | ||||
| @@ -286,9 +298,62 @@ class _AppSplashScreenState extends State<_AppSplashScreen> { | ||||
|     await widgetUpdateRandomPost(); | ||||
|   } | ||||
|  | ||||
|   Future<void> _hotkeyInitialization() async { | ||||
|     if (kIsWeb) return; | ||||
|  | ||||
|     if (Platform.isMacOS) { | ||||
|       HotKey quitHotKey = HotKey( | ||||
|         key: PhysicalKeyboardKey.keyQ, | ||||
|         modifiers: [HotKeyModifier.meta], | ||||
|         scope: HotKeyScope.inapp, | ||||
|       ); | ||||
|       await hotKeyManager.register(quitHotKey, keyUpHandler: (_) { | ||||
|         _appLifecycleListener?.dispose(); | ||||
|         SystemChannels.platform.invokeMethod('SystemNavigator.pop'); | ||||
|       }); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   Future<void> _trayInitialization() async { | ||||
|     if (kIsWeb || Platform.isAndroid || Platform.isIOS) return; | ||||
|  | ||||
|     final icon = Platform.isWindows ? 'assets/icon/tray-icon.ico' : 'assets/icon/tray-icon.png'; | ||||
|     final appVersion = await PackageInfo.fromPlatform(); | ||||
|  | ||||
|     trayManager.addListener(this); | ||||
|     await trayManager.setIcon(icon); | ||||
|  | ||||
|     Menu menu = Menu( | ||||
|       items: [ | ||||
|         MenuItem( | ||||
|           key: 'version_label', | ||||
|           label: 'Solian ${appVersion.version}+${appVersion.buildNumber}', | ||||
|           disabled: true, | ||||
|         ), | ||||
|         MenuItem.separator(), | ||||
|         MenuItem( | ||||
|           key: 'exit', | ||||
|           label: 'trayMenuExit'.tr(), | ||||
|         ), | ||||
|       ], | ||||
|     ); | ||||
|     await trayManager.setContextMenu(menu); | ||||
|   } | ||||
|  | ||||
|   AppLifecycleListener? _appLifecycleListener; | ||||
|  | ||||
|   @override | ||||
|   void initState() { | ||||
|     super.initState(); | ||||
|  | ||||
|     if (!kIsWeb && !(Platform.isIOS || Platform.isAndroid)) { | ||||
|       _appLifecycleListener = AppLifecycleListener( | ||||
|         onExitRequested: _onExitRequested, | ||||
|       ); | ||||
|     } | ||||
|  | ||||
|     _trayInitialization(); | ||||
|     _hotkeyInitialization(); | ||||
|     _initialize().then((_) { | ||||
|       _postInitialization(); | ||||
|       _tryRequestRating(); | ||||
| @@ -296,34 +361,63 @@ class _AppSplashScreenState extends State<_AppSplashScreen> { | ||||
|     }); | ||||
|   } | ||||
|  | ||||
|   Future<AppExitResponse> _onExitRequested() async { | ||||
|     appWindow.hide(); | ||||
|     return AppExitResponse.cancel; | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   void onTrayIconMouseDown() { | ||||
|     if (Platform.isWindows) { | ||||
|       context.read<NotificationProvider>().clearTray(); | ||||
|       appWindow.show(); | ||||
|     } else { | ||||
|       trayManager.popUpContextMenu(); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   void onTrayIconRightMouseDown() { | ||||
|     if (Platform.isWindows) { | ||||
|       trayManager.popUpContextMenu(); | ||||
|     } else { | ||||
|       context.read<NotificationProvider>().clearTray(); | ||||
|       appWindow.show(); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   void onTrayMenuItemClick(MenuItem menuItem) { | ||||
|     switch (menuItem.key) { | ||||
|       case 'exit': | ||||
|         _appLifecycleListener?.dispose(); | ||||
|         SystemChannels.platform.invokeMethod('SystemNavigator.pop'); | ||||
|         break; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   void dispose() { | ||||
|     if (!kIsWeb && !(Platform.isAndroid || Platform.isIOS)) { | ||||
|       trayManager.removeListener(this); | ||||
|       hotKeyManager.unregisterAll(); | ||||
|     } | ||||
|     super.dispose(); | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     if (!_isReady) { | ||||
|       return Scaffold( | ||||
|         backgroundColor: Theme.of(context).colorScheme.surface, | ||||
|         body: Container( | ||||
|           constraints: const BoxConstraints(maxWidth: 180), | ||||
|           child: Column( | ||||
|             mainAxisAlignment: MainAxisAlignment.center, | ||||
|             mainAxisSize: MainAxisSize.min, | ||||
|             children: [ | ||||
|               if (MediaQuery.of(context).platformBrightness == Brightness.dark) | ||||
|                 Image.asset("assets/icon/icon-dark.png", width: 64, height: 64) | ||||
|               else | ||||
|                 Image.asset("assets/icon/icon.png", width: 64, height: 64), | ||||
|               const Gap(6), | ||||
|               LinearProgressIndicator( | ||||
|                 backgroundColor: Theme.of(context).colorScheme.surfaceContainer, | ||||
|               ), | ||||
|               const Gap(20), | ||||
|               Text('appInitializing'.tr(), textAlign: TextAlign.center), | ||||
|               AppVersionLabel(), | ||||
|             ], | ||||
|           ), | ||||
|         ).center(), | ||||
|       ); | ||||
|     } | ||||
|  | ||||
|     return widget.child; | ||||
|     final cfg = context.read<ConfigProvider>(); | ||||
|     return NotificationListener<SizeChangedLayoutNotification>( | ||||
|       onNotification: (notification) { | ||||
|         WidgetsBinding.instance.addPostFrameCallback((_) { | ||||
|           cfg.calcDrawerSize(context); | ||||
|         }); | ||||
|         return false; | ||||
|       }, | ||||
|       child: SizeChangedLayoutNotifier( | ||||
|         child: widget.child, | ||||
|       ), | ||||
|     ); | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -1,5 +1,6 @@ | ||||
| import 'package:flutter/material.dart'; | ||||
| import 'package:provider/provider.dart'; | ||||
| import 'package:responsive_framework/responsive_framework.dart'; | ||||
| import 'package:shared_preferences/shared_preferences.dart'; | ||||
| import 'package:surface/providers/widget.dart'; | ||||
|  | ||||
| @@ -9,6 +10,14 @@ const kRtkStoreKey = 'nex_user_rtk'; | ||||
| const kNetworkServerDefault = 'https://api.sn.solsynth.dev'; | ||||
| const kNetworkServerStoreKey = 'app_server_url'; | ||||
|  | ||||
| const kAppbarTransparentStoreKey = 'app_bar_transparent'; | ||||
| const kAppBackgroundStoreKey = 'app_has_background'; | ||||
| const kAppColorSchemeStoreKey = 'app_color_scheme'; | ||||
| const kAppDrawerPreferCollapse = 'app_drawer_prefer_collapse'; | ||||
| const kAppNotifyWithHaptic = 'app_notify_with_haptic'; | ||||
| const kAppExpandPostLink = 'app_expand_post_link'; | ||||
| const kAppExpandChatLink = 'app_expand_chat_link'; | ||||
|  | ||||
| const Map<String, FilterQuality> kImageQualityLevel = { | ||||
|   'settingsImageQualityLowest': FilterQuality.none, | ||||
|   'settingsImageQualityLow': FilterQuality.low, | ||||
| @@ -29,6 +38,32 @@ class ConfigProvider extends ChangeNotifier { | ||||
|     prefs = await SharedPreferences.getInstance(); | ||||
|   } | ||||
|  | ||||
|   bool drawerIsCollapsed = false; | ||||
|   bool drawerIsExpanded = false; | ||||
|  | ||||
|   void calcDrawerSize(BuildContext context, {bool withMediaQuery = false}) { | ||||
|     bool newDrawerIsCollapsed = false; | ||||
|     bool newDrawerIsExpanded = false; | ||||
|     if (withMediaQuery) { | ||||
|       newDrawerIsCollapsed = MediaQuery.of(context).size.width < 450; | ||||
|       newDrawerIsExpanded = MediaQuery.of(context).size.width >= 451; | ||||
|     } else { | ||||
|       final rpb = ResponsiveBreakpoints.of(context); | ||||
|       newDrawerIsCollapsed = rpb.smallerOrEqualTo(MOBILE); | ||||
|       newDrawerIsExpanded = rpb.largerThan(TABLET) | ||||
|           ? (prefs.getBool(kAppDrawerPreferCollapse) ?? false) | ||||
|               ? false | ||||
|               : true | ||||
|           : false; | ||||
|     } | ||||
|  | ||||
|     if (newDrawerIsExpanded != drawerIsExpanded || newDrawerIsCollapsed != drawerIsCollapsed) { | ||||
|       drawerIsExpanded = newDrawerIsExpanded; | ||||
|       drawerIsCollapsed = newDrawerIsCollapsed; | ||||
|       notifyListeners(); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   FilterQuality get imageQuality { | ||||
|     return kImageQualityLevel.values.elementAtOrNull(prefs.getInt('app_image_quality') ?? 3) ?? FilterQuality.high; | ||||
|   } | ||||
|   | ||||
							
								
								
									
										41
									
								
								lib/providers/experience.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,41 @@ | ||||
| import 'package:intl/intl.dart'; | ||||
|  | ||||
| const List<int> kExperienceToLevelRequirements = [ | ||||
|   0, // Level 0 | ||||
|   1000, // Level 1 | ||||
|   4000, // Level 2 | ||||
|   9000, // Level 3 | ||||
|   16000, // Level 4 | ||||
|   25000, // Level 5 | ||||
|   36000, // Level 6 | ||||
|   49000, // Level 7 | ||||
|   64000, // Level 8 | ||||
|   81000, // Level 9 | ||||
|   100000, // Level 10 | ||||
|   121000, // Level 11 | ||||
|   144000, // Level 12 | ||||
|   368000 // Level 13 | ||||
| ]; | ||||
|  | ||||
| int getLevelFromExp(int experience) { | ||||
|   final exp = kExperienceToLevelRequirements.reversed.firstWhere((x) => x <= experience); | ||||
|   final idx = kExperienceToLevelRequirements.indexOf(exp); | ||||
|   return idx; | ||||
| } | ||||
|  | ||||
| double calcLevelUpProgress(int experience) { | ||||
|   final exp = kExperienceToLevelRequirements.reversed.firstWhere((x) => x <= experience); | ||||
|   final idx = kExperienceToLevelRequirements.indexOf(exp); | ||||
|   if (idx + 1 >= kExperienceToLevelRequirements.length) return 1; | ||||
|   final nextExp = kExperienceToLevelRequirements[idx + 1]; | ||||
|   return (experience - exp).abs() / (exp - nextExp).abs(); | ||||
| } | ||||
|  | ||||
| String calcLevelUpProgressLevel(int experience) { | ||||
|   final exp = kExperienceToLevelRequirements.reversed.firstWhere((x) => x <= experience); | ||||
|   final idx = kExperienceToLevelRequirements.indexOf(exp); | ||||
|   if (idx + 1 >= kExperienceToLevelRequirements.length) return 'Infinity'; | ||||
|   final nextExp = exp - kExperienceToLevelRequirements[idx + 1]; | ||||
|   final formatter = NumberFormat.compactCurrency(symbol: '', decimalDigits: 1); | ||||
|   return '${formatter.format((exp - experience).abs())}/${formatter.format(nextExp.abs())}'; | ||||
| } | ||||
| @@ -58,6 +58,11 @@ class NavigationProvider extends ChangeNotifier { | ||||
|       screen: 'realm', | ||||
|       label: 'screenRealm', | ||||
|     ), | ||||
|     AppNavDestination( | ||||
|       icon: Icon(Symbols.newspaper, weight: 400, opticalSize: 20), | ||||
|       screen: 'news', | ||||
|       label: 'screenNews', | ||||
|     ), | ||||
|     AppNavDestination( | ||||
|       icon: Icon(Symbols.photo_library, weight: 400, opticalSize: 20), | ||||
|       screen: 'album', | ||||
| @@ -83,8 +88,7 @@ class NavigationProvider extends ChangeNotifier { | ||||
|  | ||||
|   List<AppNavDestination> destinations = []; | ||||
|  | ||||
|   int get pinnedDestinationCount => | ||||
|       destinations.where((ele) => ele.isPinned).length; | ||||
|   int get pinnedDestinationCount => destinations.where((ele) => ele.isPinned).length; | ||||
|  | ||||
|   NavigationProvider() { | ||||
|     buildDestinations(kDefaultPinnedDestination); | ||||
| @@ -113,17 +117,13 @@ class NavigationProvider extends ChangeNotifier { | ||||
|   } | ||||
|  | ||||
|   bool isIndexInRange(int min, int max) { | ||||
|     return _currentIndex != null && | ||||
|         _currentIndex! >= min && | ||||
|         _currentIndex! < max; | ||||
|     return _currentIndex != null && _currentIndex! >= min && _currentIndex! < max; | ||||
|   } | ||||
|  | ||||
|   void autoDetectIndex(GoRouter? state) { | ||||
|     if (state == null) return; | ||||
|     final idx = destinations.indexWhere( | ||||
|       (ele) => | ||||
|           ele.screen == | ||||
|           state.routerDelegate.currentConfiguration.last.route.name, | ||||
|       (ele) => ele.screen == state.routerDelegate.currentConfiguration.last.route.name, | ||||
|     ); | ||||
|     _currentIndex = idx == -1 ? null : idx; | ||||
|     notifyListeners(); | ||||
|   | ||||
| @@ -4,18 +4,27 @@ import 'dart:io'; | ||||
| import 'package:firebase_messaging/firebase_messaging.dart'; | ||||
| import 'package:flutter/foundation.dart'; | ||||
| import 'package:flutter/material.dart'; | ||||
| import 'package:flutter/services.dart'; | ||||
| import 'package:flutter_udid/flutter_udid.dart'; | ||||
| import 'package:provider/provider.dart'; | ||||
| import 'package:surface/providers/config.dart'; | ||||
| import 'package:surface/providers/sn_network.dart'; | ||||
| import 'package:surface/providers/userinfo.dart'; | ||||
| import 'package:surface/providers/websocket.dart'; | ||||
| import 'package:surface/types/notification.dart'; | ||||
| import 'package:tray_manager/tray_manager.dart'; | ||||
|  | ||||
| class NotificationProvider extends ChangeNotifier { | ||||
|   late final SnNetworkProvider _sn; | ||||
|   late final UserProvider _ua; | ||||
|   late final WebSocketProvider _ws; | ||||
|   late final ConfigProvider _cfg; | ||||
|  | ||||
|   NotificationProvider(BuildContext context) { | ||||
|     _sn = context.read<SnNetworkProvider>(); | ||||
|     _ua = context.read<UserProvider>(); | ||||
|     _ws = context.read<WebSocketProvider>(); | ||||
|     _cfg = context.read<ConfigProvider>(); | ||||
|   } | ||||
|  | ||||
|   Future<void> registerPushNotifications() async { | ||||
| @@ -62,4 +71,49 @@ class NotificationProvider extends ChangeNotifier { | ||||
|       }, | ||||
|     ); | ||||
|   } | ||||
|  | ||||
|   int showingCount = 0; | ||||
|   int showingTrayCount = 0; | ||||
|   List<SnNotification> notifications = List.empty(growable: true); | ||||
|  | ||||
|   void listen() { | ||||
|     _ws.stream.stream.listen((event) { | ||||
|       if (event.method == 'notifications.new') { | ||||
|         final notification = SnNotification.fromJson(event.payload!); | ||||
|         if (showingCount < 0) showingCount = 0; | ||||
|         showingCount++; | ||||
|         showingTrayCount++; | ||||
|         notifications.add(notification); | ||||
|         Future.delayed(const Duration(seconds: 3), () { | ||||
|           if (showingCount >= 0) showingCount--; | ||||
|           notifyListeners(); | ||||
|         }); | ||||
|         notifyListeners(); | ||||
|         updateTray(); | ||||
|         final doHaptic = _cfg.prefs.getBool(kAppNotifyWithHaptic) ?? true; | ||||
|         if (doHaptic) HapticFeedback.mediumImpact(); | ||||
|       } | ||||
|     }); | ||||
|   } | ||||
|  | ||||
|   void clearTray() { | ||||
|     showingTrayCount = 0; | ||||
|     updateTray(); | ||||
|   } | ||||
|  | ||||
|   void updateTray() { | ||||
|     if (kIsWeb || Platform.isAndroid || Platform.isIOS) return; | ||||
|     if (notifications.isEmpty) { | ||||
|       trayManager.setTitle(''); | ||||
|     } else { | ||||
|       trayManager.setTitle(' ${notifications.length.toString()}'); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   void clear() { | ||||
|     showingCount = 0; | ||||
|     notifications.clear(); | ||||
|     updateTray(); | ||||
|     notifyListeners(); | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -83,12 +83,16 @@ class SnPostContentProvider { | ||||
|     int offset = 0, | ||||
|     String? type, | ||||
|     String? author, | ||||
|     Iterable<String>? categories, | ||||
|     Iterable<String>? tags, | ||||
|   }) async { | ||||
|     final resp = await _sn.client.get('/cgi/co/posts', queryParameters: { | ||||
|       'take': take, | ||||
|       'offset': offset, | ||||
|       if (type != null) 'type': type, | ||||
|       if (author != null) 'author': author, | ||||
|       if (tags?.isNotEmpty ?? false) 'tags': tags!.join(','), | ||||
|       if (categories?.isNotEmpty ?? false) 'categories': categories!.join(','), | ||||
|     }); | ||||
|     final List<SnPost> out = await _preloadRelatedDataInBatch( | ||||
|       List.from(resp.data['data']?.map((e) => SnPost.fromJson(e)) ?? []), | ||||
| @@ -118,12 +122,14 @@ class SnPostContentProvider { | ||||
|     int take = 10, | ||||
|     int offset = 0, | ||||
|     Iterable<String>? tags, | ||||
|     Iterable<String>? categories, | ||||
|   }) async { | ||||
|     final resp = await _sn.client.get('/cgi/co/posts/search', queryParameters: { | ||||
|       'take': take, | ||||
|       'offset': offset, | ||||
|       'probe': searchTerm, | ||||
|       if (tags?.isNotEmpty ?? false) 'tags': tags!.join(','), | ||||
|       if (categories?.isNotEmpty ?? false) 'categories': categories!.join(','), | ||||
|     }); | ||||
|     final List<SnPost> out = await _preloadRelatedDataInBatch( | ||||
|       List.from(resp.data['data']?.map((e) => SnPost.fromJson(e)) ?? []), | ||||
|   | ||||
| @@ -21,7 +21,7 @@ class SnAttachmentProvider { | ||||
|  | ||||
|   void putCache(Iterable<SnAttachment> items, {bool noCheck = false}) { | ||||
|     for (final item in items) { | ||||
|       if ((item.isAnalyzed && item.isUploaded) || noCheck) { | ||||
|       if (item.isAnalyzed || noCheck) { | ||||
|         _cache[item.rid] = item; | ||||
|       } | ||||
|     } | ||||
| @@ -34,15 +34,14 @@ class SnAttachmentProvider { | ||||
|  | ||||
|     final resp = await _sn.client.get('/cgi/uc/attachments/$rid/meta'); | ||||
|     final out = SnAttachment.fromJson(resp.data); | ||||
|     if (out.isAnalyzed && out.isUploaded) { | ||||
|     if (out.isAnalyzed) { | ||||
|       _cache[rid] = out; | ||||
|     } | ||||
|  | ||||
|     return out; | ||||
|   } | ||||
|  | ||||
|   Future<List<SnAttachment?>> getMultiple(List<String> rids, | ||||
|       {noCache = false}) async { | ||||
|   Future<List<SnAttachment?>> getMultiple(List<String> rids, {noCache = false}) async { | ||||
|     final result = List<SnAttachment?>.filled(rids.length, null); | ||||
|     final Map<String, int> randomMapping = {}; | ||||
|     for (int i = 0; i < rids.length; i++) { | ||||
| @@ -63,13 +62,12 @@ class SnAttachmentProvider { | ||||
|           'id': pendingFetch.join(','), | ||||
|         }, | ||||
|       ); | ||||
|       final out = resp.data['data'] | ||||
|           .map((e) => e['id'] == 0 ? null : SnAttachment.fromJson(e)) | ||||
|           .toList(); | ||||
|       final List<SnAttachment?> out = | ||||
|           resp.data['data'].map((e) => e['id'] == 0 ? null : SnAttachment.fromJson(e)).cast<SnAttachment?>().toList(); | ||||
|  | ||||
|       for (final item in out) { | ||||
|         if (item == null) continue; | ||||
|         if (item.isAnalyzed && item.isUploaded) { | ||||
|         if (item.isAnalyzed) { | ||||
|           _cache[item.rid] = item; | ||||
|         } | ||||
|         result[randomMapping[item.rid]!] = item; | ||||
| @@ -79,10 +77,7 @@ class SnAttachmentProvider { | ||||
|     return result; | ||||
|   } | ||||
|  | ||||
|   static Map<String, String> mimetypeOverrides = { | ||||
|     'mov': 'video/quicktime', | ||||
|     'mp4': 'video/mp4' | ||||
|   }; | ||||
|   static Map<String, String> mimetypeOverrides = {'mov': 'video/quicktime', 'mp4': 'video/mp4'}; | ||||
|  | ||||
|   Future<SnAttachment> directUploadOne( | ||||
|     Uint8List data, | ||||
| @@ -91,13 +86,11 @@ class SnAttachmentProvider { | ||||
|     Map<String, dynamic>? metadata, { | ||||
|     String? mimetype, | ||||
|     Function(double progress)? onProgress, | ||||
|     bool analyzeNow = false, | ||||
|   }) async { | ||||
|     final filePayload = MultipartFile.fromBytes(data, filename: filename); | ||||
|     final fileAlt = filename.contains('.') | ||||
|         ? filename.substring(0, filename.lastIndexOf('.')) | ||||
|         : filename; | ||||
|     final fileExt = | ||||
|         filename.substring(filename.lastIndexOf('.') + 1).toLowerCase(); | ||||
|     final fileAlt = filename.contains('.') ? filename.substring(0, filename.lastIndexOf('.')) : filename; | ||||
|     final fileExt = filename.substring(filename.lastIndexOf('.') + 1).toLowerCase(); | ||||
|  | ||||
|     String? mimetypeOverride; | ||||
|     if (mimetype != null) { | ||||
| @@ -116,6 +109,7 @@ class SnAttachmentProvider { | ||||
|     final resp = await _sn.client.post( | ||||
|       '/cgi/uc/attachments', | ||||
|       data: formData, | ||||
|       queryParameters: {'analyzeNow': analyzeNow}, | ||||
|       onSendProgress: (count, total) { | ||||
|         if (onProgress != null) { | ||||
|           onProgress(count / total); | ||||
| @@ -126,18 +120,15 @@ class SnAttachmentProvider { | ||||
|     return SnAttachment.fromJson(resp.data); | ||||
|   } | ||||
|  | ||||
|   Future<(SnAttachment, int)> chunkedUploadInitialize( | ||||
|   Future<(SnAttachmentFragment, int)> chunkedUploadInitialize( | ||||
|     int size, | ||||
|     String filename, | ||||
|     String pool, | ||||
|     Map<String, dynamic>? metadata, { | ||||
|     String? mimetype, | ||||
|   }) async { | ||||
|     final fileAlt = filename.contains('.') | ||||
|         ? filename.substring(0, filename.lastIndexOf('.')) | ||||
|         : filename; | ||||
|     final fileExt = | ||||
|         filename.substring(filename.lastIndexOf('.') + 1).toLowerCase(); | ||||
|     final fileAlt = filename.contains('.') ? filename.substring(0, filename.lastIndexOf('.')) : filename; | ||||
|     final fileExt = filename.substring(filename.lastIndexOf('.') + 1).toLowerCase(); | ||||
|  | ||||
|     String? mimetypeOverride; | ||||
|     if (mimetype == null && mimetypeOverrides.keys.contains(fileExt)) { | ||||
| @@ -146,7 +137,7 @@ class SnAttachmentProvider { | ||||
|       mimetypeOverride = mimetype; | ||||
|     } | ||||
|  | ||||
|     final resp = await _sn.client.post('/cgi/uc/attachments/multipart', data: { | ||||
|     final resp = await _sn.client.post('/cgi/uc/fragments', data: { | ||||
|       'alt': fileAlt, | ||||
|       'name': filename, | ||||
|       'pool': pool, | ||||
| @@ -155,21 +146,20 @@ class SnAttachmentProvider { | ||||
|       if (mimetypeOverride != null) 'mimetype': mimetypeOverride, | ||||
|     }); | ||||
|  | ||||
|     return ( | ||||
|       SnAttachment.fromJson(resp.data['meta']), | ||||
|       resp.data['chunk_size'] as int | ||||
|     ); | ||||
|     return (SnAttachmentFragment.fromJson(resp.data['meta']), resp.data['chunk_size'] as int); | ||||
|   } | ||||
|  | ||||
|   Future<SnAttachment> _chunkedUploadOnePart( | ||||
|   Future<dynamic> _chunkedUploadOnePart( | ||||
|     Uint8List data, | ||||
|     String rid, | ||||
|     String cid, { | ||||
|     Function(double progress)? onProgress, | ||||
|     bool analyzeNow = false, | ||||
|   }) async { | ||||
|     final resp = await _sn.client.post( | ||||
|       '/cgi/uc/attachments/multipart/$rid/$cid', | ||||
|       '/cgi/uc/fragments/$rid/$cid', | ||||
|       data: data, | ||||
|       queryParameters: {'analyzeNow': analyzeNow}, | ||||
|       options: Options(headers: {'Content-Type': 'application/octet-stream'}), | ||||
|       onSendProgress: (count, total) { | ||||
|         if (onProgress != null) { | ||||
| @@ -178,21 +168,28 @@ class SnAttachmentProvider { | ||||
|       }, | ||||
|     ); | ||||
|  | ||||
|     return SnAttachment.fromJson(resp.data); | ||||
|     if (resp.data['attachment'] != null) { | ||||
|       return SnAttachment.fromJson(resp.data['attachment']); | ||||
|     } else { | ||||
|       return SnAttachmentFragment.fromJson(resp.data['fragment']); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   Future<SnAttachment> chunkedUploadParts( | ||||
|     XFile file, | ||||
|     SnAttachment place, | ||||
|     SnAttachmentFragment place, | ||||
|     int chunkSize, { | ||||
|     Function(double progress)? onProgress, | ||||
|     bool analyzeNow = false, | ||||
|   }) async { | ||||
|     final Map<String, dynamic> chunks = place.fileChunks ?? {}; | ||||
|     var currentTask = 0; | ||||
|     final Map<String, dynamic> chunks = place.fileChunks; | ||||
|     var completedTasks = 0; | ||||
|  | ||||
|     final queue = Queue<Future<void>>(); | ||||
|     final activeTasks = <Future<void>>[]; | ||||
|  | ||||
|     late SnAttachment out; | ||||
|  | ||||
|     for (final entry in chunks.entries) { | ||||
|       queue.add(() async { | ||||
|         final beginCursor = entry.value * chunkSize; | ||||
| @@ -200,25 +197,28 @@ class SnAttachmentProvider { | ||||
|           (entry.value + 1) * chunkSize, | ||||
|           await file.length(), | ||||
|         ); | ||||
|         final data = Uint8List.fromList(await file | ||||
|             .openRead(beginCursor, endCursor) | ||||
|             .expand((chunk) => chunk) | ||||
|             .toList()); | ||||
|         final data = Uint8List.fromList(await file.openRead(beginCursor, endCursor).expand((chunk) => chunk).toList()); | ||||
|  | ||||
|         place = await _chunkedUploadOnePart( | ||||
|         final result = await _chunkedUploadOnePart( | ||||
|           data, | ||||
|           place.rid, | ||||
|           entry.key, | ||||
|           onProgress: (chunkProgress) { | ||||
|             final overallProgress = | ||||
|                 (currentTask + chunkProgress) / chunks.length; | ||||
|             if (onProgress != null) { | ||||
|               onProgress(overallProgress); | ||||
|             } | ||||
|           analyzeNow: analyzeNow, | ||||
|           onProgress: (progress) { | ||||
|             final overallProgress = (completedTasks + progress) / chunks.length; | ||||
|             onProgress?.call(overallProgress); | ||||
|           }, | ||||
|         ); | ||||
|  | ||||
|         currentTask++; | ||||
|         completedTasks++; | ||||
|         final overallProgress = completedTasks / chunks.length; | ||||
|         onProgress?.call(overallProgress); | ||||
|  | ||||
|         if (result is SnAttachmentFragment) { | ||||
|           place = result; | ||||
|         } else { | ||||
|           out = result as SnAttachment; | ||||
|         } | ||||
|       }()); | ||||
|     } | ||||
|  | ||||
| @@ -235,6 +235,24 @@ class SnAttachmentProvider { | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     return place; | ||||
|     return out; | ||||
|   } | ||||
|  | ||||
|   Future<SnAttachment> updateOne( | ||||
|     SnAttachment item, { | ||||
|     String? alt, | ||||
|     int? thumbnailId, | ||||
|     int? compressedId, | ||||
|     Map<String, dynamic>? metadata, | ||||
|     bool? isIndexable, | ||||
|   }) async { | ||||
|     final resp = await _sn.client.put('/cgi/uc/attachments/${item.id}', data: { | ||||
|       'alt': alt ?? item.alt, | ||||
|       'thumbnail': thumbnailId ?? item.thumbnailId, | ||||
|       'compressed': compressedId ?? item.compressedId, | ||||
|       'metadata': metadata ?? item.usermeta, | ||||
|       'is_indexable': isIndexable ?? item.isIndexable, | ||||
|     }); | ||||
|     return SnAttachment.fromJson(resp.data); | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -68,9 +68,8 @@ class SnNetworkProvider { | ||||
|     _config.initialize().then((_) { | ||||
|       _prefs = _config.prefs; | ||||
|       client.options.baseUrl = _config.serverUrl; | ||||
|       if (!context.mounted) return; | ||||
|       _home.saveWidgetData("nex_server_url", client.options.baseUrl); | ||||
|     }); | ||||
|  | ||||
|   } | ||||
|  | ||||
|   static Future<Dio> createOffContextClient() async { | ||||
| @@ -109,6 +108,10 @@ class SnNetworkProvider { | ||||
|     return client; | ||||
|   } | ||||
|  | ||||
|   Future<void> setConfigWithNative() async { | ||||
|     _home.saveWidgetData("nex_server_url", client.options.baseUrl); | ||||
|   } | ||||
|  | ||||
|   static Future<String> _getUserAgent() async { | ||||
|     final String platformInfo; | ||||
|     if (kIsWeb) { | ||||
|   | ||||
							
								
								
									
										74
									
								
								lib/providers/sn_sticker.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,74 @@ | ||||
| import 'dart:developer'; | ||||
|  | ||||
| import 'package:flutter/material.dart'; | ||||
| import 'package:provider/provider.dart'; | ||||
| import 'package:surface/providers/sn_network.dart'; | ||||
| import 'package:surface/types/attachment.dart'; | ||||
|  | ||||
| class SnStickerProvider { | ||||
|   late final SnNetworkProvider _sn; | ||||
|   final Map<String, SnSticker?> _cache = {}; | ||||
|  | ||||
|   final Map<int, List<SnSticker>> stickersByPack = {}; | ||||
|  | ||||
|   List<SnSticker> get stickers => _cache.values.where((ele) => ele != null).cast<SnSticker>().toList(); | ||||
|  | ||||
|   SnStickerProvider(BuildContext context) { | ||||
|     _sn = context.read<SnNetworkProvider>(); | ||||
|   } | ||||
|  | ||||
|   bool hasNotSticker(String alias) { | ||||
|     return _cache.containsKey(alias) && _cache[alias] == null; | ||||
|   } | ||||
|  | ||||
|   void _cacheSticker(SnSticker sticker) { | ||||
|     _cache['${sticker.pack.prefix}:${sticker.alias}'] = sticker; | ||||
|     if (stickersByPack[sticker.pack.id] == null) stickersByPack[sticker.pack.id] = List.empty(growable: true); | ||||
|     if (!stickersByPack[sticker.pack.id]!.contains(sticker)) stickersByPack[sticker.pack.id]!.add(sticker); | ||||
|   } | ||||
|  | ||||
|   Future<SnSticker?> lookupSticker(String alias) async { | ||||
|     if (_cache.containsKey(alias)) { | ||||
|       return _cache[alias]; | ||||
|     } | ||||
|  | ||||
|     try { | ||||
|       final resp = await _sn.client.get('/cgi/uc/stickers/lookup/$alias'); | ||||
|       final sticker = SnSticker.fromJson(resp.data); | ||||
|       _cacheSticker(sticker); | ||||
|  | ||||
|       return sticker; | ||||
|     } catch (err) { | ||||
|       _cache[alias] = null; | ||||
|       log('[Sticker] Failed to lookup sticker $alias: $err'); | ||||
|     } | ||||
|  | ||||
|     return null; | ||||
|   } | ||||
|  | ||||
|   Future<void> listStickerEagerly() async { | ||||
|     var count = await listSticker(); | ||||
|     for (var page = 1; count > 0; count -= 10) { | ||||
|       await listSticker(page: page); | ||||
|       page++; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   Future<int> listSticker({int page = 0}) async { | ||||
|     try { | ||||
|       final resp = await _sn.client.get('/cgi/uc/stickers', queryParameters: { | ||||
|         'take': 10, | ||||
|         'offset': page * 10, | ||||
|       }); | ||||
|       final data = resp.data; | ||||
|       final stickers = List.from(data['data']).map((ele) => SnSticker.fromJson(ele)); | ||||
|       for (final sticker in stickers) { | ||||
|         _cacheSticker(sticker); | ||||
|       } | ||||
|       return data['count'] as int; | ||||
|     } catch (err) { | ||||
|       log('[Sticker] Failed to list stickers: $err'); | ||||
|       rethrow; | ||||
|     } | ||||
|   } | ||||
| } | ||||
							
								
								
									
										184
									
								
								lib/providers/special_day.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,184 @@ | ||||
| import 'package:flutter/widgets.dart'; | ||||
| import 'package:provider/provider.dart'; | ||||
| import 'package:surface/providers/userinfo.dart'; | ||||
|  | ||||
| // Stored as key: month, day | ||||
| final Map<String, (int, int)> kSpecialDays = { | ||||
|   // Birthday is dynamically generated according to the user's profile | ||||
|   'NewYear': (1, 1), | ||||
|   'LunarNewYear': (lunarToGregorian(null, 1, 1).month, lunarToGregorian(null, 1, 1).day), | ||||
|   'MidAutumn': (lunarToGregorian(null, 8, 15).month, lunarToGregorian(null, 8, 15).day), | ||||
|   'DragonBoat': (lunarToGregorian(null, 5, 5).month, lunarToGregorian(null, 5, 5).day), | ||||
|   'ValentineDay': (2, 14), | ||||
|   'LaborDay': (5, 1), | ||||
|   'MotherDay': (5, 11), | ||||
|   'ChildrenDay': (6, 1), | ||||
|   'FatherDay': (8, 8), | ||||
|   'Halloween': (10, 31), | ||||
|   'Thanksgiving': (11, 28), | ||||
|   'MerryXmas': (12, 25), | ||||
| }; | ||||
|  | ||||
| const Map<String, String> kSpecialDaysSymbol = { | ||||
|   'Birthday': '🎂', | ||||
|   'NewYear': '🎉', | ||||
|   'LunarNewYear': '🎉', | ||||
|   'MidAutumn': '🥮', | ||||
|   'DragonBoat': '🐲', | ||||
|   'MerryXmas': '🎄', | ||||
|   'ValentineDay': '💑', | ||||
|   'LaborDay': '🏋️', | ||||
|   'MotherDay': '👩', | ||||
|   'ChildrenDay': '👶', | ||||
|   'FatherDay': '👨', | ||||
|   'Halloween': '🎃', | ||||
|   'Thanksgiving': '🎅', | ||||
| }; | ||||
|  | ||||
| class SpecialDayProvider { | ||||
|   late final UserProvider _user; | ||||
|  | ||||
|   SpecialDayProvider(BuildContext context) { | ||||
|     _user = context.read<UserProvider>(); | ||||
|   } | ||||
|  | ||||
|   List<String> getSpecialDays() { | ||||
|     final now = DateTime.now().toLocal(); | ||||
|     final birthday = _user.user?.profile?.birthday?.toLocal(); | ||||
|     final isBirthday = birthday != null && birthday.day == now.day && birthday.month == now.month; | ||||
|  | ||||
|     return [ | ||||
|       if (isBirthday) 'Birthday', | ||||
|       ...kSpecialDays.keys.where( | ||||
|         (key) => kSpecialDays[key]!.$1 == now.month && kSpecialDays[key]!.$2 == now.day, | ||||
|       ), | ||||
|     ]; | ||||
|   } | ||||
|  | ||||
|   (String, DateTime)? getLastSpecialDay() { | ||||
|     final now = DateTime.now().toLocal(); | ||||
|     final birthday = _user.user?.profile?.birthday?.toLocal(); | ||||
|  | ||||
|     final Map<String, (int, int)> specialDays = { | ||||
|       if (birthday != null) 'Birthday': (birthday.month, birthday.day), | ||||
|       ...kSpecialDays, | ||||
|     }; | ||||
|  | ||||
|     DateTime? lastDate; | ||||
|     String? lastEvent; | ||||
|  | ||||
|     for (final entry in specialDays.entries) { | ||||
|       final eventName = entry.key; | ||||
|       final (month, day) = entry.value; | ||||
|  | ||||
|       var specialDayThisYear = DateTime(now.year, month, day); | ||||
|       var specialDayLastYear = DateTime(now.year - 1, month, day); | ||||
|  | ||||
|       if (specialDayThisYear.isBefore(now)) { | ||||
|         if (lastDate == null || specialDayThisYear.isAfter(lastDate)) { | ||||
|           lastDate = specialDayThisYear; | ||||
|           lastEvent = eventName; | ||||
|         } | ||||
|       } else if (specialDayLastYear.isBefore(now)) { | ||||
|         if (lastDate == null || specialDayLastYear.isAfter(lastDate)) { | ||||
|           lastDate = specialDayLastYear; | ||||
|           lastEvent = eventName; | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     if (lastEvent != null && lastDate != null) { | ||||
|       return (lastEvent, lastDate); | ||||
|     } | ||||
|  | ||||
|     return null; | ||||
|   } | ||||
|  | ||||
|   (String, DateTime)? getNextSpecialDay() { | ||||
|     final now = DateTime.now().toLocal(); | ||||
|     final birthday = _user.user?.profile?.birthday?.toLocal(); | ||||
|  | ||||
|     // Stored as key: month, day | ||||
|     final Map<String, (int, int)> specialDays = { | ||||
|       if (birthday != null) 'Birthday': (birthday.month, birthday.day), | ||||
|       ...kSpecialDays, | ||||
|     }; | ||||
|  | ||||
|     DateTime? closestDate; | ||||
|     String? closestEvent; | ||||
|  | ||||
|     for (final entry in specialDays.entries) { | ||||
|       final eventName = entry.key; | ||||
|       final (month, day) = entry.value; | ||||
|  | ||||
|       // Calculate the special day's DateTime in the current year | ||||
|       var specialDay = DateTime(now.year, month, day); | ||||
|  | ||||
|       // If the special day has already passed this year, consider it for the next year | ||||
|       if (specialDay.isBefore(now)) { | ||||
|         specialDay = DateTime(now.year + 1, month, day); | ||||
|       } | ||||
|  | ||||
|       // Check if this special day is closer than the previously found one | ||||
|       if (closestDate == null || specialDay.isBefore(closestDate)) { | ||||
|         closestDate = specialDay; | ||||
|         closestEvent = eventName; | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     if (closestEvent != null && closestDate != null) { | ||||
|       return (closestEvent, closestDate); | ||||
|     } | ||||
|  | ||||
|     // No special day found | ||||
|     return null; | ||||
|   } | ||||
|  | ||||
|   double getSpecialDayProgress(DateTime last, DateTime next) { | ||||
|     final totalDuration = next.add(-const Duration(days: 1)).difference(last).inSeconds.toDouble(); | ||||
|     final elapsedDuration = DateTime.now().difference(last).inSeconds.toDouble(); | ||||
|     return (elapsedDuration / totalDuration).clamp(0.0, 1.0); | ||||
|   } | ||||
| } | ||||
|  | ||||
| final Map<int, LunarYear> lunarYearData = { | ||||
|   2025: LunarYear( | ||||
|     startDate: DateTime(2025, 1, 29), | ||||
|     months: [29, 30, 30, 29, 30, 29, 29, 30, 30, 29, 30, 29], | ||||
|     leapMonth: 0, | ||||
|   ), | ||||
| }; | ||||
|  | ||||
| class LunarYear { | ||||
|   final DateTime startDate; | ||||
|   final List<int> months; | ||||
|   final int leapMonth; | ||||
|  | ||||
|   LunarYear({required this.startDate, required this.months, required this.leapMonth}); | ||||
| } | ||||
|  | ||||
| DateTime lunarToGregorian(int? year, int month, int day, {bool isLeapMonth = false}) { | ||||
|   year = year ?? DateTime.now().year; | ||||
|   final lunarYear = lunarYearData[year]; | ||||
|   if (lunarYear == null) { | ||||
|     throw Exception('Lunar data for year $year not found'); | ||||
|   } | ||||
|  | ||||
|   int leapMonth = lunarYear.leapMonth; | ||||
|   if (isLeapMonth && (leapMonth == 0 || leapMonth != month)) { | ||||
|     throw Exception('Invalid leap month for year $year'); | ||||
|   } | ||||
|  | ||||
|   int daysFromStart = 0; | ||||
|   for (int i = 0; i < month - 1; i++) { | ||||
|     daysFromStart += lunarYear.months[i]; | ||||
|   } | ||||
|  | ||||
|   if (isLeapMonth) { | ||||
|     daysFromStart += lunarYear.months[month - 1]; | ||||
|   } | ||||
|  | ||||
|   daysFromStart += day - 1; | ||||
|  | ||||
|   return lunarYear.startDate.add(Duration(days: daysFromStart)); | ||||
| } | ||||
| @@ -1,3 +1,5 @@ | ||||
| import 'dart:ui'; | ||||
|  | ||||
| import 'package:flutter/foundation.dart'; | ||||
| import 'package:surface/theme.dart'; | ||||
|  | ||||
| @@ -11,8 +13,8 @@ class ThemeProvider extends ChangeNotifier { | ||||
|     }); | ||||
|   } | ||||
|  | ||||
|   void reloadTheme({bool? useMaterial3}) { | ||||
|     createAppThemeSet().then((value) { | ||||
|   void reloadTheme({Color? seedColorOverride, bool? useMaterial3}) { | ||||
|     createAppThemeSet(seedColorOverride: seedColorOverride, useMaterial3: useMaterial3).then((value) { | ||||
|       theme = value; | ||||
|       notifyListeners(); | ||||
|     }); | ||||
|   | ||||
| @@ -1,12 +1,10 @@ | ||||
| import 'dart:developer'; | ||||
|  | ||||
| import 'package:flutter/material.dart'; | ||||
| import 'package:path/path.dart'; | ||||
| import 'package:provider/provider.dart'; | ||||
| import 'package:shared_preferences/shared_preferences.dart'; | ||||
| import 'package:surface/providers/config.dart'; | ||||
| import 'package:surface/providers/sn_network.dart'; | ||||
| import 'package:surface/providers/widget.dart'; | ||||
| import 'package:surface/types/account.dart'; | ||||
|  | ||||
| class UserProvider extends ChangeNotifier { | ||||
| @@ -14,12 +12,10 @@ class UserProvider extends ChangeNotifier { | ||||
|   SnAccount? user; | ||||
|  | ||||
|   late final SnNetworkProvider _sn; | ||||
|   late final HomeWidgetProvider _home; | ||||
|   late final ConfigProvider _config; | ||||
|  | ||||
|   UserProvider(BuildContext context) { | ||||
|     _sn = context.read<SnNetworkProvider>(); | ||||
|     _home = context.read<HomeWidgetProvider>(); | ||||
|     _config = context.read<ConfigProvider>(); | ||||
|   } | ||||
|  | ||||
| @@ -32,10 +28,10 @@ class UserProvider extends ChangeNotifier { | ||||
|     final value = _config.prefs.getString(kAtkStoreKey); | ||||
|     isAuthorized = value != null; | ||||
|     notifyListeners(); | ||||
|     refreshUser().then((value) { | ||||
|     refreshUser().then((value) async { | ||||
|       if (value != null) { | ||||
|         log('Logged in as @${value.name}'); | ||||
|         _home.saveWidgetData('user', value.toJson()); | ||||
|         log('Atk: ${await atk}'); | ||||
|       } | ||||
|     }); | ||||
|   } | ||||
| @@ -57,4 +53,11 @@ class UserProvider extends ChangeNotifier { | ||||
|     user = null; | ||||
|     notifyListeners(); | ||||
|   } | ||||
|  | ||||
|   void setLanguage(String? value) { | ||||
|     if (value == null) return; | ||||
|     if (user == null) return; | ||||
|     user = user!.copyWith(language: value); | ||||
|     notifyListeners(); | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -33,9 +33,18 @@ class WebSocketProvider extends ChangeNotifier { | ||||
|     await connect(); | ||||
|   } | ||||
|  | ||||
|   Completer<void>? _connectCompleter; | ||||
|  | ||||
|   Future<void> connect({noRetry = false}) async { | ||||
|     if(_connectCompleter != null) { | ||||
|       await _connectCompleter!.future; | ||||
|       _connectCompleter = null; | ||||
|     } | ||||
|  | ||||
|     _connectCompleter = Completer<void>(); | ||||
|  | ||||
|     if (!_ua.isAuthorized) return; | ||||
|     if (isConnected) { | ||||
|     if (isConnected || conn != null) { | ||||
|       disconnect(); | ||||
|     } | ||||
|  | ||||
| @@ -64,12 +73,13 @@ class WebSocketProvider extends ChangeNotifier { | ||||
|         log('Retry connecting to websocket in 3 seconds...'); | ||||
|         return Future.delayed( | ||||
|           const Duration(seconds: 3), | ||||
|           () => connect(noRetry: true), | ||||
|               () => connect(noRetry: true), | ||||
|         ); | ||||
|       } | ||||
|     } finally { | ||||
|       isBusy = false; | ||||
|       notifyListeners(); | ||||
|       _connectCompleter!.complete(); | ||||
|     } | ||||
|   } | ||||
|  | ||||
| @@ -97,7 +107,7 @@ class WebSocketProvider extends ChangeNotifier { | ||||
|       onError: (err) { | ||||
|         isConnected = false; | ||||
|         notifyListeners(); | ||||
|         Future.delayed(const Duration(seconds: 11), () => connect()); | ||||
|         Future.delayed(const Duration(seconds: 1), () => connect()); | ||||
|       }, | ||||
|     ); | ||||
|   } | ||||
|   | ||||
| @@ -13,7 +13,7 @@ class HomeWidgetProvider { | ||||
|  | ||||
|   Future<void> initialize() async { | ||||
|     if (kIsWeb || !(Platform.isAndroid || Platform.isIOS)) return; | ||||
|     if (!kIsWeb && Platform.isIOS) { | ||||
|     if (Platform.isIOS) { | ||||
|       await HomeWidget.setAppGroupId("group.solsynth.solian"); | ||||
|     } | ||||
|   } | ||||
| @@ -47,6 +47,7 @@ class HomeWidgetProvider { | ||||
| } | ||||
|  | ||||
| Future<void> widgetUpdateRandomPost() async { | ||||
|   if (kIsWeb || (!Platform.isAndroid && !Platform.isIOS)) return; | ||||
|   final snc = await SnNetworkProvider.createOffContextClient(); | ||||
|   final resp = await snc.get('/cgi/co/recommendations/shuffle?take=1'); | ||||
|   final post = SnPost.fromJson(resp.data['data'][0]); | ||||
|   | ||||
							
								
								
									
										422
									
								
								lib/router.dart
									
									
									
									
									
								
							
							
						
						| @@ -3,7 +3,9 @@ import 'package:flutter/material.dart'; | ||||
| import 'package:go_router/go_router.dart'; | ||||
| import 'package:surface/screens/abuse_report.dart'; | ||||
| import 'package:surface/screens/account.dart'; | ||||
| import 'package:surface/screens/account/pfp.dart'; | ||||
| import 'package:surface/screens/account/account_settings.dart'; | ||||
| import 'package:surface/screens/account/factor_settings.dart'; | ||||
| import 'package:surface/screens/account/profile_page.dart'; | ||||
| import 'package:surface/screens/account/profile_edit.dart'; | ||||
| import 'package:surface/screens/account/publishers/publisher_edit.dart'; | ||||
| import 'package:surface/screens/account/publishers/publisher_new.dart'; | ||||
| @@ -19,6 +21,8 @@ import 'package:surface/screens/chat/room.dart'; | ||||
| import 'package:surface/screens/explore.dart'; | ||||
| import 'package:surface/screens/friend.dart'; | ||||
| import 'package:surface/screens/home.dart'; | ||||
| import 'package:surface/screens/news/news_detail.dart'; | ||||
| import 'package:surface/screens/news/news_list.dart'; | ||||
| import 'package:surface/screens/notification.dart'; | ||||
| import 'package:surface/screens/post/post_detail.dart'; | ||||
| import 'package:surface/screens/post/post_editor.dart'; | ||||
| @@ -29,287 +33,227 @@ import 'package:surface/screens/realm/manage.dart'; | ||||
| import 'package:surface/screens/realm/realm_detail.dart'; | ||||
| import 'package:surface/screens/settings.dart'; | ||||
| import 'package:surface/screens/sharing.dart'; | ||||
| import 'package:surface/screens/wallet.dart'; | ||||
| import 'package:surface/types/post.dart'; | ||||
| import 'package:surface/widgets/about.dart'; | ||||
| import 'package:surface/widgets/navigation/app_background.dart'; | ||||
| import 'package:surface/widgets/navigation/app_scaffold.dart'; | ||||
|  | ||||
| Widget _fadeThroughTransition( | ||||
|     BuildContext context, Animation<double> animation, Animation<double> secondaryAnimation, Widget child) { | ||||
|   return FadeThroughTransition( | ||||
|     animation: animation, | ||||
|     secondaryAnimation: secondaryAnimation, | ||||
|     fillColor: Colors.transparent, | ||||
|     child: child, | ||||
|   ); | ||||
| } | ||||
|  | ||||
| final _appRoutes = [ | ||||
|   ShellRoute( | ||||
|     builder: (context, state, child) => AppPageScaffold( | ||||
|       body: child, | ||||
|       showAppBar: false, | ||||
|     ), | ||||
|   GoRoute( | ||||
|     path: '/', | ||||
|     name: 'home', | ||||
|     builder: (context, state) => const HomeScreen(), | ||||
|   ), | ||||
|   GoRoute( | ||||
|     path: '/posts', | ||||
|     name: 'explore', | ||||
|     builder: (context, state) => const ExploreScreen(), | ||||
|     routes: [ | ||||
|       GoRoute( | ||||
|         path: '/', | ||||
|         name: 'home', | ||||
|         pageBuilder: (context, state) => NoTransitionPage( | ||||
|           child: const HomeScreen(), | ||||
|         path: '/write/:mode', | ||||
|         name: 'postEditor', | ||||
|         builder: (context, state) => PostEditorScreen( | ||||
|           mode: state.pathParameters['mode']!, | ||||
|           postEditId: int.tryParse( | ||||
|             state.uri.queryParameters['editing'] ?? '', | ||||
|           ), | ||||
|           postReplyId: int.tryParse( | ||||
|             state.uri.queryParameters['replying'] ?? '', | ||||
|           ), | ||||
|           postRepostId: int.tryParse( | ||||
|             state.uri.queryParameters['reposting'] ?? '', | ||||
|           ), | ||||
|           extraProps: state.extra as PostEditorExtra?, | ||||
|         ), | ||||
|       ), | ||||
|       GoRoute( | ||||
|         path: '/posts', | ||||
|         name: 'explore', | ||||
|         pageBuilder: (context, state) => NoTransitionPage( | ||||
|           child: const ExploreScreen(), | ||||
|         ), | ||||
|         routes: [ | ||||
|           GoRoute( | ||||
|             path: '/write/:mode', | ||||
|             name: 'postEditor', | ||||
|             builder: (context, state) => AppBackground( | ||||
|               child: PostEditorScreen( | ||||
|                 mode: state.pathParameters['mode']!, | ||||
|                 postEditId: int.tryParse( | ||||
|                   state.uri.queryParameters['editing'] ?? '', | ||||
|                 ), | ||||
|                 postReplyId: int.tryParse( | ||||
|                   state.uri.queryParameters['replying'] ?? '', | ||||
|                 ), | ||||
|                 postRepostId: int.tryParse( | ||||
|                   state.uri.queryParameters['reposting'] ?? '', | ||||
|                 ), | ||||
|                 extraProps: state.extra as PostEditorExtraProps?, | ||||
|               ), | ||||
|             ), | ||||
|           ), | ||||
|           GoRoute( | ||||
|             path: '/search', | ||||
|             name: 'postSearch', | ||||
|             builder: (context, state) => const AppBackground( | ||||
|               child: PostSearchScreen(), | ||||
|             ), | ||||
|           ), | ||||
|           GoRoute( | ||||
|             path: '/publishers/:name', | ||||
|             name: 'postPublisher', | ||||
|             builder: (context, state) => AppBackground( | ||||
|               child: PostPublisherScreen(name: state.pathParameters['name']!), | ||||
|             ), | ||||
|           ), | ||||
|           GoRoute( | ||||
|             path: '/:slug', | ||||
|             name: 'postDetail', | ||||
|             builder: (context, state) => AppBackground( | ||||
|               child: PostDetailScreen( | ||||
|                 slug: state.pathParameters['slug']!, | ||||
|                 preload: state.extra as SnPost?, | ||||
|               ), | ||||
|             ), | ||||
|           ), | ||||
|         ], | ||||
|       ), | ||||
|       GoRoute( | ||||
|         path: '/account', | ||||
|         name: 'account', | ||||
|         pageBuilder: (context, state) => NoTransitionPage( | ||||
|           child: const AccountScreen(), | ||||
|         ), | ||||
|         routes: [], | ||||
|       ), | ||||
|       GoRoute( | ||||
|         path: '/chat', | ||||
|         name: 'chat', | ||||
|         pageBuilder: (context, state) => NoTransitionPage( | ||||
|           child: const ChatScreen(), | ||||
|         ), | ||||
|         routes: [ | ||||
|           GoRoute( | ||||
|             path: '/:scope/:alias', | ||||
|             name: 'chatRoom', | ||||
|             builder: (context, state) => AppBackground( | ||||
|               child: ChatRoomScreen( | ||||
|                 scope: state.pathParameters['scope']!, | ||||
|                 alias: state.pathParameters['alias']!, | ||||
|               ), | ||||
|             ), | ||||
|           ), | ||||
|           GoRoute( | ||||
|             path: '/:scope/:alias/call', | ||||
|             name: 'chatCallRoom', | ||||
|             builder: (context, state) => AppBackground( | ||||
|               child: CallRoomScreen( | ||||
|                 scope: state.pathParameters['scope']!, | ||||
|                 alias: state.pathParameters['alias']!, | ||||
|               ), | ||||
|             ), | ||||
|           ), | ||||
|           GoRoute( | ||||
|             path: '/:scope/:alias/detail', | ||||
|             name: 'channelDetail', | ||||
|             builder: (context, state) => AppBackground( | ||||
|               child: ChannelDetailScreen( | ||||
|                 scope: state.pathParameters['scope']!, | ||||
|                 alias: state.pathParameters['alias']!, | ||||
|               ), | ||||
|             ), | ||||
|           ), | ||||
|           GoRoute( | ||||
|             path: '/manage', | ||||
|             name: 'chatManage', | ||||
|             pageBuilder: (context, state) => CustomTransitionPage( | ||||
|               child: ChatManageScreen( | ||||
|                 editingChannelAlias: state.uri.queryParameters['editing'], | ||||
|               ), | ||||
|               transitionsBuilder: (context, animation, secondaryAnimation, child) { | ||||
|                 return FadeThroughTransition( | ||||
|                   animation: animation, | ||||
|                   secondaryAnimation: secondaryAnimation, | ||||
|                   fillColor: Colors.transparent, | ||||
|                   child: AppBackground( | ||||
|                     child: child, | ||||
|                   ), | ||||
|                 ); | ||||
|               }, | ||||
|             ), | ||||
|           ), | ||||
|           GoRoute( | ||||
|             path: '/:alias', | ||||
|             name: 'realmDetail', | ||||
|             builder: (context, state) => AppBackground( | ||||
|               child: RealmDetailScreen(alias: state.pathParameters['alias']!), | ||||
|             ), | ||||
|           ), | ||||
|         ], | ||||
|       ), | ||||
|       GoRoute( | ||||
|         path: '/realm', | ||||
|         name: 'realm', | ||||
|         pageBuilder: (context, state) => NoTransitionPage( | ||||
|           child: const RealmScreen(), | ||||
|         ), | ||||
|         routes: [ | ||||
|           GoRoute( | ||||
|             path: '/manage', | ||||
|             name: 'realmManage', | ||||
|             pageBuilder: (context, state) => CustomTransitionPage( | ||||
|               child: RealmManageScreen( | ||||
|                 editingRealmAlias: state.uri.queryParameters['editing'], | ||||
|               ), | ||||
|               transitionsBuilder: (context, animation, secondaryAnimation, child) { | ||||
|                 return FadeThroughTransition( | ||||
|                   animation: animation, | ||||
|                   secondaryAnimation: secondaryAnimation, | ||||
|                   fillColor: Colors.transparent, | ||||
|                   child: AppBackground( | ||||
|                     child: child, | ||||
|                   ), | ||||
|                 ); | ||||
|               }, | ||||
|             ), | ||||
|           ), | ||||
|         ], | ||||
|       ), | ||||
|       GoRoute( | ||||
|         path: '/album', | ||||
|         name: 'album', | ||||
|         pageBuilder: (context, state) => NoTransitionPage( | ||||
|           child: const AlbumScreen(), | ||||
|         path: '/search', | ||||
|         name: 'postSearch', | ||||
|         builder: (context, state) => PostSearchScreen( | ||||
|           initialTags: state.uri.queryParameters['tags']?.split(','), | ||||
|           initialCategories: state.uri.queryParameters['categories']?.split(','), | ||||
|         ), | ||||
|       ), | ||||
|       GoRoute( | ||||
|         path: '/friend', | ||||
|         name: 'friend', | ||||
|         pageBuilder: (context, state) => NoTransitionPage( | ||||
|           child: const FriendScreen(), | ||||
|         ), | ||||
|         path: '/publishers/:name', | ||||
|         name: 'postPublisher', | ||||
|         builder: (context, state) => PostPublisherScreen(name: state.pathParameters['name']!), | ||||
|       ), | ||||
|       GoRoute( | ||||
|         path: '/notification', | ||||
|         name: 'notification', | ||||
|         pageBuilder: (context, state) => NoTransitionPage( | ||||
|           child: const NotificationScreen(), | ||||
|         path: '/:slug', | ||||
|         name: 'postDetail', | ||||
|         builder: (context, state) => PostDetailScreen( | ||||
|           slug: state.pathParameters['slug']!, | ||||
|           preload: state.extra as SnPost?, | ||||
|         ), | ||||
|       ), | ||||
|     ], | ||||
|   ), | ||||
|   ShellRoute( | ||||
|     builder: (context, state, child) => AppPageScaffold(body: child), | ||||
|   GoRoute(path: '/account', name: 'account', builder: (context, state) => const AccountScreen(), routes: [ | ||||
|     GoRoute( | ||||
|       path: '/wallet', | ||||
|       name: 'accountWallet', | ||||
|       builder: (context, state) => const WalletScreen(), | ||||
|     ), | ||||
|     GoRoute( | ||||
|       path: '/settings', | ||||
|       name: 'accountSettings', | ||||
|       builder: (context, state) => AccountSettingsScreen(), | ||||
|     ), | ||||
|     GoRoute( | ||||
|       path: '/settings/factors', | ||||
|       name: 'factorSettings', | ||||
|       builder: (context, state) => FactorSettingsScreen(), | ||||
|     ), | ||||
|     GoRoute( | ||||
|       path: '/profile/edit', | ||||
|       name: 'accountProfileEdit', | ||||
|       builder: (context, state) => ProfileEditScreen(), | ||||
|     ), | ||||
|     GoRoute( | ||||
|       path: '/publishers', | ||||
|       name: 'accountPublishers', | ||||
|       builder: (context, state) => PublisherScreen(), | ||||
|     ), | ||||
|     GoRoute( | ||||
|       path: '/publishers/new', | ||||
|       name: 'accountPublisherNew', | ||||
|       builder: (context, state) => AccountPublisherNewScreen(), | ||||
|     ), | ||||
|     GoRoute( | ||||
|       path: '/publishers/edit/:name', | ||||
|       name: 'accountPublisherEdit', | ||||
|       builder: (context, state) => AccountPublisherEditScreen( | ||||
|         name: state.pathParameters['name']!, | ||||
|       ), | ||||
|     ), | ||||
|     GoRoute( | ||||
|       path: '/:name', | ||||
|       name: 'accountProfilePage', | ||||
|       pageBuilder: (context, state) => NoTransitionPage( | ||||
|         child: UserScreen(name: state.pathParameters['name']!), | ||||
|       ), | ||||
|     ), | ||||
|   ]), | ||||
|   GoRoute( | ||||
|     path: '/chat', | ||||
|     name: 'chat', | ||||
|     builder: (context, state) => const ChatScreen(), | ||||
|     routes: [ | ||||
|       GoRoute( | ||||
|         path: '/auth/login', | ||||
|         name: 'authLogin', | ||||
|         builder: (context, state) => const AppBackground( | ||||
|           child: LoginScreen(), | ||||
|         path: '/:scope/:alias', | ||||
|         name: 'chatRoom', | ||||
|         builder: (context, state) => ChatRoomScreen( | ||||
|           scope: state.pathParameters['scope']!, | ||||
|           alias: state.pathParameters['alias']!, | ||||
|           extra: state.extra as ChatRoomScreenExtra?, | ||||
|         ), | ||||
|       ), | ||||
|       GoRoute( | ||||
|         path: '/auth/register', | ||||
|         name: 'authRegister', | ||||
|         builder: (context, state) => const AppBackground( | ||||
|           child: RegisterScreen(), | ||||
|         path: '/:scope/:alias/call', | ||||
|         name: 'chatCallRoom', | ||||
|         builder: (context, state) => CallRoomScreen( | ||||
|           scope: state.pathParameters['scope']!, | ||||
|           alias: state.pathParameters['alias']!, | ||||
|         ), | ||||
|       ), | ||||
|       GoRoute( | ||||
|         path: '/reports', | ||||
|         name: 'abuseReport', | ||||
|         builder: (context, state) => const AppBackground( | ||||
|           child: AbuseReportScreen(), | ||||
|         path: '/:scope/:alias/detail', | ||||
|         name: 'channelDetail', | ||||
|         builder: (context, state) => ChannelDetailScreen( | ||||
|           scope: state.pathParameters['scope']!, | ||||
|           alias: state.pathParameters['alias']!, | ||||
|         ), | ||||
|       ), | ||||
|       GoRoute( | ||||
|         path: '/account/profile/edit', | ||||
|         name: 'accountProfileEdit', | ||||
|         builder: (context, state) => const AppBackground( | ||||
|           child: ProfileEditScreen(), | ||||
|         ), | ||||
|       ), | ||||
|       GoRoute( | ||||
|         path: '/account/publishers', | ||||
|         name: 'accountPublishers', | ||||
|         builder: (context, state) => const AppBackground( | ||||
|           child: PublisherScreen(), | ||||
|         ), | ||||
|       ), | ||||
|       GoRoute( | ||||
|         path: '/account/publishers/new', | ||||
|         name: 'accountPublisherNew', | ||||
|         builder: (context, state) => const AppBackground( | ||||
|           child: AccountPublisherNewScreen(), | ||||
|         ), | ||||
|       ), | ||||
|       GoRoute( | ||||
|         path: '/account/publishers/edit/:name', | ||||
|         name: 'accountPublisherEdit', | ||||
|         builder: (context, state) => AppBackground( | ||||
|           child: AccountPublisherEditScreen( | ||||
|             name: state.pathParameters['name']!, | ||||
|           ), | ||||
|         path: '/manage', | ||||
|         name: 'chatManage', | ||||
|         builder: (context, state) => ChatManageScreen( | ||||
|           editingChannelAlias: state.uri.queryParameters['editing'], | ||||
|         ), | ||||
|       ), | ||||
|     ], | ||||
|   ), | ||||
|   GoRoute( | ||||
|     path: '/account/:name', | ||||
|     name: 'accountProfilePage', | ||||
|     pageBuilder: (context, state) => NoTransitionPage( | ||||
|       child: UserScreen(name: state.pathParameters['name']!), | ||||
|     path: '/realm', | ||||
|     name: 'realm', | ||||
|     pageBuilder: (context, state) => CustomTransitionPage( | ||||
|       transitionsBuilder: _fadeThroughTransition, | ||||
|       child: const RealmScreen(), | ||||
|     ), | ||||
|   ), | ||||
|   ShellRoute( | ||||
|     builder: (context, state, child) => AppPageScaffold(body: child), | ||||
|     routes: [ | ||||
|       GoRoute( | ||||
|         path: '/settings', | ||||
|         name: 'settings', | ||||
|         builder: (context, state) => const AppBackground( | ||||
|           child: SettingsScreen(), | ||||
|         path: '/:alias', | ||||
|         name: 'realmDetail', | ||||
|         builder: (context, state) => RealmDetailScreen(alias: state.pathParameters['alias']!), | ||||
|       ), | ||||
|       GoRoute( | ||||
|         path: '/manage', | ||||
|         name: 'realmManage', | ||||
|         builder: (context, state) => RealmManageScreen( | ||||
|           editingRealmAlias: state.uri.queryParameters['editing'], | ||||
|         ), | ||||
|       ), | ||||
|     ], | ||||
|   ), | ||||
|   ShellRoute( | ||||
|     builder: (context, state, child) => AppPageScaffold(body: child), | ||||
|     routes: [ | ||||
|       GoRoute( | ||||
|         path: '/about', | ||||
|         name: 'about', | ||||
|         builder: (context, state) => const AppBackground( | ||||
|           child: AboutScreen(), | ||||
|         ), | ||||
|   GoRoute(path: '/news', name: 'news', builder: (context, state) => const NewsScreen(), routes: [ | ||||
|     GoRoute( | ||||
|       path: '/:hash', | ||||
|       name: 'newsDetail', | ||||
|       builder: (context, state) => NewsDetailScreen( | ||||
|         hash: state.pathParameters['hash']!, | ||||
|       ), | ||||
|     ], | ||||
|     ), | ||||
|   ]), | ||||
|   GoRoute( | ||||
|     path: '/album', | ||||
|     name: 'album', | ||||
|     builder: (context, state) => const AlbumScreen(), | ||||
|   ), | ||||
|   GoRoute( | ||||
|     path: '/friend', | ||||
|     name: 'friend', | ||||
|     builder: (context, state) => const FriendScreen(), | ||||
|   ), | ||||
|   GoRoute( | ||||
|     path: '/notification', | ||||
|     name: 'notification', | ||||
|     builder: (context, state) => const NotificationScreen(), | ||||
|   ), | ||||
|   GoRoute( | ||||
|     path: '/auth/login', | ||||
|     name: 'authLogin', | ||||
|     builder: (context, state) => LoginScreen(), | ||||
|   ), | ||||
|   GoRoute( | ||||
|     path: '/auth/register', | ||||
|     name: 'authRegister', | ||||
|     builder: (context, state) => RegisterScreen(), | ||||
|   ), | ||||
|   GoRoute( | ||||
|     path: '/reports', | ||||
|     name: 'abuseReport', | ||||
|     builder: (context, state) => AbuseReportScreen(), | ||||
|   ), | ||||
|   GoRoute( | ||||
|     path: '/settings', | ||||
|     name: 'settings', | ||||
|     builder: (context, state) => SettingsScreen(), | ||||
|   ), | ||||
|   GoRoute( | ||||
|     path: '/about', | ||||
|     name: 'about', | ||||
|     builder: (context, state) => AboutScreen(), | ||||
|   ), | ||||
| ]; | ||||
|  | ||||
|   | ||||
| @@ -6,6 +6,7 @@ import 'package:provider/provider.dart'; | ||||
| import 'package:styled_widget/styled_widget.dart'; | ||||
| import 'package:surface/providers/sn_network.dart'; | ||||
| import 'package:surface/widgets/dialog.dart'; | ||||
| import 'package:surface/widgets/navigation/app_scaffold.dart'; | ||||
|  | ||||
| import '../types/account.dart'; | ||||
|  | ||||
| @@ -56,7 +57,11 @@ class _AbuseReportScreenState extends State<AbuseReportScreen> { | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     return Scaffold( | ||||
|     return AppScaffold( | ||||
|       appBar: AppBar( | ||||
|         leading: const PageBackButton(), | ||||
|         title: Text('screenAbuseReport').tr(), | ||||
|       ), | ||||
|       body: Column( | ||||
|         children: [ | ||||
|           ListTile( | ||||
| @@ -73,6 +78,7 @@ class _AbuseReportScreenState extends State<AbuseReportScreen> { | ||||
|           else | ||||
|             Expanded( | ||||
|               child: ListView.builder( | ||||
|                 padding: EdgeInsets.only(top: 8), | ||||
|                 itemCount: _reports.length, | ||||
|                 itemBuilder: (context, idx) { | ||||
|                   return ListTile( | ||||
|   | ||||
| @@ -1,3 +1,5 @@ | ||||
| import 'dart:ui'; | ||||
|  | ||||
| import 'package:easy_localization/easy_localization.dart'; | ||||
| import 'package:flutter/material.dart'; | ||||
| import 'package:gap/gap.dart'; | ||||
| @@ -12,6 +14,8 @@ import 'package:surface/providers/websocket.dart'; | ||||
| import 'package:surface/widgets/account/account_image.dart'; | ||||
| import 'package:surface/widgets/app_bar_leading.dart'; | ||||
| import 'package:surface/widgets/dialog.dart'; | ||||
| import 'package:surface/widgets/navigation/app_scaffold.dart'; | ||||
| import 'package:surface/widgets/universal_image.dart'; | ||||
|  | ||||
| class AccountScreen extends StatelessWidget { | ||||
|   const AccountScreen({super.key}); | ||||
| @@ -19,11 +23,51 @@ class AccountScreen extends StatelessWidget { | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     final ua = context.watch<UserProvider>(); | ||||
|     final sn = context.read<SnNetworkProvider>(); | ||||
|  | ||||
|     return Scaffold( | ||||
|     return AppScaffold( | ||||
|       appBar: AppBar( | ||||
|         leading: AutoAppBarLeading(), | ||||
|         title: Text("screenAccount").tr(), | ||||
|         title: Text( | ||||
|           "screenAccount", | ||||
|           style: TextStyle( | ||||
|             color: Colors.white, | ||||
|             shadows: [ | ||||
|               Shadow( | ||||
|                 offset: Offset(1, 1), | ||||
|                 blurRadius: 5.0, | ||||
|                 color: Color.fromARGB(255, 0, 0, 0), | ||||
|               ), | ||||
|             ], | ||||
|           ), | ||||
|         ).tr(), | ||||
|         flexibleSpace: ua.user != null && ua.user!.banner.isNotEmpty | ||||
|             ? Stack( | ||||
|                 fit: StackFit.expand, | ||||
|                 children: [ | ||||
|                   AutoResizeUniversalImage(sn.getAttachmentUrl(ua.user!.banner), fit: BoxFit.cover), | ||||
|                   Positioned( | ||||
|                     top: 0, | ||||
|                     left: 0, | ||||
|                     right: 0, | ||||
|                     height: 56 + MediaQuery.of(context).padding.top, | ||||
|                     child: ClipRect( | ||||
|                       child: BackdropFilter( | ||||
|                         filter: ImageFilter.blur( | ||||
|                           sigmaX: 10, | ||||
|                           sigmaY: 10, | ||||
|                         ), | ||||
|                         child: Container( | ||||
|                           color: Colors.black.withOpacity( | ||||
|                             clampDouble(10 * 0.1, 0, 0.5), | ||||
|                           ), | ||||
|                         ), | ||||
|                       ), | ||||
|                     ), | ||||
|                   ), | ||||
|                 ], | ||||
|               ) | ||||
|             : null, | ||||
|         actions: [ | ||||
|           IconButton( | ||||
|             icon: const Icon(Symbols.settings, fill: 1), | ||||
| @@ -82,16 +126,6 @@ class _AuthorizedAccountScreen extends StatelessWidget { | ||||
|             ); | ||||
|           }).padding(all: 20), | ||||
|         ).padding(horizontal: 8, top: 16, bottom: 4), | ||||
|         ListTile( | ||||
|           title: Text('accountProfileEdit').tr(), | ||||
|           subtitle: Text('accountProfileEditSubtitle').tr(), | ||||
|           contentPadding: const EdgeInsets.symmetric(horizontal: 24), | ||||
|           leading: const Icon(Symbols.contact_page), | ||||
|           trailing: const Icon(Symbols.chevron_right), | ||||
|           onTap: () { | ||||
|             GoRouter.of(context).pushNamed('accountProfileEdit'); | ||||
|           }, | ||||
|         ), | ||||
|         ListTile( | ||||
|           title: Text('accountPublishers').tr(), | ||||
|           subtitle: Text('accountPublishersSubtitle').tr(), | ||||
| @@ -112,6 +146,36 @@ class _AuthorizedAccountScreen extends StatelessWidget { | ||||
|             GoRouter.of(context).pushNamed('abuseReport'); | ||||
|           }, | ||||
|         ), | ||||
|         ListTile( | ||||
|           title: Text('factorSettings').tr(), | ||||
|           subtitle: Text('factorSettingsSubtitle').tr(), | ||||
|           contentPadding: const EdgeInsets.symmetric(horizontal: 24), | ||||
|           leading: const Icon(Symbols.lock), | ||||
|           trailing: const Icon(Symbols.chevron_right), | ||||
|           onTap: () { | ||||
|             GoRouter.of(context).pushNamed('factorSettings'); | ||||
|           }, | ||||
|         ), | ||||
|         ListTile( | ||||
|           title: Text('accountWallet').tr(), | ||||
|           subtitle: Text('accountWalletSubtitle').tr(), | ||||
|           contentPadding: const EdgeInsets.symmetric(horizontal: 24), | ||||
|           leading: const Icon(Symbols.wallet), | ||||
|           trailing: const Icon(Symbols.chevron_right), | ||||
|           onTap: () { | ||||
|             GoRouter.of(context).pushNamed('accountWallet'); | ||||
|           }, | ||||
|         ), | ||||
|         ListTile( | ||||
|           title: Text('accountSettings').tr(), | ||||
|           subtitle: Text('accountSettingsSubtitle').tr(), | ||||
|           contentPadding: const EdgeInsets.symmetric(horizontal: 24), | ||||
|           leading: const Icon(Symbols.manage_accounts), | ||||
|           trailing: const Icon(Symbols.chevron_right), | ||||
|           onTap: () { | ||||
|             GoRouter.of(context).pushNamed('accountSettings'); | ||||
|           }, | ||||
|         ), | ||||
|         ListTile( | ||||
|           title: Text('accountLogout').tr(), | ||||
|           subtitle: Text('accountLogoutSubtitle').tr(), | ||||
| @@ -133,33 +197,6 @@ class _AuthorizedAccountScreen extends StatelessWidget { | ||||
|             await Hive.initFlutter(); | ||||
|           }, | ||||
|         ), | ||||
|         ListTile( | ||||
|           title: Text('accountDeletion'.tr()), | ||||
|           subtitle: Text('accountDeletionActionDescription'.tr()), | ||||
|           contentPadding: const EdgeInsets.symmetric(horizontal: 24), | ||||
|           leading: const Icon(Symbols.person_cancel), | ||||
|           trailing: const Icon(Symbols.chevron_right), | ||||
|           onTap: () { | ||||
|             context | ||||
|                 .showConfirmDialog( | ||||
|               'accountDeletion'.tr(), | ||||
|               'accountDeletionDescription'.tr(), | ||||
|             ) | ||||
|                 .then((value) { | ||||
|               if (!value || !context.mounted) return; | ||||
|               final sn = context.read<SnNetworkProvider>(); | ||||
|               sn.client.post('/cgi/id/users/me/deletion').then((value) { | ||||
|                 if (context.mounted) { | ||||
|                   context.showSnackbar('accountDeletionSubmitted'.tr()); | ||||
|                 } | ||||
|               }).catchError((err) { | ||||
|                 if (context.mounted) { | ||||
|                   context.showErrorDialog(err); | ||||
|                 } | ||||
|               }); | ||||
|             }); | ||||
|           }, | ||||
|         ), | ||||
|       ], | ||||
|     ); | ||||
|   } | ||||
|   | ||||
							
								
								
									
										126
									
								
								lib/screens/account/account_settings.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,126 @@ | ||||
| import 'package:collection/collection.dart'; | ||||
| import 'package:dropdown_button2/dropdown_button2.dart'; | ||||
| import 'package:easy_localization/easy_localization.dart'; | ||||
| import 'package:flutter/material.dart'; | ||||
| import 'package:go_router/go_router.dart'; | ||||
| import 'package:material_symbols_icons/symbols.dart'; | ||||
| import 'package:provider/provider.dart'; | ||||
| import 'package:styled_widget/styled_widget.dart'; | ||||
| import 'package:surface/providers/sn_network.dart'; | ||||
| import 'package:surface/providers/userinfo.dart'; | ||||
| import 'package:surface/widgets/dialog.dart'; | ||||
| import 'package:surface/widgets/navigation/app_scaffold.dart'; | ||||
| import 'package:intl/locale.dart'; | ||||
|  | ||||
| class AccountSettingsScreen extends StatelessWidget { | ||||
|   const AccountSettingsScreen({super.key}); | ||||
|  | ||||
|   Future<void> _setAccountLanguage(BuildContext context, Locale? value) async { | ||||
|     if (value == null) return; | ||||
|     try { | ||||
|       final sn = context.read<SnNetworkProvider>(); | ||||
|       final ua = context.read<UserProvider>(); | ||||
|       await sn.client.put('/cgi/id/users/me/language', data: { | ||||
|         'language': value.toString(), | ||||
|       }); | ||||
|       if (!context.mounted) return; | ||||
|       context.showSnackbar('accountSettingsApplied'.tr()); | ||||
|       await ua.refreshUser(); | ||||
|     } catch (err) { | ||||
|       if (!context.mounted) return; | ||||
|       context.showErrorDialog(err); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     final ua = context.watch<UserProvider>(); | ||||
|  | ||||
|     return AppScaffold( | ||||
|       appBar: AppBar( | ||||
|         leading: PageBackButton(), | ||||
|         title: Text('screenAccountSettings').tr(), | ||||
|       ), | ||||
|       body: SingleChildScrollView( | ||||
|         child: Column( | ||||
|           crossAxisAlignment: CrossAxisAlignment.start, | ||||
|           children: [ | ||||
|             ListTile( | ||||
|               title: Text('settingsAccountLanguage').tr(), | ||||
|               subtitle: Text('settingsAccountLanguageDescription').tr(), | ||||
|               contentPadding: const EdgeInsets.only(left: 24, right: 17), | ||||
|               leading: const Icon(Symbols.translate), | ||||
|               trailing: DropdownButtonHideUnderline( | ||||
|                 child: DropdownButton2<Locale?>( | ||||
|                   isExpanded: true, | ||||
|                   items: [ | ||||
|                     ...EasyLocalization.of(context)!.supportedLocales.mapIndexed((idx, ele) { | ||||
|                       return DropdownMenuItem<Locale?>( | ||||
|                         value: Locale.parse(ele.toString()), | ||||
|                         child: Text('${ele.languageCode}-${ele.countryCode}').fontSize(14), | ||||
|                       ); | ||||
|                     }), | ||||
|                   ], | ||||
|                   value: ua.user?.language != null ? Locale.parse(ua.user!.language) : Locale.parse('en-US'), | ||||
|                   onChanged: (Locale? value) { | ||||
|                     if (value == null) return; | ||||
|                     _setAccountLanguage(context, value); | ||||
|                     ua.setLanguage(value.toString()); | ||||
|                   }, | ||||
|                   buttonStyleData: const ButtonStyleData( | ||||
|                     padding: EdgeInsets.symmetric( | ||||
|                       horizontal: 16, | ||||
|                       vertical: 5, | ||||
|                     ), | ||||
|                     height: 40, | ||||
|                     width: 160, | ||||
|                   ), | ||||
|                   menuItemStyleData: const MenuItemStyleData( | ||||
|                     height: 40, | ||||
|                   ), | ||||
|                 ), | ||||
|               ), | ||||
|             ), | ||||
|             ListTile( | ||||
|               title: Text('accountProfileEdit').tr(), | ||||
|               subtitle: Text('accountProfileEditSubtitle').tr(), | ||||
|               contentPadding: const EdgeInsets.symmetric(horizontal: 24), | ||||
|               leading: const Icon(Symbols.contact_page), | ||||
|               trailing: const Icon(Symbols.chevron_right), | ||||
|               onTap: () { | ||||
|                 GoRouter.of(context).pushNamed('accountProfileEdit'); | ||||
|               }, | ||||
|             ), | ||||
|             ListTile( | ||||
|               title: Text('accountDeletion'.tr()), | ||||
|               subtitle: Text('accountDeletionActionDescription'.tr()), | ||||
|               contentPadding: const EdgeInsets.symmetric(horizontal: 24), | ||||
|               leading: const Icon(Symbols.person_cancel), | ||||
|               trailing: const Icon(Symbols.chevron_right), | ||||
|               onTap: () { | ||||
|                 context | ||||
|                     .showConfirmDialog( | ||||
|                   'accountDeletion'.tr(), | ||||
|                   'accountDeletionDescription'.tr(), | ||||
|                 ) | ||||
|                     .then((value) { | ||||
|                   if (!value || !context.mounted) return; | ||||
|                   final sn = context.read<SnNetworkProvider>(); | ||||
|                   sn.client.post('/cgi/id/users/me/deletion').then((value) { | ||||
|                     if (context.mounted) { | ||||
|                       context.showSnackbar('accountDeletionSubmitted'.tr()); | ||||
|                     } | ||||
|                   }).catchError((err) { | ||||
|                     if (context.mounted) { | ||||
|                       context.showErrorDialog(err); | ||||
|                     } | ||||
|                   }); | ||||
|                 }); | ||||
|               }, | ||||
|             ), | ||||
|           ], | ||||
|         ), | ||||
|       ), | ||||
|     ); | ||||
|   } | ||||
| } | ||||
							
								
								
									
										294
									
								
								lib/screens/account/factor_settings.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,294 @@ | ||||
| import 'package:dropdown_button2/dropdown_button2.dart'; | ||||
| import 'package:easy_localization/easy_localization.dart'; | ||||
| import 'package:flutter/material.dart'; | ||||
| import 'package:gap/gap.dart'; | ||||
| import 'package:material_symbols_icons/symbols.dart'; | ||||
| import 'package:provider/provider.dart'; | ||||
| import 'package:qr_flutter/qr_flutter.dart'; | ||||
| import 'package:styled_widget/styled_widget.dart'; | ||||
| import 'package:surface/providers/sn_network.dart'; | ||||
| import 'package:surface/types/auth.dart'; | ||||
| import 'package:surface/widgets/dialog.dart'; | ||||
| import 'package:surface/widgets/loading_indicator.dart'; | ||||
| import 'package:surface/widgets/navigation/app_scaffold.dart'; | ||||
|  | ||||
| final Map<int, (String, String, IconData)> kFactorTypes = { | ||||
|   0: ('authFactorPassword', 'authFactorPasswordDescription', Symbols.password), | ||||
|   1: ('authFactorEmail', 'authFactorEmailDescription', Symbols.email), | ||||
|   2: ('authFactorTOTP', 'authFactorTOTPDescription', Symbols.timer), | ||||
|   3: ('authFactorInAppNotify', 'authFactorInAppNotifyDescription', Symbols.notifications_active), | ||||
| }; | ||||
|  | ||||
| class FactorSettingsScreen extends StatefulWidget { | ||||
|   const FactorSettingsScreen({super.key}); | ||||
|  | ||||
|   @override | ||||
|   State<FactorSettingsScreen> createState() => _FactorSettingsScreenState(); | ||||
| } | ||||
|  | ||||
| class _FactorSettingsScreenState extends State<FactorSettingsScreen> { | ||||
|   bool _isBusy = false; | ||||
|   List<SnAuthFactor>? _factors; | ||||
|  | ||||
|   Future<void> _fetchFactors() async { | ||||
|     try { | ||||
|       setState(() => _isBusy = true); | ||||
|       final sn = context.read<SnNetworkProvider>(); | ||||
|       final resp = await sn.client.get('/cgi/id/users/me/factors'); | ||||
|       _factors = List<SnAuthFactor>.from( | ||||
|         resp.data?.map((e) => SnAuthFactor.fromJson(e as Map<String, dynamic>)).toList() ?? [], | ||||
|       ); | ||||
|     } catch (err) { | ||||
|       if (!mounted) return; | ||||
|       context.showErrorDialog(err); | ||||
|     } finally { | ||||
|       setState(() => _isBusy = false); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   void initState() { | ||||
|     super.initState(); | ||||
|     _fetchFactors(); | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     return AppScaffold( | ||||
|       appBar: AppBar( | ||||
|         leading: PageBackButton(), | ||||
|         title: Text('screenFactorSettings').tr(), | ||||
|       ), | ||||
|       body: Column( | ||||
|         crossAxisAlignment: CrossAxisAlignment.start, | ||||
|         children: [ | ||||
|           LoadingIndicator( | ||||
|             isActive: _isBusy, | ||||
|           ), | ||||
|           ListTile( | ||||
|             title: Text('authFactorAdd').tr(), | ||||
|             subtitle: Text('authFactorAddSubtitle').tr(), | ||||
|             contentPadding: const EdgeInsets.symmetric(horizontal: 24), | ||||
|             leading: const Icon(Symbols.add), | ||||
|             trailing: const Icon(Symbols.chevron_right), | ||||
|             onTap: () { | ||||
|               showDialog( | ||||
|                 context: context, | ||||
|                 builder: (context) => _FactorNewDialog( | ||||
|                   currentlyHave: _factors!, | ||||
|                 ), | ||||
|               ).then((val) { | ||||
|                 if (val == true) _fetchFactors(); | ||||
|               }); | ||||
|             }, | ||||
|           ), | ||||
|           const Divider(height: 1), | ||||
|           Expanded( | ||||
|             child: MediaQuery.removePadding( | ||||
|               context: context, | ||||
|               removeTop: true, | ||||
|               child: RefreshIndicator( | ||||
|                 onRefresh: _fetchFactors, | ||||
|                 child: ListView.builder( | ||||
|                   itemCount: _factors?.length ?? 0, | ||||
|                   itemBuilder: (context, idx) { | ||||
|                     final ele = _factors![idx]; | ||||
|                     return ListTile( | ||||
|                       title: Text(kFactorTypes[ele.type]!.$1).tr(), | ||||
|                       subtitle: Text(kFactorTypes[ele.type]!.$2).tr(), | ||||
|                       contentPadding: const EdgeInsets.only(left: 24, right: 12), | ||||
|                       leading: Icon(kFactorTypes[ele.type]!.$3), | ||||
|                       trailing: IconButton( | ||||
|                         icon: const Icon(Symbols.close), | ||||
|                         onPressed: ele.type > 0 | ||||
|                             ? () { | ||||
|                                 context | ||||
|                                     .showConfirmDialog( | ||||
|                                   'authFactorDelete'.tr(), | ||||
|                                   'authFactorDeleteDescription'.tr(args: [kFactorTypes[ele.type]!.$1.tr()]), | ||||
|                                 ) | ||||
|                                     .then((val) async { | ||||
|                                   if (!val) return; | ||||
|                                   try { | ||||
|                                     if (!context.mounted) return; | ||||
|                                     final sn = context.read<SnNetworkProvider>(); | ||||
|                                     await sn.client.delete('/cgi/id/users/me/factors/${ele.id}'); | ||||
|                                     _fetchFactors(); | ||||
|                                   } catch (err) { | ||||
|                                     if (!context.mounted) return; | ||||
|                                     context.showErrorDialog(err); | ||||
|                                   } | ||||
|                                 }); | ||||
|                               } | ||||
|                             : null, | ||||
|                       ), | ||||
|                     ); | ||||
|                   }, | ||||
|                 ), | ||||
|               ), | ||||
|             ), | ||||
|           ), | ||||
|         ], | ||||
|       ), | ||||
|     ); | ||||
|   } | ||||
| } | ||||
|  | ||||
| class _FactorNewDialog extends StatefulWidget { | ||||
|   final List<SnAuthFactor> currentlyHave; | ||||
|  | ||||
|   const _FactorNewDialog({required this.currentlyHave}); | ||||
|  | ||||
|   @override | ||||
|   State<_FactorNewDialog> createState() => _FactorNewDialogState(); | ||||
| } | ||||
|  | ||||
| class _FactorNewDialogState extends State<_FactorNewDialog> { | ||||
|   int? _factorType; | ||||
|   bool _isBusy = false; | ||||
|  | ||||
|   Future<void> _submit() async { | ||||
|     try { | ||||
|       setState(() => _isBusy = true); | ||||
|       final sn = context.read<SnNetworkProvider>(); | ||||
|       final resp = await sn.client.post('/cgi/id/users/me/factors', data: { | ||||
|         'type': _factorType, | ||||
|       }); | ||||
|       final factor = SnAuthFactor.fromJson(resp.data); | ||||
|       if (!mounted) return; | ||||
|       if (factor.type == 2) { | ||||
|         await showModalBottomSheet( | ||||
|           context: context, | ||||
|           builder: (context) => _FactorTotpFactorDialog(factor: factor), | ||||
|         ); | ||||
|       } | ||||
|       if (!mounted) return; | ||||
|       Navigator.of(context).pop(true); | ||||
|     } catch (err) { | ||||
|       if (!mounted) return; | ||||
|       context.showErrorDialog(err); | ||||
|     } finally { | ||||
|       setState(() => _isBusy = false); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     return AlertDialog( | ||||
|       title: Text('authFactorAdd').tr(), | ||||
|       content: Column( | ||||
|         mainAxisSize: MainAxisSize.min, | ||||
|         children: [ | ||||
|           DropdownButtonHideUnderline( | ||||
|             child: DropdownButton2<int>( | ||||
|               hint: Text( | ||||
|                 'Select Item', | ||||
|                 style: TextStyle( | ||||
|                   fontSize: 14, | ||||
|                 ), | ||||
|                 overflow: TextOverflow.ellipsis, | ||||
|               ), | ||||
|               value: _factorType, | ||||
|               items: kFactorTypes.entries.map( | ||||
|                 (ele) { | ||||
|                   final contains = widget.currentlyHave.map((ele) => ele.type).contains(ele.key); | ||||
|                   return DropdownMenuItem<int>( | ||||
|                     enabled: !contains, | ||||
|                     value: ele.key, | ||||
|                     child: Text( | ||||
|                       ele.value.$1.tr(), | ||||
|                       style: const TextStyle( | ||||
|                         fontSize: 14, | ||||
|                       ), | ||||
|                     ).opacity(contains ? 0.75 : 1), | ||||
|                   ); | ||||
|                 }, | ||||
|               ).toList(), | ||||
|               onChanged: (val) => setState(() { | ||||
|                 _factorType = val; | ||||
|               }), | ||||
|               buttonStyleData: ButtonStyleData( | ||||
|                 height: 50, | ||||
|                 padding: const EdgeInsets.only(left: 14, right: 14), | ||||
|                 decoration: BoxDecoration( | ||||
|                   borderRadius: BorderRadius.circular(14), | ||||
|                   border: Border.all( | ||||
|                     color: Theme.of(context).dividerColor, | ||||
|                   ), | ||||
|                 ), | ||||
|               ), | ||||
|             ), | ||||
|           ), | ||||
|         ], | ||||
|       ), | ||||
|       actions: [ | ||||
|         TextButton( | ||||
|           onPressed: _isBusy ? null : () => Navigator.of(context).pop(), | ||||
|           child: Text('dialogCancel').tr(), | ||||
|         ), | ||||
|         TextButton( | ||||
|           onPressed: _isBusy ? null : () => _submit(), | ||||
|           child: Text('dialogConfirm').tr(), | ||||
|         ), | ||||
|       ], | ||||
|     ); | ||||
|   } | ||||
| } | ||||
|  | ||||
| class _FactorTotpFactorDialog extends StatelessWidget { | ||||
|   final SnAuthFactor factor; | ||||
|  | ||||
|   const _FactorTotpFactorDialog({required this.factor}); | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     return Center( | ||||
|       child: Column( | ||||
|         mainAxisSize: MainAxisSize.min, | ||||
|         crossAxisAlignment: CrossAxisAlignment.center, | ||||
|         children: [ | ||||
|           Center( | ||||
|             child: Text( | ||||
|               'totpPostSetup', | ||||
|               textAlign: TextAlign.center, | ||||
|               style: Theme.of(context).textTheme.titleLarge, | ||||
|             ).tr().width(280), | ||||
|           ), | ||||
|           const Gap(4), | ||||
|           Center( | ||||
|             child: Text( | ||||
|               'totpPostSetupDescription', | ||||
|               textAlign: TextAlign.center, | ||||
|               style: Theme.of(context).textTheme.bodySmall, | ||||
|             ).tr().width(280), | ||||
|           ), | ||||
|           const Gap(16), | ||||
|           QrImageView( | ||||
|             padding: EdgeInsets.zero, | ||||
|             data: factor.config!['url'], | ||||
|             errorCorrectionLevel: QrErrorCorrectLevel.H, | ||||
|             version: QrVersions.auto, | ||||
|             size: 160, | ||||
|             gapless: true, | ||||
|             eyeStyle: QrEyeStyle( | ||||
|               eyeShape: QrEyeShape.circle, | ||||
|               color: Theme.of(context).colorScheme.onSurface, | ||||
|             ), | ||||
|             dataModuleStyle: QrDataModuleStyle( | ||||
|               dataModuleShape: QrDataModuleShape.square, | ||||
|               color: Theme.of(context).colorScheme.onSurface, | ||||
|             ), | ||||
|           ), | ||||
|           const Gap(16), | ||||
|           Center( | ||||
|             child: Text( | ||||
|               'totpNeverShare', | ||||
|               textAlign: TextAlign.center, | ||||
|               style: Theme.of(context).textTheme.bodyMedium, | ||||
|             ).tr().bold().width(280), | ||||
|           ), | ||||
|         ], | ||||
|       ), | ||||
|     ); | ||||
|   } | ||||
| } | ||||
| @@ -18,6 +18,7 @@ import 'package:surface/providers/userinfo.dart'; | ||||
| import 'package:surface/widgets/account/account_image.dart'; | ||||
| import 'package:surface/widgets/dialog.dart'; | ||||
| import 'package:surface/widgets/loading_indicator.dart'; | ||||
| import 'package:surface/widgets/navigation/app_scaffold.dart'; | ||||
| import 'package:surface/widgets/universal_image.dart'; | ||||
|  | ||||
| class ProfileEditScreen extends StatefulWidget { | ||||
| @@ -81,8 +82,7 @@ class _ProfileEditScreenState extends State<ProfileEditScreen> { | ||||
|             onDateTimeChanged: (DateTime newDate) { | ||||
|               setState(() { | ||||
|                 _birthday = newDate; | ||||
|                 _birthdayController.text = | ||||
|                     DateFormat(_kDateFormat).format(_birthday!); | ||||
|                 _birthdayController.text = DateFormat(_kDateFormat).format(_birthday!); | ||||
|               }); | ||||
|             }, | ||||
|           ), | ||||
| @@ -96,11 +96,9 @@ class _ProfileEditScreenState extends State<ProfileEditScreen> { | ||||
|     if (image == null) return; | ||||
|     if (!mounted) return; | ||||
|  | ||||
|     final ImageProvider imageProvider = | ||||
|         kIsWeb ? NetworkImage(image.path) : FileImage(File(image.path)); | ||||
|     final aspectRatios = place == 'banner' | ||||
|         ? [CropAspectRatio(width: 16, height: 7)] | ||||
|         : [CropAspectRatio(width: 1, height: 1)]; | ||||
|     final ImageProvider imageProvider = kIsWeb ? NetworkImage(image.path) : FileImage(File(image.path)); | ||||
|     final aspectRatios = | ||||
|         place == 'banner' ? [CropAspectRatio(width: 16, height: 7)] : [CropAspectRatio(width: 1, height: 1)]; | ||||
|     final result = (!kIsWeb && (Platform.isIOS || Platform.isMacOS)) | ||||
|         ? await showCupertinoImageCropper( | ||||
|             // ignore: use_build_context_synchronously | ||||
| @@ -122,10 +120,7 @@ class _ProfileEditScreenState extends State<ProfileEditScreen> { | ||||
|  | ||||
|     setState(() => _isBusy = true); | ||||
|  | ||||
|     final rawBytes = | ||||
|         (await result.uiImage.toByteData(format: ImageByteFormat.png))! | ||||
|             .buffer | ||||
|             .asUint8List(); | ||||
|     final rawBytes = (await result.uiImage.toByteData(format: ImageByteFormat.png))!.buffer.asUint8List(); | ||||
|  | ||||
|     try { | ||||
|       final attachment = await attach.directUploadOne( | ||||
| @@ -212,136 +207,141 @@ class _ProfileEditScreenState extends State<ProfileEditScreen> { | ||||
|  | ||||
|     final sn = context.read<SnNetworkProvider>(); | ||||
|  | ||||
|     return SingleChildScrollView( | ||||
|       child: Column( | ||||
|         crossAxisAlignment: CrossAxisAlignment.start, | ||||
|         children: [ | ||||
|           LoadingIndicator(isActive: _isBusy), | ||||
|           const Gap(24), | ||||
|           Stack( | ||||
|             clipBehavior: Clip.none, | ||||
|             children: [ | ||||
|               Material( | ||||
|                 elevation: 0, | ||||
|                 child: InkWell( | ||||
|                   child: ClipRRect( | ||||
|                     borderRadius: const BorderRadius.all(Radius.circular(8)), | ||||
|                     child: AspectRatio( | ||||
|                       aspectRatio: 16 / 9, | ||||
|                       child: Container( | ||||
|                         color: | ||||
|                             Theme.of(context).colorScheme.surfaceContainerHigh, | ||||
|                         child: _banner != null | ||||
|                             ? AutoResizeUniversalImage( | ||||
|                                 sn.getAttachmentUrl(_banner!), | ||||
|                                 fit: BoxFit.cover, | ||||
|                               ) | ||||
|                             : const SizedBox.shrink(), | ||||
|     return AppScaffold( | ||||
|       appBar: AppBar( | ||||
|         leading: const PageBackButton(), | ||||
|         title: Text('screenAccountProfileEdit').tr(), | ||||
|       ), | ||||
|       body: SingleChildScrollView( | ||||
|         child: Column( | ||||
|           crossAxisAlignment: CrossAxisAlignment.start, | ||||
|           children: [ | ||||
|             LoadingIndicator(isActive: _isBusy), | ||||
|             const Gap(24), | ||||
|             Stack( | ||||
|               clipBehavior: Clip.none, | ||||
|               children: [ | ||||
|                 Material( | ||||
|                   elevation: 0, | ||||
|                   child: InkWell( | ||||
|                     child: ClipRRect( | ||||
|                       borderRadius: const BorderRadius.all(Radius.circular(8)), | ||||
|                       child: AspectRatio( | ||||
|                         aspectRatio: 16 / 9, | ||||
|                         child: Container( | ||||
|                           color: Theme.of(context).colorScheme.surfaceContainerHigh, | ||||
|                           child: _banner != null | ||||
|                               ? AutoResizeUniversalImage( | ||||
|                                   sn.getAttachmentUrl(_banner!), | ||||
|                                   fit: BoxFit.cover, | ||||
|                                 ) | ||||
|                               : const SizedBox.shrink(), | ||||
|                         ), | ||||
|                       ), | ||||
|                     ), | ||||
|                   ), | ||||
|                   onTap: () { | ||||
|                     _updateImage('banner'); | ||||
|                   }, | ||||
|                 ), | ||||
|               ), | ||||
|               Positioned( | ||||
|                 bottom: -28, | ||||
|                 left: 16, | ||||
|                 child: Material( | ||||
|                   elevation: 2, | ||||
|                   borderRadius: const BorderRadius.all(Radius.circular(40)), | ||||
|                   child: InkWell( | ||||
|                     child: AccountImage(content: _avatar, radius: 40), | ||||
|                     onTap: () { | ||||
|                       _updateImage('avatar'); | ||||
|                       _updateImage('banner'); | ||||
|                     }, | ||||
|                   ), | ||||
|                 ), | ||||
|               ), | ||||
|             ], | ||||
|           ).padding(horizontal: padding), | ||||
|           const Gap(8 + 28), | ||||
|           Column( | ||||
|             children: [ | ||||
|               TextField( | ||||
|                 readOnly: true, | ||||
|                 controller: _usernameController, | ||||
|                 decoration: InputDecoration( | ||||
|                   border: const UnderlineInputBorder(), | ||||
|                   labelText: 'fieldUsername'.tr(), | ||||
|                   helperText: 'fieldUsernameCannotEditHint'.tr(), | ||||
|                 ), | ||||
|               ), | ||||
|               const Gap(4), | ||||
|               TextField( | ||||
|                 controller: _nicknameController, | ||||
|                 decoration: InputDecoration( | ||||
|                   border: const UnderlineInputBorder(), | ||||
|                   labelText: 'fieldNickname'.tr(), | ||||
|                 ), | ||||
|               ), | ||||
|               const Gap(4), | ||||
|               Row( | ||||
|                 children: [ | ||||
|                   Flexible( | ||||
|                     flex: 1, | ||||
|                     child: TextField( | ||||
|                       controller: _firstNameController, | ||||
|                       decoration: InputDecoration( | ||||
|                         border: const UnderlineInputBorder(), | ||||
|                         labelText: 'fieldFirstName'.tr(), | ||||
|                       ), | ||||
|                 Positioned( | ||||
|                   bottom: -28, | ||||
|                   left: 16, | ||||
|                   child: Material( | ||||
|                     elevation: 2, | ||||
|                     borderRadius: const BorderRadius.all(Radius.circular(40)), | ||||
|                     child: InkWell( | ||||
|                       child: AccountImage(content: _avatar, radius: 40), | ||||
|                       onTap: () { | ||||
|                         _updateImage('avatar'); | ||||
|                       }, | ||||
|                     ), | ||||
|                   ), | ||||
|                   const Gap(8), | ||||
|                   Flexible( | ||||
|                     flex: 1, | ||||
|                     child: TextField( | ||||
|                       controller: _lastNameController, | ||||
|                       decoration: InputDecoration( | ||||
|                         border: const UnderlineInputBorder(), | ||||
|                         labelText: 'fieldLastName'.tr(), | ||||
|                 ), | ||||
|               ], | ||||
|             ).padding(horizontal: padding), | ||||
|             const Gap(8 + 28), | ||||
|             Column( | ||||
|               children: [ | ||||
|                 TextField( | ||||
|                   readOnly: true, | ||||
|                   controller: _usernameController, | ||||
|                   decoration: InputDecoration( | ||||
|                     border: const UnderlineInputBorder(), | ||||
|                     labelText: 'fieldUsername'.tr(), | ||||
|                     helperText: 'fieldUsernameCannotEditHint'.tr(), | ||||
|                   ), | ||||
|                 ), | ||||
|                 const Gap(4), | ||||
|                 TextField( | ||||
|                   controller: _nicknameController, | ||||
|                   decoration: InputDecoration( | ||||
|                     border: const UnderlineInputBorder(), | ||||
|                     labelText: 'fieldNickname'.tr(), | ||||
|                   ), | ||||
|                 ), | ||||
|                 const Gap(4), | ||||
|                 Row( | ||||
|                   children: [ | ||||
|                     Flexible( | ||||
|                       flex: 1, | ||||
|                       child: TextField( | ||||
|                         controller: _firstNameController, | ||||
|                         decoration: InputDecoration( | ||||
|                           border: const UnderlineInputBorder(), | ||||
|                           labelText: 'fieldFirstName'.tr(), | ||||
|                         ), | ||||
|                       ), | ||||
|                     ), | ||||
|                     const Gap(8), | ||||
|                     Flexible( | ||||
|                       flex: 1, | ||||
|                       child: TextField( | ||||
|                         controller: _lastNameController, | ||||
|                         decoration: InputDecoration( | ||||
|                           border: const UnderlineInputBorder(), | ||||
|                           labelText: 'fieldLastName'.tr(), | ||||
|                         ), | ||||
|                       ), | ||||
|                     ), | ||||
|                   ], | ||||
|                 ), | ||||
|                 const Gap(4), | ||||
|                 TextField( | ||||
|                   controller: _descriptionController, | ||||
|                   keyboardType: TextInputType.multiline, | ||||
|                   maxLines: null, | ||||
|                   minLines: 3, | ||||
|                   decoration: InputDecoration( | ||||
|                     border: const UnderlineInputBorder(), | ||||
|                     labelText: 'fieldDescription'.tr(), | ||||
|                   ), | ||||
|                 ], | ||||
|               ), | ||||
|               const Gap(4), | ||||
|               TextField( | ||||
|                 controller: _descriptionController, | ||||
|                 keyboardType: TextInputType.multiline, | ||||
|                 maxLines: null, | ||||
|                 minLines: 3, | ||||
|                 decoration: InputDecoration( | ||||
|                   border: const UnderlineInputBorder(), | ||||
|                   labelText: 'fieldDescription'.tr(), | ||||
|                 ), | ||||
|               ), | ||||
|               const Gap(4), | ||||
|               TextField( | ||||
|                 controller: _birthdayController, | ||||
|                 readOnly: true, | ||||
|                 decoration: InputDecoration( | ||||
|                   border: const UnderlineInputBorder(), | ||||
|                   labelText: 'fieldBirthday'.tr(), | ||||
|                 const Gap(4), | ||||
|                 TextField( | ||||
|                   controller: _birthdayController, | ||||
|                   readOnly: true, | ||||
|                   decoration: InputDecoration( | ||||
|                     border: const UnderlineInputBorder(), | ||||
|                     labelText: 'fieldBirthday'.tr(), | ||||
|                   ), | ||||
|                   onTap: () => _selectBirthday(), | ||||
|                 ), | ||||
|                 onTap: () => _selectBirthday(), | ||||
|               ), | ||||
|             ], | ||||
|           ).padding(horizontal: padding + 8), | ||||
|           const Gap(12), | ||||
|           Row( | ||||
|             mainAxisAlignment: MainAxisAlignment.end, | ||||
|             children: [ | ||||
|               ElevatedButton.icon( | ||||
|                 onPressed: _isBusy ? null : _updateUserInfo, | ||||
|                 icon: const Icon(Symbols.save), | ||||
|                 label: Text('apply').tr(), | ||||
|               ), | ||||
|             ], | ||||
|           ).padding(horizontal: padding), | ||||
|         ], | ||||
|               ], | ||||
|             ).padding(horizontal: padding + 8), | ||||
|             const Gap(12), | ||||
|             Row( | ||||
|               mainAxisAlignment: MainAxisAlignment.end, | ||||
|               children: [ | ||||
|                 ElevatedButton.icon( | ||||
|                   onPressed: _isBusy ? null : _updateUserInfo, | ||||
|                   icon: const Icon(Symbols.save), | ||||
|                   label: Text('apply').tr(), | ||||
|                 ), | ||||
|               ], | ||||
|             ).padding(horizontal: padding), | ||||
|           ], | ||||
|         ), | ||||
|       ), | ||||
|     ); | ||||
|   } | ||||
|   | ||||
| @@ -1,6 +1,7 @@ | ||||
| import 'dart:ui'; | ||||
| 
 | ||||
| import 'package:easy_localization/easy_localization.dart'; | ||||
| import 'package:fl_chart/fl_chart.dart'; | ||||
| import 'package:flutter/material.dart'; | ||||
| import 'package:gap/gap.dart'; | ||||
| import 'package:go_router/go_router.dart'; | ||||
| @@ -9,10 +10,12 @@ import 'package:material_symbols_icons/symbols.dart'; | ||||
| import 'package:provider/provider.dart'; | ||||
| import 'package:relative_time/relative_time.dart'; | ||||
| import 'package:styled_widget/styled_widget.dart'; | ||||
| import 'package:surface/providers/experience.dart'; | ||||
| import 'package:surface/providers/relationship.dart'; | ||||
| import 'package:surface/providers/sn_network.dart'; | ||||
| import 'package:surface/screens/abuse_report.dart'; | ||||
| import 'package:surface/types/account.dart'; | ||||
| import 'package:surface/types/check_in.dart'; | ||||
| import 'package:surface/types/post.dart'; | ||||
| import 'package:surface/widgets/account/account_image.dart'; | ||||
| import 'package:surface/widgets/dialog.dart'; | ||||
| @@ -61,6 +64,19 @@ class _UserScreenState extends State<UserScreen> with SingleTickerProviderStateM | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   Future<List<SnCheckInRecord>> _getCheckInRecords() async { | ||||
|     try { | ||||
|       final sn = context.read<SnNetworkProvider>(); | ||||
|       final resp = await sn.client.get('/cgi/id/users/${widget.name}/check-in?take=14'); | ||||
|       return List.from( | ||||
|         resp.data['data']?.map((x) => SnCheckInRecord.fromJson(x)) ?? [], | ||||
|       ); | ||||
|     } catch (err) { | ||||
|       if (mounted) context.showErrorDialog(err); | ||||
|       rethrow; | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   SnAccountStatusInfo? _status; | ||||
| 
 | ||||
|   Future<void> _fetchStatus() async { | ||||
| @@ -225,68 +241,76 @@ class _UserScreenState extends State<UserScreen> with SingleTickerProviderStateM | ||||
|     final sn = context.read<SnNetworkProvider>(); | ||||
| 
 | ||||
|     return Scaffold( | ||||
|       backgroundColor: Colors.transparent, | ||||
|       body: CustomScrollView( | ||||
|         controller: _scrollController, | ||||
|         slivers: [ | ||||
|           SliverAppBar( | ||||
|             expandedHeight: _appBarHeight, | ||||
|             title: _account == null | ||||
|                 ? Text('loading').tr() | ||||
|                 : RichText( | ||||
|                     textAlign: TextAlign.center, | ||||
|                     text: TextSpan(children: [ | ||||
|                       TextSpan( | ||||
|                         text: _account!.nick, | ||||
|                         style: Theme.of(context).textTheme.titleLarge!.copyWith( | ||||
|                               color: Theme.of(context).appBarTheme.foregroundColor!, | ||||
|                               shadows: labelShadows, | ||||
|                             ), | ||||
|                       ), | ||||
|                       const TextSpan(text: '\n'), | ||||
|                       TextSpan( | ||||
|                         text: '@${_account!.name}', | ||||
|                         style: Theme.of(context).textTheme.bodySmall!.copyWith( | ||||
|                               color: Theme.of(context).appBarTheme.foregroundColor!, | ||||
|                               shadows: labelShadows, | ||||
|                             ), | ||||
|                       ), | ||||
|                     ]), | ||||
|           Theme( | ||||
|             data: Theme.of(context).copyWith( | ||||
|               appBarTheme: Theme.of(context).appBarTheme.copyWith( | ||||
|                     foregroundColor: Colors.white, | ||||
|                   ), | ||||
|             pinned: true, | ||||
|             flexibleSpace: _account != null | ||||
|                 ? Stack( | ||||
|                     fit: StackFit.expand, | ||||
|                     children: [ | ||||
|                       UniversalImage( | ||||
|                         sn.getAttachmentUrl(_account!.banner), | ||||
|                         fit: BoxFit.cover, | ||||
|                         height: imageHeight, | ||||
|                         width: _appBarWidth, | ||||
|                         cacheHeight: imageHeight, | ||||
|                         cacheWidth: _appBarWidth, | ||||
|                       ), | ||||
|                       Positioned( | ||||
|                         top: 0, | ||||
|                         left: 0, | ||||
|                         right: 0, | ||||
|                         height: 56 + MediaQuery.of(context).padding.top, | ||||
|                         child: ClipRect( | ||||
|                           child: BackdropFilter( | ||||
|                             filter: ImageFilter.blur( | ||||
|                               sigmaX: _appBarBlur, | ||||
|                               sigmaY: _appBarBlur, | ||||
|                             ), | ||||
|                             child: Container( | ||||
|                               color: Colors.black.withOpacity( | ||||
|                                 clampDouble(_appBarBlur * 0.1, 0, 0.5), | ||||
|             ), | ||||
|             child: SliverAppBar( | ||||
|               expandedHeight: _appBarHeight, | ||||
|               title: _account == null | ||||
|                   ? Text('loading').tr() | ||||
|                   : RichText( | ||||
|                       textAlign: TextAlign.center, | ||||
|                       text: TextSpan(children: [ | ||||
|                         TextSpan( | ||||
|                           text: _account!.nick, | ||||
|                           style: Theme.of(context).textTheme.titleLarge!.copyWith( | ||||
|                                 color: Colors.white, | ||||
|                                 shadows: labelShadows, | ||||
|                               ), | ||||
|                         ), | ||||
|                         const TextSpan(text: '\n'), | ||||
|                         TextSpan( | ||||
|                           text: '@${_account!.name}', | ||||
|                           style: Theme.of(context).textTheme.bodySmall!.copyWith( | ||||
|                                 color: Colors.white, | ||||
|                                 shadows: labelShadows, | ||||
|                               ), | ||||
|                         ), | ||||
|                       ]), | ||||
|                     ), | ||||
|               pinned: true, | ||||
|               flexibleSpace: _account != null | ||||
|                   ? Stack( | ||||
|                       fit: StackFit.expand, | ||||
|                       children: [ | ||||
|                         UniversalImage( | ||||
|                           sn.getAttachmentUrl(_account!.banner), | ||||
|                           fit: BoxFit.cover, | ||||
|                           height: imageHeight, | ||||
|                           width: _appBarWidth, | ||||
|                           cacheHeight: imageHeight, | ||||
|                           cacheWidth: _appBarWidth, | ||||
|                         ), | ||||
|                         Positioned( | ||||
|                           top: 0, | ||||
|                           left: 0, | ||||
|                           right: 0, | ||||
|                           height: 56 + MediaQuery.of(context).padding.top, | ||||
|                           child: ClipRect( | ||||
|                             child: BackdropFilter( | ||||
|                               filter: ImageFilter.blur( | ||||
|                                 sigmaX: _appBarBlur, | ||||
|                                 sigmaY: _appBarBlur, | ||||
|                               ), | ||||
|                               child: Container( | ||||
|                                 color: Colors.black.withOpacity( | ||||
|                                   clampDouble(_appBarBlur * 0.1, 0, 0.5), | ||||
|                                 ), | ||||
|                               ), | ||||
|                             ), | ||||
|                           ), | ||||
|                         ), | ||||
|                       ), | ||||
|                     ], | ||||
|                   ) | ||||
|                 : null, | ||||
|                       ], | ||||
|                     ) | ||||
|                   : null, | ||||
|             ), | ||||
|           ), | ||||
|           if (_account != null) | ||||
|             SliverToBoxAdapter( | ||||
| @@ -430,6 +454,7 @@ class _UserScreenState extends State<UserScreen> with SingleTickerProviderStateM | ||||
|                   Column( | ||||
|                     children: [ | ||||
|                       Row( | ||||
|                         crossAxisAlignment: CrossAxisAlignment.center, | ||||
|                         children: [ | ||||
|                           const Icon(Symbols.calendar_add_on), | ||||
|                           const Gap(8), | ||||
| @@ -437,6 +462,7 @@ class _UserScreenState extends State<UserScreen> with SingleTickerProviderStateM | ||||
|                         ], | ||||
|                       ), | ||||
|                       Row( | ||||
|                         crossAxisAlignment: CrossAxisAlignment.center, | ||||
|                         children: [ | ||||
|                           const Icon(Symbols.cake), | ||||
|                           const Gap(8), | ||||
| @@ -450,6 +476,7 @@ class _UserScreenState extends State<UserScreen> with SingleTickerProviderStateM | ||||
|                         ], | ||||
|                       ), | ||||
|                       Row( | ||||
|                         crossAxisAlignment: CrossAxisAlignment.center, | ||||
|                         children: [ | ||||
|                           const Icon(Symbols.identity_platform), | ||||
|                           const Gap(8), | ||||
| @@ -459,6 +486,26 @@ class _UserScreenState extends State<UserScreen> with SingleTickerProviderStateM | ||||
|                           ).opacity(0.8), | ||||
|                         ], | ||||
|                       ), | ||||
|                       Row( | ||||
|                         crossAxisAlignment: CrossAxisAlignment.center, | ||||
|                         children: [ | ||||
|                           const Icon(Symbols.star), | ||||
|                           const Gap(8), | ||||
|                           Text('Lv${getLevelFromExp(_account?.profile?.experience ?? 0)}'), | ||||
|                           const Gap(8), | ||||
|                           Text(calcLevelUpProgressLevel(_account?.profile?.experience ?? 0)).fontSize(11).opacity(0.5), | ||||
|                           const Gap(8), | ||||
|                           Container( | ||||
|                             width: double.infinity, | ||||
|                             constraints: const BoxConstraints(maxWidth: 160), | ||||
|                             child: LinearProgressIndicator( | ||||
|                               value: calcLevelUpProgress(_account?.profile?.experience ?? 0), | ||||
|                               borderRadius: BorderRadius.circular(8), | ||||
|                               backgroundColor: Theme.of(context).colorScheme.surfaceContainer, | ||||
|                             ).alignment(Alignment.centerLeft), | ||||
|                           ), | ||||
|                         ], | ||||
|                       ), | ||||
|                     ], | ||||
|                   ).padding(horizontal: 8), | ||||
|                 ], | ||||
| @@ -466,6 +513,33 @@ class _UserScreenState extends State<UserScreen> with SingleTickerProviderStateM | ||||
|             ), | ||||
|           SliverToBoxAdapter(child: const Divider()), | ||||
|           const SliverGap(12), | ||||
|           SliverToBoxAdapter( | ||||
|             child: FutureBuilder<List<SnCheckInRecord>>( | ||||
|               future: _getCheckInRecords(), | ||||
|               builder: (context, snapshot) { | ||||
|                 if (!snapshot.hasData) return const SizedBox.shrink(); | ||||
|                 if (snapshot.data!.length <= 1) { | ||||
|                   return Text( | ||||
|                     'accountCheckInNoRecords', | ||||
|                     textAlign: TextAlign.center, | ||||
|                   ).tr().fontWeight(FontWeight.bold).center().padding(horizontal: 20, vertical: 8); | ||||
|                 } | ||||
|                 final records = snapshot.data!; | ||||
|                 return SizedBox( | ||||
|                   width: double.infinity, | ||||
|                   height: 240, | ||||
|                   child: CheckInRecordChart(records: records), | ||||
|                 ).padding( | ||||
|                   right: 24, | ||||
|                   left: 16, | ||||
|                   top: 12, | ||||
|                 ); | ||||
|               }, | ||||
|             ), | ||||
|           ), | ||||
|           const SliverGap(12), | ||||
|           SliverToBoxAdapter(child: const Divider()), | ||||
|           const SliverGap(12), | ||||
|           SliverToBoxAdapter( | ||||
|             child: Column( | ||||
|               crossAxisAlignment: CrossAxisAlignment.start, | ||||
| @@ -521,7 +595,7 @@ class _UserScreenState extends State<UserScreen> with SingleTickerProviderStateM | ||||
|                 subtitle: Text('@${ele.name}'), | ||||
|                 trailing: const Icon(Symbols.chevron_right), | ||||
|                 onTap: () { | ||||
|                   GoRouter.of(context).pushNamed( | ||||
|                   GoRouter.of(context).goNamed( | ||||
|                     'postPublisher', | ||||
|                     pathParameters: {'name': ele.name}, | ||||
|                   ); | ||||
| @@ -534,3 +608,105 @@ class _UserScreenState extends State<UserScreen> with SingleTickerProviderStateM | ||||
|     ); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| class CheckInRecordChart extends StatelessWidget { | ||||
|   const CheckInRecordChart({ | ||||
|     super.key, | ||||
|     required this.records, | ||||
|   }); | ||||
| 
 | ||||
|   final List<SnCheckInRecord> records; | ||||
| 
 | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     return LineChart( | ||||
|       LineChartData( | ||||
|         lineBarsData: [ | ||||
|           LineChartBarData( | ||||
|             color: Theme.of(context).colorScheme.primary, | ||||
|             belowBarData: BarAreaData( | ||||
|               show: true, | ||||
|               gradient: LinearGradient( | ||||
|                 colors: List.filled( | ||||
|                   records.length, | ||||
|                   Theme.of(context).colorScheme.primary.withOpacity(0.3), | ||||
|                 ).toList(), | ||||
|               ), | ||||
|             ), | ||||
|             spots: records | ||||
|                 .map( | ||||
|                   (x) => FlSpot( | ||||
|                     x.createdAt | ||||
|                         .copyWith( | ||||
|                           hour: 0, | ||||
|                           minute: 0, | ||||
|                           second: 0, | ||||
|                           millisecond: 0, | ||||
|                           microsecond: 0, | ||||
|                         ) | ||||
|                         .millisecondsSinceEpoch | ||||
|                         .toDouble(), | ||||
|                     x.resultTier.toDouble(), | ||||
|                   ), | ||||
|                 ) | ||||
|                 .toList(), | ||||
|           ) | ||||
|         ], | ||||
|         lineTouchData: LineTouchData( | ||||
|           touchTooltipData: LineTouchTooltipData( | ||||
|             getTooltipItems: (spots) => spots | ||||
|                 .map( | ||||
|                   (spot) => LineTooltipItem( | ||||
|                     '${kCheckInResultTierSymbols[spot.y.toInt()]}\n${DateFormat('MM/dd').format(DateTime.fromMillisecondsSinceEpoch(spot.x.toInt()))}', | ||||
|                     TextStyle( | ||||
|                       color: Theme.of(context).colorScheme.onSurface, | ||||
|                     ), | ||||
|                   ), | ||||
|                 ) | ||||
|                 .toList(), | ||||
|             getTooltipColor: (_) => Theme.of(context).colorScheme.surfaceContainerHigh, | ||||
|           ), | ||||
|         ), | ||||
|         titlesData: FlTitlesData( | ||||
|           topTitles: const AxisTitles( | ||||
|             sideTitles: SideTitles(showTitles: false), | ||||
|           ), | ||||
|           rightTitles: const AxisTitles( | ||||
|             sideTitles: SideTitles(showTitles: false), | ||||
|           ), | ||||
|           leftTitles: AxisTitles( | ||||
|             sideTitles: SideTitles( | ||||
|               showTitles: true, | ||||
|               reservedSize: 40, | ||||
|               interval: 1, | ||||
|               getTitlesWidget: (value, _) => Align( | ||||
|                 alignment: Alignment.centerRight, | ||||
|                 child: Text( | ||||
|                   kCheckInResultTierSymbols[value.toInt()], | ||||
|                   textAlign: TextAlign.right, | ||||
|                 ).padding(right: 8), | ||||
|               ), | ||||
|             ), | ||||
|           ), | ||||
|           bottomTitles: AxisTitles( | ||||
|             sideTitles: SideTitles( | ||||
|               showTitles: true, | ||||
|               reservedSize: 28, | ||||
|               interval: 86400000, | ||||
|               getTitlesWidget: (value, _) => Text( | ||||
|                 DateFormat('dd').format( | ||||
|                   DateTime.fromMillisecondsSinceEpoch( | ||||
|                     value.toInt(), | ||||
|                   ), | ||||
|                 ), | ||||
|                 textAlign: TextAlign.center, | ||||
|               ).padding(top: 8), | ||||
|             ), | ||||
|           ), | ||||
|         ), | ||||
|         gridData: const FlGridData(show: false), | ||||
|         borderData: FlBorderData(show: false), | ||||
|       ), | ||||
|     ); | ||||
|   } | ||||
| } | ||||
| @@ -18,19 +18,19 @@ import 'package:surface/types/post.dart'; | ||||
| import 'package:surface/widgets/account/account_image.dart'; | ||||
| import 'package:surface/widgets/dialog.dart'; | ||||
| import 'package:surface/widgets/loading_indicator.dart'; | ||||
| import 'package:surface/widgets/navigation/app_scaffold.dart'; | ||||
| import 'package:surface/widgets/universal_image.dart'; | ||||
|  | ||||
| class AccountPublisherEditScreen extends StatefulWidget { | ||||
|   final String name; | ||||
|  | ||||
|   const AccountPublisherEditScreen({super.key, required this.name}); | ||||
|  | ||||
|   @override | ||||
|   State<AccountPublisherEditScreen> createState() => | ||||
|       _AccountPublisherEditScreenState(); | ||||
|   State<AccountPublisherEditScreen> createState() => _AccountPublisherEditScreenState(); | ||||
| } | ||||
|  | ||||
| class _AccountPublisherEditScreenState | ||||
|     extends State<AccountPublisherEditScreen> { | ||||
| class _AccountPublisherEditScreenState extends State<AccountPublisherEditScreen> { | ||||
|   bool _isBusy = false; | ||||
|  | ||||
|   SnPublisher? _publisher; | ||||
| @@ -54,7 +54,7 @@ class _AccountPublisherEditScreenState | ||||
|       _publisher = SnPublisher.fromJson(resp.data); | ||||
|       _syncWidget(); | ||||
|     } catch (err) { | ||||
|       context.showErrorDialog(err); | ||||
|       if (mounted) context.showErrorDialog(err); | ||||
|     } finally { | ||||
|       setState(() => _isBusy = false); | ||||
|     } | ||||
| @@ -75,9 +75,9 @@ class _AccountPublisherEditScreenState | ||||
|         'name': _nameController.text, | ||||
|         'description': _descriptionController.text, | ||||
|       }); | ||||
|       Navigator.pop(context, true); | ||||
|       if (mounted) Navigator.pop(context, true); | ||||
|     } catch (err) { | ||||
|       context.showErrorDialog(err); | ||||
|       if(mounted) context.showErrorDialog(err); | ||||
|     } finally { | ||||
|       setState(() => _isBusy = false); | ||||
|     } | ||||
| @@ -108,11 +108,9 @@ class _AccountPublisherEditScreenState | ||||
|     if (image == null) return; | ||||
|     if (!mounted) return; | ||||
|  | ||||
|     final ImageProvider imageProvider = | ||||
|         kIsWeb ? NetworkImage(image.path) : FileImage(File(image.path)); | ||||
|     final aspectRatios = place == 'banner' | ||||
|         ? [CropAspectRatio(width: 16, height: 7)] | ||||
|         : [CropAspectRatio(width: 1, height: 1)]; | ||||
|     final ImageProvider imageProvider = kIsWeb ? NetworkImage(image.path) : FileImage(File(image.path)); | ||||
|     final aspectRatios = | ||||
|         place == 'banner' ? [CropAspectRatio(width: 16, height: 7)] : [CropAspectRatio(width: 1, height: 1)]; | ||||
|     final result = (!kIsWeb && (Platform.isIOS || Platform.isMacOS)) | ||||
|         ? await showCupertinoImageCropper( | ||||
|             // ignore: use_build_context_synchronously | ||||
| @@ -134,10 +132,7 @@ class _AccountPublisherEditScreenState | ||||
|  | ||||
|     setState(() => _isBusy = true); | ||||
|  | ||||
|     final rawBytes = | ||||
|         (await result.uiImage.toByteData(format: ImageByteFormat.png))! | ||||
|             .buffer | ||||
|             .asUint8List(); | ||||
|     final rawBytes = (await result.uiImage.toByteData(format: ImageByteFormat.png))!.buffer.asUint8List(); | ||||
|  | ||||
|     try { | ||||
|       final attachment = await attach.directUploadOne( | ||||
| @@ -182,7 +177,7 @@ class _AccountPublisherEditScreenState | ||||
|   Widget build(BuildContext context) { | ||||
|     final sn = context.read<SnNetworkProvider>(); | ||||
|  | ||||
|     return Scaffold( | ||||
|     return AppScaffold( | ||||
|       body: SingleChildScrollView( | ||||
|         child: Column( | ||||
|           children: [ | ||||
| @@ -199,9 +194,7 @@ class _AccountPublisherEditScreenState | ||||
|                       child: AspectRatio( | ||||
|                         aspectRatio: 16 / 9, | ||||
|                         child: Container( | ||||
|                           color: Theme.of(context) | ||||
|                               .colorScheme | ||||
|                               .surfaceContainerHigh, | ||||
|                           color: Theme.of(context).colorScheme.surfaceContainerHigh, | ||||
|                           child: _banner != null | ||||
|                               ? AutoResizeUniversalImage( | ||||
|                                   sn.getAttachmentUrl(_banner!), | ||||
| @@ -240,8 +233,7 @@ class _AccountPublisherEditScreenState | ||||
|                 labelText: 'fieldUsername'.tr(), | ||||
|                 helperText: 'fieldUsernameCannotEditHint'.tr(), | ||||
|               ), | ||||
|               onTapOutside: (_) => | ||||
|                   FocusManager.instance.primaryFocus?.unfocus(), | ||||
|               onTapOutside: (_) => FocusManager.instance.primaryFocus?.unfocus(), | ||||
|             ), | ||||
|             const Gap(4), | ||||
|             TextField( | ||||
| @@ -249,8 +241,7 @@ class _AccountPublisherEditScreenState | ||||
|               decoration: InputDecoration( | ||||
|                 labelText: 'fieldNickname'.tr(), | ||||
|               ), | ||||
|               onTapOutside: (_) => | ||||
|                   FocusManager.instance.primaryFocus?.unfocus(), | ||||
|               onTapOutside: (_) => FocusManager.instance.primaryFocus?.unfocus(), | ||||
|             ), | ||||
|             const Gap(4), | ||||
|             TextField( | ||||
| @@ -260,8 +251,7 @@ class _AccountPublisherEditScreenState | ||||
|               decoration: InputDecoration( | ||||
|                 labelText: 'fieldDescription'.tr(), | ||||
|               ), | ||||
|               onTapOutside: (_) => | ||||
|                   FocusManager.instance.primaryFocus?.unfocus(), | ||||
|               onTapOutside: (_) => FocusManager.instance.primaryFocus?.unfocus(), | ||||
|             ), | ||||
|             const Gap(12), | ||||
|             Row( | ||||
|   | ||||
| @@ -10,6 +10,7 @@ import 'package:surface/providers/userinfo.dart'; | ||||
| import 'package:surface/types/realm.dart'; | ||||
| import 'package:surface/widgets/account/account_image.dart'; | ||||
| import 'package:surface/widgets/dialog.dart'; | ||||
| import 'package:surface/widgets/navigation/app_scaffold.dart'; | ||||
|  | ||||
| class AccountPublisherNewScreen extends StatefulWidget { | ||||
|   const AccountPublisherNewScreen({super.key}); | ||||
| @@ -24,7 +25,11 @@ class _AccountPublisherNewScreenState extends State<AccountPublisherNewScreen> { | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     return Scaffold( | ||||
|     return  AppScaffold( | ||||
|       appBar: AppBar( | ||||
|         leading: const PageBackButton(), | ||||
|         title: Text('screenAccountPublisherNew').tr(), | ||||
|       ), | ||||
|       body: SingleChildScrollView( | ||||
|         child: Column( | ||||
|           children: [ | ||||
| @@ -201,7 +206,7 @@ class _PublisherNewPersonalState extends State<_PublisherNewPersonal> { | ||||
| } | ||||
|  | ||||
| class _PublisherNewOrganization extends StatefulWidget { | ||||
|   const _PublisherNewOrganization({super.key}); | ||||
|   const _PublisherNewOrganization(); | ||||
|  | ||||
|   @override | ||||
|   State<_PublisherNewOrganization> createState() => | ||||
|   | ||||
| @@ -10,6 +10,7 @@ import 'package:surface/types/post.dart'; | ||||
| import 'package:surface/widgets/account/account_image.dart'; | ||||
| import 'package:surface/widgets/dialog.dart'; | ||||
| import 'package:surface/widgets/loading_indicator.dart'; | ||||
| import 'package:surface/widgets/navigation/app_scaffold.dart'; | ||||
|  | ||||
| class PublisherScreen extends StatefulWidget { | ||||
|   const PublisherScreen({super.key}); | ||||
| @@ -32,8 +33,7 @@ class _PublisherScreenState extends State<PublisherScreen> { | ||||
|  | ||||
|     try { | ||||
|       final resp = await sn.client.get('/cgi/co/publishers/me'); | ||||
|       final List<SnPublisher> out = List<SnPublisher>.from( | ||||
|           resp.data?.map((e) => SnPublisher.fromJson(e)) ?? []); | ||||
|       final List<SnPublisher> out = List<SnPublisher>.from(resp.data?.map((e) => SnPublisher.fromJson(e)) ?? []); | ||||
|  | ||||
|       if (!mounted) return; | ||||
|  | ||||
| @@ -53,7 +53,11 @@ class _PublisherScreenState extends State<PublisherScreen> { | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     return Scaffold( | ||||
|     return AppScaffold( | ||||
|       appBar: AppBar( | ||||
|         leading: const PageBackButton(), | ||||
|         title: Text('screenAccountPublishers').tr(), | ||||
|       ), | ||||
|       body: Column( | ||||
|         children: [ | ||||
|           ListTile( | ||||
| @@ -62,9 +66,7 @@ class _PublisherScreenState extends State<PublisherScreen> { | ||||
|             contentPadding: const EdgeInsets.symmetric(horizontal: 24), | ||||
|             leading: const Icon(Symbols.add_circle), | ||||
|             onTap: () { | ||||
|               GoRouter.of(context) | ||||
|                   .pushNamed('accountPublisherNew') | ||||
|                   .then((value) { | ||||
|               GoRouter.of(context).pushNamed('accountPublisherNew').then((value) { | ||||
|                 if (value == true) { | ||||
|                   _publishers.clear(); | ||||
|                   _fetchPublishers(); | ||||
| @@ -75,48 +77,52 @@ class _PublisherScreenState extends State<PublisherScreen> { | ||||
|           const Divider(height: 1), | ||||
|           LoadingIndicator(isActive: _isBusy), | ||||
|           Expanded( | ||||
|             child: RefreshIndicator( | ||||
|               onRefresh: () { | ||||
|                 _publishers.clear(); | ||||
|                 return _fetchPublishers(); | ||||
|               }, | ||||
|               child: ListView.builder( | ||||
|                 itemCount: _publishers.length, | ||||
|                 itemBuilder: (context, idx) { | ||||
|                   final publisher = _publishers[idx]; | ||||
|                   return ListTile( | ||||
|                     title: Text(publisher.nick), | ||||
|                     subtitle: Text('@${publisher.name}'), | ||||
|                     contentPadding: const EdgeInsets.symmetric(horizontal: 16), | ||||
|                     leading: AccountImage(content: publisher.avatar), | ||||
|                     trailing: PopupMenuButton( | ||||
|                       itemBuilder: (BuildContext context) => [ | ||||
|                         PopupMenuItem( | ||||
|                           child: Row( | ||||
|                             children: [ | ||||
|                               const Icon(Symbols.edit), | ||||
|                               const Gap(16), | ||||
|                               Text('edit').tr(), | ||||
|                             ], | ||||
|                           ), | ||||
|                           onTap: () { | ||||
|                             GoRouter.of(context).pushNamed( | ||||
|                               'accountPublisherEdit', | ||||
|                               pathParameters: { | ||||
|                                 'name': publisher.name, | ||||
|                               }, | ||||
|                             ).then((value) { | ||||
|                               if (value == true) { | ||||
|                                 _publishers.clear(); | ||||
|                                 _fetchPublishers(); | ||||
|                               } | ||||
|                             }); | ||||
|                           }, | ||||
|                         ), | ||||
|                       ], | ||||
|                     ), | ||||
|                   ); | ||||
|             child: MediaQuery.removePadding( | ||||
|               context: context, | ||||
|               removeTop: true, | ||||
|               child: RefreshIndicator( | ||||
|                 onRefresh: () { | ||||
|                   _publishers.clear(); | ||||
|                   return _fetchPublishers(); | ||||
|                 }, | ||||
|                 child: ListView.builder( | ||||
|                   itemCount: _publishers.length, | ||||
|                   itemBuilder: (context, idx) { | ||||
|                     final publisher = _publishers[idx]; | ||||
|                     return ListTile( | ||||
|                       title: Text(publisher.nick), | ||||
|                       subtitle: Text('@${publisher.name}'), | ||||
|                       contentPadding: const EdgeInsets.symmetric(horizontal: 16), | ||||
|                       leading: AccountImage(content: publisher.avatar), | ||||
|                       trailing: PopupMenuButton( | ||||
|                         itemBuilder: (BuildContext context) => [ | ||||
|                           PopupMenuItem( | ||||
|                             child: Row( | ||||
|                               children: [ | ||||
|                                 const Icon(Symbols.edit), | ||||
|                                 const Gap(16), | ||||
|                                 Text('edit').tr(), | ||||
|                               ], | ||||
|                             ), | ||||
|                             onTap: () { | ||||
|                               GoRouter.of(context).pushNamed( | ||||
|                                 'accountPublisherEdit', | ||||
|                                 pathParameters: { | ||||
|                                   'name': publisher.name, | ||||
|                                 }, | ||||
|                               ).then((value) { | ||||
|                                 if (value == true) { | ||||
|                                   _publishers.clear(); | ||||
|                                   _fetchPublishers(); | ||||
|                                 } | ||||
|                               }); | ||||
|                             }, | ||||
|                           ), | ||||
|                         ], | ||||
|                       ), | ||||
|                     ); | ||||
|                   }, | ||||
|                 ), | ||||
|               ), | ||||
|             ), | ||||
|           ), | ||||
|   | ||||
| @@ -11,6 +11,7 @@ import 'package:surface/widgets/app_bar_leading.dart'; | ||||
| import 'package:surface/widgets/attachment/attachment_zoom.dart'; | ||||
| import 'package:surface/widgets/attachment/attachment_item.dart'; | ||||
| import 'package:surface/widgets/dialog.dart'; | ||||
| import 'package:surface/widgets/navigation/app_scaffold.dart'; | ||||
| import 'package:uuid/uuid.dart'; | ||||
|  | ||||
| class AlbumScreen extends StatefulWidget { | ||||
| @@ -82,7 +83,7 @@ class _AlbumScreenState extends State<AlbumScreen> { | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     return Scaffold( | ||||
|     return AppScaffold( | ||||
|       body: CustomScrollView( | ||||
|         controller: _scrollController, | ||||
|         slivers: [ | ||||
|   | ||||
| @@ -7,17 +7,14 @@ import 'package:provider/provider.dart'; | ||||
| import 'package:styled_widget/styled_widget.dart'; | ||||
| import 'package:surface/providers/sn_network.dart'; | ||||
| import 'package:surface/providers/userinfo.dart'; | ||||
| import 'package:surface/screens/account/factor_settings.dart'; | ||||
| import 'package:surface/types/auth.dart'; | ||||
| import 'package:surface/widgets/dialog.dart'; | ||||
| import 'package:surface/widgets/navigation/app_scaffold.dart'; | ||||
| import 'package:url_launcher/url_launcher_string.dart'; | ||||
|  | ||||
| import '../../providers/websocket.dart'; | ||||
|  | ||||
| final Map<int, (String label, IconData icon, bool isOtp)> _factorLabelMap = { | ||||
|   0: ('authFactorPassword'.tr(), Symbols.password, false), | ||||
|   1: ('authFactorEmail'.tr(), Symbols.email, true), | ||||
| }; | ||||
|  | ||||
| class LoginScreen extends StatefulWidget { | ||||
|   const LoginScreen({super.key}); | ||||
|  | ||||
| @@ -35,67 +32,73 @@ class _LoginScreenState extends State<LoginScreen> { | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     return Theme( | ||||
|       data: Theme.of(context).copyWith(canvasColor: Colors.transparent), | ||||
|       child: SingleChildScrollView( | ||||
|         child: PageTransitionSwitcher( | ||||
|           transitionBuilder: ( | ||||
|             Widget child, | ||||
|             Animation<double> primaryAnimation, | ||||
|             Animation<double> secondaryAnimation, | ||||
|           ) { | ||||
|             return SharedAxisTransition( | ||||
|               animation: primaryAnimation, | ||||
|               secondaryAnimation: secondaryAnimation, | ||||
|               transitionType: SharedAxisTransitionType.horizontal, | ||||
|               child: Container( | ||||
|                 constraints: BoxConstraints(maxWidth: 380), | ||||
|                 child: child, | ||||
|               ), | ||||
|             ); | ||||
|           }, | ||||
|           child: switch (_period % 3) { | ||||
|             1 => _LoginPickerScreen( | ||||
|                 key: const ValueKey(1), | ||||
|                 ticket: _currentTicket, | ||||
|                 factors: _factors, | ||||
|                 onTicket: (p0) => setState(() { | ||||
|                   _currentTicket = p0; | ||||
|                 }), | ||||
|                 onPickFactor: (p0) => setState(() { | ||||
|                   _factorPicked = p0; | ||||
|                 }), | ||||
|                 onNext: () => setState(() { | ||||
|                   _period++; | ||||
|                 }), | ||||
|               ), | ||||
|             2 => _LoginCheckScreen( | ||||
|                 key: const ValueKey(2), | ||||
|                 ticket: _currentTicket, | ||||
|                 factor: _factorPicked, | ||||
|                 onTicket: (p0) => setState(() { | ||||
|                   _currentTicket = p0; | ||||
|                 }), | ||||
|                 onNext: () => setState(() { | ||||
|                   _period = 1; | ||||
|                 }), | ||||
|               ), | ||||
|             _ => _LoginLookupScreen( | ||||
|                 key: const ValueKey(0), | ||||
|                 ticket: _currentTicket, | ||||
|                 onTicket: (p0) => setState(() { | ||||
|                   _currentTicket = p0; | ||||
|                 }), | ||||
|                 onFactor: (p0) => setState(() { | ||||
|                   _factors = p0; | ||||
|                 }), | ||||
|                 onNext: () => setState(() { | ||||
|                   _period++; | ||||
|                 }), | ||||
|               ), | ||||
|           }, | ||||
|         ).padding(all: 24), | ||||
|       ).center(), | ||||
|     return AppScaffold( | ||||
|       appBar: AppBar( | ||||
|         leading: const PageBackButton(), | ||||
|         title: Text('screenAuthLogin').tr(), | ||||
|       ), | ||||
|       body: Theme( | ||||
|         data: Theme.of(context).copyWith(canvasColor: Colors.transparent), | ||||
|         child: SingleChildScrollView( | ||||
|           child: PageTransitionSwitcher( | ||||
|             transitionBuilder: ( | ||||
|               Widget child, | ||||
|               Animation<double> primaryAnimation, | ||||
|               Animation<double> secondaryAnimation, | ||||
|             ) { | ||||
|               return SharedAxisTransition( | ||||
|                 animation: primaryAnimation, | ||||
|                 secondaryAnimation: secondaryAnimation, | ||||
|                 transitionType: SharedAxisTransitionType.horizontal, | ||||
|                 child: Container( | ||||
|                   constraints: BoxConstraints(maxWidth: 380), | ||||
|                   child: child, | ||||
|                 ), | ||||
|               ); | ||||
|             }, | ||||
|             child: switch (_period % 3) { | ||||
|               1 => _LoginPickerScreen( | ||||
|                   key: const ValueKey(1), | ||||
|                   ticket: _currentTicket, | ||||
|                   factors: _factors, | ||||
|                   onTicket: (p0) => setState(() { | ||||
|                     _currentTicket = p0; | ||||
|                   }), | ||||
|                   onPickFactor: (p0) => setState(() { | ||||
|                     _factorPicked = p0; | ||||
|                   }), | ||||
|                   onNext: () => setState(() { | ||||
|                     _period++; | ||||
|                   }), | ||||
|                 ), | ||||
|               2 => _LoginCheckScreen( | ||||
|                   key: const ValueKey(2), | ||||
|                   ticket: _currentTicket, | ||||
|                   factor: _factorPicked, | ||||
|                   onTicket: (p0) => setState(() { | ||||
|                     _currentTicket = p0; | ||||
|                   }), | ||||
|                   onNext: () => setState(() { | ||||
|                     _period = 1; | ||||
|                   }), | ||||
|                 ), | ||||
|               _ => _LoginLookupScreen( | ||||
|                   key: const ValueKey(0), | ||||
|                   ticket: _currentTicket, | ||||
|                   onTicket: (p0) => setState(() { | ||||
|                     _currentTicket = p0; | ||||
|                   }), | ||||
|                   onFactor: (p0) => setState(() { | ||||
|                     _factors = p0; | ||||
|                   }), | ||||
|                   onNext: () => setState(() { | ||||
|                     _period++; | ||||
|                   }), | ||||
|                 ), | ||||
|             }, | ||||
|           ).padding(all: 24), | ||||
|         ).center(), | ||||
|       ), | ||||
|     ); | ||||
|   } | ||||
| } | ||||
| @@ -105,6 +108,7 @@ class _LoginCheckScreen extends StatefulWidget { | ||||
|   final SnAuthFactor? factor; | ||||
|   final Function(SnAuthTicket?) onTicket; | ||||
|   final Function onNext; | ||||
|  | ||||
|   const _LoginCheckScreen({ | ||||
|     super.key, | ||||
|     required this.ticket, | ||||
| @@ -204,7 +208,7 @@ class _LoginCheckScreenState extends State<_LoginCheckScreen> { | ||||
|           controller: _passwordController, | ||||
|           obscureText: true, | ||||
|           autofillHints: [ | ||||
|             (_factorLabelMap[widget.factor!.type]?.$3 ?? true) | ||||
|             widget.factor!.type == 0 | ||||
|                 ? AutofillHints.password | ||||
|                 : AutofillHints.oneTimeCode | ||||
|           ], | ||||
| @@ -243,6 +247,7 @@ class _LoginPickerScreen extends StatefulWidget { | ||||
|   final Function(SnAuthTicket?) onTicket; | ||||
|   final Function(SnAuthFactor) onPickFactor; | ||||
|   final Function onNext; | ||||
|  | ||||
|   const _LoginPickerScreen({ | ||||
|     super.key, | ||||
|     required this.ticket, | ||||
| @@ -322,11 +327,11 @@ class _LoginPickerScreenState extends State<_LoginPickerScreen> { | ||||
|                           ), | ||||
|                         ), | ||||
|                         secondary: Icon( | ||||
|                           _factorLabelMap[x.type]?.$2 ?? Symbols.question_mark, | ||||
|                           kFactorTypes[x.type]?.$3 ?? Symbols.question_mark, | ||||
|                         ), | ||||
|                         title: Text( | ||||
|                           _factorLabelMap[x.type]?.$1 ?? 'unknown'.tr(), | ||||
|                         ), | ||||
|                           kFactorTypes[x.type]?.$1 ?? 'unknown', | ||||
|                         ).tr(), | ||||
|                         enabled: !widget.ticket!.factorTrail.contains(x.id), | ||||
|                         value: _factorPicked == x.id, | ||||
|                         onChanged: (value) { | ||||
| @@ -373,6 +378,7 @@ class _LoginLookupScreen extends StatefulWidget { | ||||
|   final Function(SnAuthTicket?) onTicket; | ||||
|   final Function(List<SnAuthFactor>?) onFactor; | ||||
|   final Function onNext; | ||||
|  | ||||
|   const _LoginLookupScreen({ | ||||
|     super.key, | ||||
|     required this.ticket, | ||||
| @@ -406,9 +412,11 @@ class _LoginLookupScreenState extends State<_LoginLookupScreen> { | ||||
|       await sn.client.post('/cgi/id/users/me/password-reset', data: { | ||||
|         'user_id': lookupResp.data['id'], | ||||
|       }); | ||||
|       context.showModalDialog('done'.tr(), 'signinResetPasswordSent'.tr()); | ||||
|       if (mounted) { | ||||
|         context.showModalDialog('done'.tr(), 'signinResetPasswordSent'.tr()); | ||||
|       } | ||||
|     } catch (err) { | ||||
|       context.showErrorDialog(err); | ||||
|       if (mounted) context.showErrorDialog(err); | ||||
|     } finally { | ||||
|       setState(() => _isBusy = false); | ||||
|     } | ||||
| @@ -443,7 +451,7 @@ class _LoginLookupScreenState extends State<_LoginLookupScreen> { | ||||
|  | ||||
|       widget.onNext(); | ||||
|     } catch (err) { | ||||
|       context.showErrorDialog(err); | ||||
|       if (mounted) context.showErrorDialog(err); | ||||
|       return; | ||||
|     } finally { | ||||
|       setState(() => _isBusy = false); | ||||
|   | ||||
| @@ -8,6 +8,7 @@ import 'package:provider/provider.dart'; | ||||
| import 'package:styled_widget/styled_widget.dart'; | ||||
| import 'package:surface/providers/sn_network.dart'; | ||||
| import 'package:surface/widgets/dialog.dart'; | ||||
| import 'package:surface/widgets/navigation/app_scaffold.dart'; | ||||
| import 'package:url_launcher/url_launcher_string.dart'; | ||||
|  | ||||
| class RegisterScreen extends StatefulWidget { | ||||
| @@ -43,6 +44,7 @@ class _RegisterScreenState extends State<RegisterScreen> { | ||||
|         'nick': nickname, | ||||
|         'email': email, | ||||
|         'password': password, | ||||
|         'language': EasyLocalization.of(context)!.currentLocale.toString(), | ||||
|       }); | ||||
|  | ||||
|       if (!context.mounted) return; | ||||
| @@ -54,175 +56,178 @@ class _RegisterScreenState extends State<RegisterScreen> { | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     return StyledWidget(Container( | ||||
|       constraints: const BoxConstraints(maxWidth: 380), | ||||
|       child: SingleChildScrollView( | ||||
|         child: Column( | ||||
|           crossAxisAlignment: CrossAxisAlignment.start, | ||||
|           children: [ | ||||
|             Align( | ||||
|               alignment: Alignment.centerLeft, | ||||
|               child: CircleAvatar( | ||||
|                 radius: 26, | ||||
|                 child: const Icon( | ||||
|                   Symbols.person_add, | ||||
|                   size: 28, | ||||
|                 ), | ||||
|               ).padding(bottom: 8), | ||||
|             ), | ||||
|             Text( | ||||
|               'screenAuthRegister', | ||||
|               style: const TextStyle( | ||||
|                 fontSize: 28, | ||||
|                 fontWeight: FontWeight.w900, | ||||
|     return AppScaffold( | ||||
|       appBar: AppBar( | ||||
|         leading: const PageBackButton(), | ||||
|         title: Text('screenAuthRegister').tr(), | ||||
|       ), | ||||
|       body: StyledWidget(Container( | ||||
|         constraints: const BoxConstraints(maxWidth: 380), | ||||
|         child: SingleChildScrollView( | ||||
|           child: Column( | ||||
|             crossAxisAlignment: CrossAxisAlignment.start, | ||||
|             children: [ | ||||
|               Align( | ||||
|                 alignment: Alignment.centerLeft, | ||||
|                 child: CircleAvatar( | ||||
|                   radius: 26, | ||||
|                   child: const Icon( | ||||
|                     Symbols.person_add, | ||||
|                     size: 28, | ||||
|                   ), | ||||
|                 ).padding(bottom: 8), | ||||
|               ), | ||||
|             ).tr().padding(left: 4, bottom: 16), | ||||
|             Form( | ||||
|               key: _formKey, | ||||
|               autovalidateMode: AutovalidateMode.onUserInteraction, | ||||
|               child: Column( | ||||
|                 children: [ | ||||
|                   TextFormField( | ||||
|                     validator: (value) { | ||||
|                       if (value == null || value.length < 4 || value.length > 32) { | ||||
|                         return 'fieldUsernameLengthLimit'.tr(args: [4.toString(), 32.toString()]); | ||||
|                       } | ||||
|                       if (!RegExp(r'^[a-zA-Z0-9_]+$').hasMatch(value)) { | ||||
|                         return 'fieldUsernameAlphanumOnly'.tr(); | ||||
|                       } | ||||
|                       return null; | ||||
|                     }, | ||||
|                     autocorrect: false, | ||||
|                     enableSuggestions: false, | ||||
|                     controller: _usernameController, | ||||
|                     autofillHints: const [AutofillHints.username], | ||||
|                     decoration: InputDecoration( | ||||
|                       isDense: true, | ||||
|                       border: const UnderlineInputBorder(), | ||||
|                       labelText: 'fieldUsername'.tr(), | ||||
|                     ), | ||||
|                     onTapOutside: (_) => FocusManager.instance.primaryFocus?.unfocus(), | ||||
|                   ), | ||||
|                   const Gap(12), | ||||
|                   TextFormField( | ||||
|                     validator: (value) { | ||||
|                       if (value == null || value.length < 4 || value.length > 32) { | ||||
|                         return 'fieldNicknameLengthLimit'.tr(args: [4.toString(), 32.toString()]); | ||||
|                       } | ||||
|                       return null; | ||||
|                     }, | ||||
|                     autocorrect: false, | ||||
|                     enableSuggestions: false, | ||||
|                     controller: _nicknameController, | ||||
|                     autofillHints: const [AutofillHints.nickname], | ||||
|                     decoration: InputDecoration( | ||||
|                       isDense: true, | ||||
|                       border: const UnderlineInputBorder(), | ||||
|                       labelText: 'fieldNickname'.tr(), | ||||
|                     ), | ||||
|                     onTapOutside: (_) => FocusManager.instance.primaryFocus?.unfocus(), | ||||
|                   ), | ||||
|                   const Gap(12), | ||||
|                   TextFormField( | ||||
|                     validator: (value) { | ||||
|                       if (value == null || value.isEmpty) { | ||||
|                         return 'fieldCannotBeEmpty'.tr(); | ||||
|                       } | ||||
|                       if (!EmailValidator.validate(value)) { | ||||
|                         return 'fieldEmailAddressMustBeValid'.tr(); | ||||
|                       } | ||||
|                       return null; | ||||
|                     }, | ||||
|                     autocorrect: false, | ||||
|                     enableSuggestions: false, | ||||
|                     controller: _emailController, | ||||
|                     autofillHints: const [AutofillHints.email], | ||||
|                     decoration: InputDecoration( | ||||
|                       isDense: true, | ||||
|                       border: const UnderlineInputBorder(), | ||||
|                       labelText: 'fieldEmail'.tr(), | ||||
|                     ), | ||||
|                     onTapOutside: (_) => FocusManager.instance.primaryFocus?.unfocus(), | ||||
|                   ), | ||||
|                   const Gap(12), | ||||
|                   TextFormField( | ||||
|                     validator: (value) { | ||||
|                       if (value == null || value.isEmpty) { | ||||
|                         return 'fieldCannotBeEmpty'.tr(); | ||||
|                       } | ||||
|                       return null; | ||||
|                     }, | ||||
|                     obscureText: true, | ||||
|                     autocorrect: false, | ||||
|                     enableSuggestions: false, | ||||
|                     autofillHints: const [AutofillHints.password], | ||||
|                     controller: _passwordController, | ||||
|                     decoration: InputDecoration( | ||||
|                       isDense: true, | ||||
|                       border: const UnderlineInputBorder(), | ||||
|                       labelText: 'fieldPassword'.tr(), | ||||
|                     ), | ||||
|                     onTapOutside: (_) => FocusManager.instance.primaryFocus?.unfocus(), | ||||
|                   ), | ||||
|                 ], | ||||
|               ).padding(horizontal: 7), | ||||
|             ), | ||||
|             const Gap(16), | ||||
|             Align( | ||||
|               alignment: Alignment.centerRight, | ||||
|               child: StyledWidget( | ||||
|                 Container( | ||||
|                   constraints: const BoxConstraints(maxWidth: 290), | ||||
|                   child: Column( | ||||
|                     crossAxisAlignment: CrossAxisAlignment.end, | ||||
|                     children: [ | ||||
|                       Text( | ||||
|                         'termAcceptNextWithAgree'.tr(), | ||||
|                         textAlign: TextAlign.end, | ||||
|                         style: Theme.of(context).textTheme.bodySmall!.copyWith( | ||||
|                           color: Theme.of(context) | ||||
|                               .colorScheme | ||||
|                               .onSurface | ||||
|                               .withAlpha((255 * 0.75).round()), | ||||
|                         ), | ||||
|               Text( | ||||
|                 'screenAuthRegister', | ||||
|                 style: const TextStyle( | ||||
|                   fontSize: 28, | ||||
|                   fontWeight: FontWeight.w900, | ||||
|                 ), | ||||
|               ).tr().padding(left: 4, bottom: 16), | ||||
|               Form( | ||||
|                 key: _formKey, | ||||
|                 autovalidateMode: AutovalidateMode.onUserInteraction, | ||||
|                 child: Column( | ||||
|                   children: [ | ||||
|                     TextFormField( | ||||
|                       validator: (value) { | ||||
|                         if (value == null || value.length < 4 || value.length > 32) { | ||||
|                           return 'fieldUsernameLengthLimit'.tr(args: [4.toString(), 32.toString()]); | ||||
|                         } | ||||
|                         if (!RegExp(r'^[a-zA-Z0-9_]+$').hasMatch(value)) { | ||||
|                           return 'fieldUsernameAlphanumOnly'.tr(); | ||||
|                         } | ||||
|                         return null; | ||||
|                       }, | ||||
|                       autocorrect: false, | ||||
|                       enableSuggestions: false, | ||||
|                       controller: _usernameController, | ||||
|                       autofillHints: const [AutofillHints.username], | ||||
|                       decoration: InputDecoration( | ||||
|                         isDense: true, | ||||
|                         border: const UnderlineInputBorder(), | ||||
|                         labelText: 'fieldUsername'.tr(), | ||||
|                       ), | ||||
|                       Material( | ||||
|                         color: Colors.transparent, | ||||
|                         child: InkWell( | ||||
|                           child: Row( | ||||
|                             mainAxisSize: MainAxisSize.min, | ||||
|                             children: [ | ||||
|                               Text('termAcceptLink'.tr()), | ||||
|                               const Gap(4), | ||||
|                               const Icon(Symbols.launch, size: 14), | ||||
|                             ], | ||||
|                       onTapOutside: (_) => FocusManager.instance.primaryFocus?.unfocus(), | ||||
|                     ), | ||||
|                     const Gap(12), | ||||
|                     TextFormField( | ||||
|                       validator: (value) { | ||||
|                         if (value == null || value.length < 4 || value.length > 32) { | ||||
|                           return 'fieldNicknameLengthLimit'.tr(args: [4.toString(), 32.toString()]); | ||||
|                         } | ||||
|                         return null; | ||||
|                       }, | ||||
|                       autocorrect: false, | ||||
|                       enableSuggestions: false, | ||||
|                       controller: _nicknameController, | ||||
|                       autofillHints: const [AutofillHints.nickname], | ||||
|                       decoration: InputDecoration( | ||||
|                         isDense: true, | ||||
|                         border: const UnderlineInputBorder(), | ||||
|                         labelText: 'fieldNickname'.tr(), | ||||
|                       ), | ||||
|                       onTapOutside: (_) => FocusManager.instance.primaryFocus?.unfocus(), | ||||
|                     ), | ||||
|                     const Gap(12), | ||||
|                     TextFormField( | ||||
|                       validator: (value) { | ||||
|                         if (value == null || value.isEmpty) { | ||||
|                           return 'fieldCannotBeEmpty'.tr(); | ||||
|                         } | ||||
|                         if (!EmailValidator.validate(value)) { | ||||
|                           return 'fieldEmailAddressMustBeValid'.tr(); | ||||
|                         } | ||||
|                         return null; | ||||
|                       }, | ||||
|                       autocorrect: false, | ||||
|                       enableSuggestions: false, | ||||
|                       controller: _emailController, | ||||
|                       autofillHints: const [AutofillHints.email], | ||||
|                       decoration: InputDecoration( | ||||
|                         isDense: true, | ||||
|                         border: const UnderlineInputBorder(), | ||||
|                         labelText: 'fieldEmail'.tr(), | ||||
|                       ), | ||||
|                       onTapOutside: (_) => FocusManager.instance.primaryFocus?.unfocus(), | ||||
|                     ), | ||||
|                     const Gap(12), | ||||
|                     TextFormField( | ||||
|                       validator: (value) { | ||||
|                         if (value == null || value.isEmpty) { | ||||
|                           return 'fieldCannotBeEmpty'.tr(); | ||||
|                         } | ||||
|                         return null; | ||||
|                       }, | ||||
|                       obscureText: true, | ||||
|                       autocorrect: false, | ||||
|                       enableSuggestions: false, | ||||
|                       autofillHints: const [AutofillHints.password], | ||||
|                       controller: _passwordController, | ||||
|                       decoration: InputDecoration( | ||||
|                         isDense: true, | ||||
|                         border: const UnderlineInputBorder(), | ||||
|                         labelText: 'fieldPassword'.tr(), | ||||
|                       ), | ||||
|                       onTapOutside: (_) => FocusManager.instance.primaryFocus?.unfocus(), | ||||
|                     ), | ||||
|                   ], | ||||
|                 ).padding(horizontal: 7), | ||||
|               ), | ||||
|               const Gap(16), | ||||
|               Align( | ||||
|                 alignment: Alignment.centerRight, | ||||
|                 child: StyledWidget( | ||||
|                   Container( | ||||
|                     constraints: const BoxConstraints(maxWidth: 290), | ||||
|                     child: Column( | ||||
|                       crossAxisAlignment: CrossAxisAlignment.end, | ||||
|                       children: [ | ||||
|                         Text( | ||||
|                           'termAcceptNextWithAgree'.tr(), | ||||
|                           textAlign: TextAlign.end, | ||||
|                           style: Theme.of(context).textTheme.bodySmall!.copyWith( | ||||
|                                 color: Theme.of(context).colorScheme.onSurface.withAlpha((255 * 0.75).round()), | ||||
|                               ), | ||||
|                         ), | ||||
|                         Material( | ||||
|                           color: Colors.transparent, | ||||
|                           child: InkWell( | ||||
|                             child: Row( | ||||
|                               mainAxisSize: MainAxisSize.min, | ||||
|                               children: [ | ||||
|                                 Text('termAcceptLink'.tr()), | ||||
|                                 const Gap(4), | ||||
|                                 const Icon(Symbols.launch, size: 14), | ||||
|                               ], | ||||
|                             ), | ||||
|                             onTap: () { | ||||
|                               launchUrlString('https://solsynth.dev/terms'); | ||||
|                             }, | ||||
|                           ), | ||||
|                           onTap: () { | ||||
|                             launchUrlString('https://solsynth.dev/terms'); | ||||
|                           }, | ||||
|                         ), | ||||
|                       ), | ||||
|                       ], | ||||
|                     ), | ||||
|                   ), | ||||
|                 ).padding(horizontal: 16), | ||||
|               ), | ||||
|               Align( | ||||
|                 alignment: Alignment.centerRight, | ||||
|                 child: TextButton( | ||||
|                   onPressed: () => _performAction(context), | ||||
|                   child: Row( | ||||
|                     mainAxisSize: MainAxisSize.min, | ||||
|                     children: [ | ||||
|                       Text('next').tr(), | ||||
|                       const Icon(Symbols.chevron_right), | ||||
|                     ], | ||||
|                   ), | ||||
|                 ), | ||||
|               ).padding(horizontal: 16), | ||||
|             ), | ||||
|             Align( | ||||
|               alignment: Alignment.centerRight, | ||||
|               child: TextButton( | ||||
|                 onPressed: () => _performAction(context), | ||||
|                 child: Row( | ||||
|                   mainAxisSize: MainAxisSize.min, | ||||
|                   children: [ | ||||
|                     Text('next').tr(), | ||||
|                     const Icon(Symbols.chevron_right), | ||||
|                   ], | ||||
|                 ), | ||||
|               ), | ||||
|             ), | ||||
|           ], | ||||
|             ], | ||||
|           ), | ||||
|         ), | ||||
|       ), | ||||
|     )).padding(all: 24).center(); | ||||
|       )).padding(all: 24).center(), | ||||
|     ); | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -13,6 +13,7 @@ import 'package:surface/widgets/account/account_select.dart'; | ||||
| import 'package:surface/widgets/app_bar_leading.dart'; | ||||
| import 'package:surface/widgets/dialog.dart'; | ||||
| import 'package:surface/widgets/loading_indicator.dart'; | ||||
| import 'package:surface/widgets/navigation/app_scaffold.dart'; | ||||
| import 'package:surface/widgets/unauthorized_hint.dart'; | ||||
| import 'package:uuid/uuid.dart'; | ||||
|  | ||||
| @@ -120,7 +121,7 @@ class _ChatScreenState extends State<ChatScreen> { | ||||
|     final ua = context.read<UserProvider>(); | ||||
|  | ||||
|     if (!ua.isAuthorized) { | ||||
|       return Scaffold( | ||||
|       return AppScaffold( | ||||
|         appBar: AppBar( | ||||
|           leading: AutoAppBarLeading(), | ||||
|           title: Text('screenChat').tr(), | ||||
| @@ -131,7 +132,7 @@ class _ChatScreenState extends State<ChatScreen> { | ||||
|       ); | ||||
|     } | ||||
|  | ||||
|     return Scaffold( | ||||
|     return AppScaffold( | ||||
|       appBar: AppBar( | ||||
|         leading: AutoAppBarLeading(), | ||||
|         title: Text('screenChat').tr(), | ||||
| @@ -195,22 +196,58 @@ class _ChatScreenState extends State<ChatScreen> { | ||||
|         children: [ | ||||
|           LoadingIndicator(isActive: _isBusy), | ||||
|           Expanded( | ||||
|             child: RefreshIndicator( | ||||
|               onRefresh: () => Future.sync(() => _refreshChannels()), | ||||
|               child: ListView.builder( | ||||
|                 itemCount: _channels?.length ?? 0, | ||||
|                 itemBuilder: (context, idx) { | ||||
|                   final channel = _channels![idx]; | ||||
|                   final lastMessage = _lastMessages?[channel.id]; | ||||
|             child: MediaQuery.removePadding( | ||||
|               context: context, | ||||
|               removeTop: true, | ||||
|               child: RefreshIndicator( | ||||
|                 onRefresh: () => Future.sync(() => _refreshChannels()), | ||||
|                 child: ListView.builder( | ||||
|                   itemCount: _channels?.length ?? 0, | ||||
|                   itemBuilder: (context, idx) { | ||||
|                     final channel = _channels![idx]; | ||||
|                     final lastMessage = _lastMessages?[channel.id]; | ||||
|  | ||||
|                   if (channel.type == 1) { | ||||
|                     final otherMember = channel.members?.cast<SnChannelMember?>().firstWhere( | ||||
|                           (ele) => ele?.accountId != ua.user?.id, | ||||
|                           orElse: () => null, | ||||
|                         ); | ||||
|                     if (channel.type == 1) { | ||||
|                       final otherMember = channel.members?.cast<SnChannelMember?>().firstWhere( | ||||
|                             (ele) => ele?.accountId != ua.user?.id, | ||||
|                             orElse: () => null, | ||||
|                           ); | ||||
|  | ||||
|                       return ListTile( | ||||
|                         title: Text(ud.getAccountFromCache(otherMember?.accountId)?.nick ?? channel.name), | ||||
|                         subtitle: lastMessage != null | ||||
|                             ? Text( | ||||
|                                 '${ud.getAccountFromCache(lastMessage.sender.accountId)?.nick}: ${lastMessage.body['text'] ?? 'Unable preview'}', | ||||
|                                 maxLines: 1, | ||||
|                                 overflow: TextOverflow.ellipsis, | ||||
|                               ) | ||||
|                             : Text( | ||||
|                                 'channelDirectMessageDescription'.tr(args: [ | ||||
|                                   '@${ud.getAccountFromCache(otherMember?.accountId)?.name}', | ||||
|                                 ]), | ||||
|                                 maxLines: 1, | ||||
|                                 overflow: TextOverflow.ellipsis, | ||||
|                               ), | ||||
|                         contentPadding: const EdgeInsets.symmetric(horizontal: 16), | ||||
|                         leading: AccountImage( | ||||
|                           content: ud.getAccountFromCache(otherMember?.accountId)?.avatar, | ||||
|                         ), | ||||
|                         onTap: () { | ||||
|                           GoRouter.of(context).pushNamed( | ||||
|                             'chatRoom', | ||||
|                             pathParameters: { | ||||
|                               'scope': channel.realm?.alias ?? 'global', | ||||
|                               'alias': channel.alias, | ||||
|                             }, | ||||
|                           ).then((value) { | ||||
|                             if (mounted) _refreshChannels(); | ||||
|                           }); | ||||
|                         }, | ||||
|                       ); | ||||
|                     } | ||||
|  | ||||
|                     return ListTile( | ||||
|                       title: Text(ud.getAccountFromCache(otherMember?.accountId)?.nick ?? channel.name), | ||||
|                       title: Text(channel.name), | ||||
|                       subtitle: lastMessage != null | ||||
|                           ? Text( | ||||
|                               '${ud.getAccountFromCache(lastMessage.sender.accountId)?.nick}: ${lastMessage.body['text'] ?? 'Unable preview'}', | ||||
| @@ -218,15 +255,14 @@ class _ChatScreenState extends State<ChatScreen> { | ||||
|                               overflow: TextOverflow.ellipsis, | ||||
|                             ) | ||||
|                           : Text( | ||||
|                               'channelDirectMessageDescription'.tr(args: [ | ||||
|                                 '@${ud.getAccountFromCache(otherMember?.accountId)?.name}', | ||||
|                               ]), | ||||
|                               channel.description, | ||||
|                               maxLines: 1, | ||||
|                               overflow: TextOverflow.ellipsis, | ||||
|                             ), | ||||
|                       contentPadding: const EdgeInsets.symmetric(horizontal: 16), | ||||
|                       leading: AccountImage( | ||||
|                         content: ud.getAccountFromCache(otherMember?.accountId)?.avatar, | ||||
|                         content: null, | ||||
|                         fallbackWidget: const Icon(Symbols.chat, size: 20), | ||||
|                       ), | ||||
|                       onTap: () { | ||||
|                         GoRouter.of(context).pushNamed( | ||||
| @@ -240,39 +276,8 @@ class _ChatScreenState extends State<ChatScreen> { | ||||
|                         }); | ||||
|                       }, | ||||
|                     ); | ||||
|                   } | ||||
|  | ||||
|                   return ListTile( | ||||
|                     title: Text(channel.name), | ||||
|                     subtitle: lastMessage != null | ||||
|                         ? Text( | ||||
|                             '${ud.getAccountFromCache(lastMessage.sender.accountId)?.nick}: ${lastMessage.body['text'] ?? 'Unable preview'}', | ||||
|                             maxLines: 1, | ||||
|                             overflow: TextOverflow.ellipsis, | ||||
|                           ) | ||||
|                         : Text( | ||||
|                             channel.description, | ||||
|                             maxLines: 1, | ||||
|                             overflow: TextOverflow.ellipsis, | ||||
|                           ), | ||||
|                     contentPadding: const EdgeInsets.symmetric(horizontal: 16), | ||||
|                     leading: AccountImage( | ||||
|                       content: null, | ||||
|                       fallbackWidget: const Icon(Symbols.chat, size: 20), | ||||
|                     ), | ||||
|                     onTap: () { | ||||
|                       GoRouter.of(context).pushNamed( | ||||
|                         'chatRoom', | ||||
|                         pathParameters: { | ||||
|                           'scope': channel.realm?.alias ?? 'global', | ||||
|                           'alias': channel.alias, | ||||
|                         }, | ||||
|                       ).then((value) { | ||||
|                         if (value == true) _refreshChannels(); | ||||
|                       }); | ||||
|                     }, | ||||
|                   ); | ||||
|                 }, | ||||
|                   }, | ||||
|                 ), | ||||
|               ), | ||||
|             ), | ||||
|           ), | ||||
|   | ||||
| @@ -9,10 +9,12 @@ import 'package:styled_widget/styled_widget.dart'; | ||||
| import 'package:surface/providers/chat_call.dart'; | ||||
| import 'package:surface/widgets/chat/call/call_controls.dart'; | ||||
| import 'package:surface/widgets/chat/call/call_participant.dart'; | ||||
| import 'package:surface/widgets/navigation/app_scaffold.dart'; | ||||
|  | ||||
| class CallRoomScreen extends StatefulWidget { | ||||
|   final String scope; | ||||
|   final String alias; | ||||
|  | ||||
|   const CallRoomScreen({super.key, required this.scope, required this.alias}); | ||||
|  | ||||
|   @override | ||||
| @@ -35,8 +37,7 @@ class _CallRoomScreenState extends State<CallRoomScreen> { | ||||
|     return Stack( | ||||
|       children: [ | ||||
|         Container( | ||||
|           color: | ||||
|               Theme.of(context).colorScheme.surfaceContainer.withOpacity(0.75), | ||||
|           color: Theme.of(context).colorScheme.surfaceContainer.withOpacity(0.75), | ||||
|           child: call.focusTrack != null | ||||
|               ? InteractiveParticipantWidget( | ||||
|                   isFixedAvatar: false, | ||||
| @@ -71,8 +72,7 @@ class _CallRoomScreenState extends State<CallRoomScreen> { | ||||
|                       color: Theme.of(context).cardColor, | ||||
|                       participant: track, | ||||
|                       onTap: () { | ||||
|                         if (track.participant.sid != | ||||
|                             call.focusTrack?.participant.sid) { | ||||
|                         if (track.participant.sid != call.focusTrack?.participant.sid) { | ||||
|                           call.setFocusTrack(track); | ||||
|                         } | ||||
|                       }, | ||||
| @@ -114,14 +114,10 @@ class _CallRoomScreenState extends State<CallRoomScreen> { | ||||
|             child: ClipRRect( | ||||
|               borderRadius: const BorderRadius.all(Radius.circular(8)), | ||||
|               child: InteractiveParticipantWidget( | ||||
|                 color: Theme.of(context) | ||||
|                     .colorScheme | ||||
|                     .surfaceContainerHigh | ||||
|                     .withOpacity(0.75), | ||||
|                 color: Theme.of(context).colorScheme.surfaceContainerHigh.withOpacity(0.75), | ||||
|                 participant: track, | ||||
|                 onTap: () { | ||||
|                   if (track.participant.sid != | ||||
|                       call.focusTrack?.participant.sid) { | ||||
|                   if (track.participant.sid != call.focusTrack?.participant.sid) { | ||||
|                     call.setFocusTrack(track); | ||||
|                   } | ||||
|                 }, | ||||
| @@ -152,157 +148,134 @@ class _CallRoomScreenState extends State<CallRoomScreen> { | ||||
|     return ListenableBuilder( | ||||
|         listenable: call, | ||||
|         builder: (context, _) { | ||||
|           return Scaffold( | ||||
|           return AppScaffold( | ||||
|             appBar: AppBar( | ||||
|               title: RichText( | ||||
|                 textAlign: TextAlign.center, | ||||
|                 text: TextSpan(children: [ | ||||
|                   TextSpan( | ||||
|                     text: 'call'.tr(), | ||||
|                     style: Theme.of(context) | ||||
|                         .textTheme | ||||
|                         .titleLarge! | ||||
|                         .copyWith(color: Colors.white), | ||||
|                     style: Theme.of(context).textTheme.titleLarge!.copyWith(color: Colors.white), | ||||
|                   ), | ||||
|                   const TextSpan(text: '\n'), | ||||
|                   TextSpan( | ||||
|                     text: call.lastDuration.toString(), | ||||
|                     style: Theme.of(context) | ||||
|                         .textTheme | ||||
|                         .bodySmall! | ||||
|                         .copyWith(color: Colors.white), | ||||
|                     style: Theme.of(context).textTheme.bodySmall!.copyWith(color: Colors.white), | ||||
|                   ), | ||||
|                 ]), | ||||
|               ), | ||||
|             ), | ||||
|             body: SafeArea( | ||||
|               child: GestureDetector( | ||||
|                 behavior: HitTestBehavior.translucent, | ||||
|                 child: Column( | ||||
|                   children: [ | ||||
|             body: GestureDetector( | ||||
|               behavior: HitTestBehavior.translucent, | ||||
|               child: Column( | ||||
|                 children: [ | ||||
|                   SizedBox( | ||||
|                     width: MediaQuery.of(context).size.width, | ||||
|                     height: 64, | ||||
|                     child: Row( | ||||
|                       mainAxisAlignment: MainAxisAlignment.spaceBetween, | ||||
|                       crossAxisAlignment: CrossAxisAlignment.center, | ||||
|                       children: [ | ||||
|                         Builder(builder: (context) { | ||||
|                           final call = context.read<ChatCallProvider>(); | ||||
|                           final connectionQuality = | ||||
|                               call.room.localParticipant?.connectionQuality ?? livekit.ConnectionQuality.unknown; | ||||
|                           return Expanded( | ||||
|                             child: Column( | ||||
|                               mainAxisSize: MainAxisSize.min, | ||||
|                               crossAxisAlignment: CrossAxisAlignment.start, | ||||
|                               children: [ | ||||
|                                 Row( | ||||
|                                   children: [ | ||||
|                                     Text( | ||||
|                                       call.channel?.name ?? 'unknown'.tr(), | ||||
|                                       style: const TextStyle( | ||||
|                                         fontWeight: FontWeight.bold, | ||||
|                                       ), | ||||
|                                     ), | ||||
|                                     const Gap(6), | ||||
|                                     Text(call.lastDuration.toString()) | ||||
|                                   ], | ||||
|                                 ), | ||||
|                                 Row( | ||||
|                                   children: [ | ||||
|                                     Text( | ||||
|                                       { | ||||
|                                         livekit.ConnectionState.disconnected: 'callStatusDisconnected'.tr(), | ||||
|                                         livekit.ConnectionState.connected: 'callStatusConnected'.tr(), | ||||
|                                         livekit.ConnectionState.connecting: 'callStatusConnecting'.tr(), | ||||
|                                         livekit.ConnectionState.reconnecting: 'callStatusReconnecting'.tr(), | ||||
|                                       }[call.room.connectionState]!, | ||||
|                                     ), | ||||
|                                     const Gap(6), | ||||
|                                     if (connectionQuality != livekit.ConnectionQuality.unknown) | ||||
|                                       Icon( | ||||
|                                         { | ||||
|                                           livekit.ConnectionQuality.excellent: Icons.signal_cellular_alt, | ||||
|                                           livekit.ConnectionQuality.good: Icons.signal_cellular_alt_2_bar, | ||||
|                                           livekit.ConnectionQuality.poor: Icons.signal_cellular_alt_1_bar, | ||||
|                                         }[connectionQuality], | ||||
|                                         color: { | ||||
|                                           livekit.ConnectionQuality.excellent: Colors.green, | ||||
|                                           livekit.ConnectionQuality.good: Colors.orange, | ||||
|                                           livekit.ConnectionQuality.poor: Colors.red, | ||||
|                                         }[connectionQuality], | ||||
|                                         size: 16, | ||||
|                                       ) | ||||
|                                     else | ||||
|                                       const SizedBox( | ||||
|                                         width: 12, | ||||
|                                         height: 12, | ||||
|                                         child: CircularProgressIndicator( | ||||
|                                           color: Colors.white, | ||||
|                                           strokeWidth: 2, | ||||
|                                         ), | ||||
|                                       ).padding(all: 3), | ||||
|                                   ], | ||||
|                                 ), | ||||
|                               ], | ||||
|                             ), | ||||
|                           ); | ||||
|                         }), | ||||
|                         Row( | ||||
|                           children: [ | ||||
|                             IconButton( | ||||
|                               icon: _layoutMode == 0 ? const Icon(Icons.view_list) : const Icon(Icons.grid_view), | ||||
|                               onPressed: () { | ||||
|                                 _switchLayout(); | ||||
|                               }, | ||||
|                             ), | ||||
|                           ], | ||||
|                         ), | ||||
|                       ], | ||||
|                     ).padding(left: 20, right: 16), | ||||
|                   ), | ||||
|                   Expanded( | ||||
|                     child: Material( | ||||
|                       color: Theme.of(context).colorScheme.surfaceContainerLow, | ||||
|                       child: Builder( | ||||
|                         builder: (context) { | ||||
|                           switch (_layoutMode) { | ||||
|                             case 1: | ||||
|                               return _buildGridLayout(); | ||||
|                             default: | ||||
|                               return _buildListLayout(); | ||||
|                           } | ||||
|                         }, | ||||
|                       ), | ||||
|                     ), | ||||
|                   ), | ||||
|                   if (call.room.localParticipant != null) | ||||
|                     SizedBox( | ||||
|                       width: MediaQuery.of(context).size.width, | ||||
|                       height: 64, | ||||
|                       child: Row( | ||||
|                         mainAxisAlignment: MainAxisAlignment.spaceBetween, | ||||
|                         crossAxisAlignment: CrossAxisAlignment.center, | ||||
|                         children: [ | ||||
|                           Builder(builder: (context) { | ||||
|                             final call = context.read<ChatCallProvider>(); | ||||
|                             final connectionQuality = | ||||
|                                 call.room.localParticipant?.connectionQuality ?? | ||||
|                                     livekit.ConnectionQuality.unknown; | ||||
|                             return Expanded( | ||||
|                               child: Column( | ||||
|                                 mainAxisSize: MainAxisSize.min, | ||||
|                                 crossAxisAlignment: CrossAxisAlignment.start, | ||||
|                                 children: [ | ||||
|                                   Row( | ||||
|                                     children: [ | ||||
|                                       Text( | ||||
|                                         call.channel?.name ?? 'unknown'.tr(), | ||||
|                                         style: const TextStyle( | ||||
|                                           fontWeight: FontWeight.bold, | ||||
|                                         ), | ||||
|                                       ), | ||||
|                                       const Gap(6), | ||||
|                                       Text(call.lastDuration.toString()) | ||||
|                                     ], | ||||
|                                   ), | ||||
|                                   Row( | ||||
|                                     children: [ | ||||
|                                       Text( | ||||
|                                         { | ||||
|                                           livekit.ConnectionState.disconnected: | ||||
|                                               'callStatusDisconnected'.tr(), | ||||
|                                           livekit.ConnectionState.connected: | ||||
|                                               'callStatusConnected'.tr(), | ||||
|                                           livekit.ConnectionState.connecting: | ||||
|                                               'callStatusConnecting'.tr(), | ||||
|                                           livekit.ConnectionState.reconnecting: | ||||
|                                               'callStatusReconnecting'.tr(), | ||||
|                                         }[call.room.connectionState]!, | ||||
|                                       ), | ||||
|                                       const Gap(6), | ||||
|                                       if (connectionQuality != | ||||
|                                           livekit.ConnectionQuality.unknown) | ||||
|                                         Icon( | ||||
|                                           { | ||||
|                                             livekit.ConnectionQuality.excellent: | ||||
|                                                 Icons.signal_cellular_alt, | ||||
|                                             livekit.ConnectionQuality.good: | ||||
|                                                 Icons.signal_cellular_alt_2_bar, | ||||
|                                             livekit.ConnectionQuality.poor: | ||||
|                                                 Icons.signal_cellular_alt_1_bar, | ||||
|                                           }[connectionQuality], | ||||
|                                           color: { | ||||
|                                             livekit.ConnectionQuality.excellent: | ||||
|                                                 Colors.green, | ||||
|                                             livekit.ConnectionQuality.good: | ||||
|                                                 Colors.orange, | ||||
|                                             livekit.ConnectionQuality.poor: | ||||
|                                                 Colors.red, | ||||
|                                           }[connectionQuality], | ||||
|                                           size: 16, | ||||
|                                         ) | ||||
|                                       else | ||||
|                                         const SizedBox( | ||||
|                                           width: 12, | ||||
|                                           height: 12, | ||||
|                                           child: CircularProgressIndicator( | ||||
|                                             color: Colors.white, | ||||
|                                             strokeWidth: 2, | ||||
|                                           ), | ||||
|                                         ).padding(all: 3), | ||||
|                                     ], | ||||
|                                   ), | ||||
|                                 ], | ||||
|                               ), | ||||
|                             ); | ||||
|                           }), | ||||
|                           Row( | ||||
|                             children: [ | ||||
|                               IconButton( | ||||
|                                 icon: _layoutMode == 0 | ||||
|                                     ? const Icon(Icons.view_list) | ||||
|                                     : const Icon(Icons.grid_view), | ||||
|                                 onPressed: () { | ||||
|                                   _switchLayout(); | ||||
|                                 }, | ||||
|                               ), | ||||
|                             ], | ||||
|                           ), | ||||
|                         ], | ||||
|                       ).padding(left: 20, right: 16), | ||||
|                     ), | ||||
|                     Expanded( | ||||
|                       child: Material( | ||||
|                         color: | ||||
|                             Theme.of(context).colorScheme.surfaceContainerLow, | ||||
|                         child: Builder( | ||||
|                           builder: (context) { | ||||
|                             switch (_layoutMode) { | ||||
|                               case 1: | ||||
|                                 return _buildGridLayout(); | ||||
|                               default: | ||||
|                                 return _buildListLayout(); | ||||
|                             } | ||||
|                           }, | ||||
|                         ), | ||||
|                       child: ControlsWidget( | ||||
|                         call.room, | ||||
|                         call.room.localParticipant!, | ||||
|                       ), | ||||
|                     ), | ||||
|                     if (call.room.localParticipant != null) | ||||
|                       SizedBox( | ||||
|                         width: MediaQuery.of(context).size.width, | ||||
|                         child: ControlsWidget( | ||||
|                           call.room, | ||||
|                           call.room.localParticipant!, | ||||
|                         ), | ||||
|                       ), | ||||
|                   ], | ||||
|                 ), | ||||
|                 onTap: () {}, | ||||
|                 ], | ||||
|               ), | ||||
|               onTap: () {}, | ||||
|             ), | ||||
|           ); | ||||
|         }); | ||||
|   | ||||
| @@ -10,15 +10,19 @@ import 'package:surface/providers/channel.dart'; | ||||
| import 'package:surface/providers/sn_network.dart'; | ||||
| import 'package:surface/providers/user_directory.dart'; | ||||
| import 'package:surface/providers/userinfo.dart'; | ||||
| import 'package:surface/types/account.dart'; | ||||
| import 'package:surface/types/chat.dart'; | ||||
| import 'package:surface/widgets/account/account_image.dart'; | ||||
| import 'package:surface/widgets/account/account_select.dart'; | ||||
| import 'package:surface/widgets/dialog.dart'; | ||||
| import 'package:surface/widgets/loading_indicator.dart'; | ||||
| import 'package:surface/widgets/navigation/app_scaffold.dart'; | ||||
| import 'package:very_good_infinite_list/very_good_infinite_list.dart'; | ||||
|  | ||||
| class ChannelDetailScreen extends StatefulWidget { | ||||
|   final String scope; | ||||
|   final String alias; | ||||
|  | ||||
|   const ChannelDetailScreen({ | ||||
|     super.key, | ||||
|     required this.scope, | ||||
| @@ -54,8 +58,7 @@ class _ChannelDetailScreenState extends State<ChannelDetailScreen> { | ||||
|  | ||||
|     try { | ||||
|       final sn = context.read<SnNetworkProvider>(); | ||||
|       final resp = await sn.client | ||||
|           .get('/cgi/im/channels/${_channel!.keyPath}/members/me'); | ||||
|       final resp = await sn.client.get('/cgi/im/channels/${_channel!.keyPath}/members/me'); | ||||
|       _profile = SnChannelMember.fromJson(resp.data); | ||||
|       _notifyLevel = _profile!.notify; | ||||
|       if (!mounted) return; | ||||
| @@ -142,6 +145,25 @@ class _ChannelDetailScreenState extends State<ChannelDetailScreen> { | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   Future<void> _addMember(SnAccount related) async { | ||||
|     setState(() => _isBusy = true); | ||||
|  | ||||
|     try { | ||||
|       final sn = context.read<SnNetworkProvider>(); | ||||
|       await sn.client.post( | ||||
|         '/cgi/im/channels/${_channel!.keyPath}/members', | ||||
|         data: {'related': related.name}, | ||||
|       ); | ||||
|       if (!mounted) return; | ||||
|       context.showSnackbar('channelMemberAdded'.tr()); | ||||
|     } catch (err) { | ||||
|       if (!mounted) return; | ||||
|       context.showErrorDialog(err); | ||||
|     } finally { | ||||
|       setState(() => _isBusy = false); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   void _showChannelProfileDetail() { | ||||
|     showDialog( | ||||
|       context: context, | ||||
| @@ -165,13 +187,16 @@ class _ChannelDetailScreenState extends State<ChannelDetailScreen> { | ||||
|     ); | ||||
|   } | ||||
|  | ||||
|   void _showMemberAdd() { | ||||
|     showModalBottomSheet( | ||||
|   void _showMemberAdd() async { | ||||
|     final user = await showModalBottomSheet<SnAccount?>( | ||||
|       context: context, | ||||
|       builder: (context) => _NewChannelMemberWidget( | ||||
|         channel: _channel!, | ||||
|       builder: (context) => AccountSelect( | ||||
|         title: 'channelMemberAdd'.tr(), | ||||
|       ), | ||||
|     ); | ||||
|     if (!mounted) return; | ||||
|     if (user == null) return; | ||||
|     _addMember(user); | ||||
|   } | ||||
|  | ||||
|   @override | ||||
| @@ -189,7 +214,7 @@ class _ChannelDetailScreenState extends State<ChannelDetailScreen> { | ||||
|  | ||||
|     final isOwned = ua.isAuthorized && _channel?.accountId == ua.user?.id; | ||||
|  | ||||
|     return Scaffold( | ||||
|     return AppScaffold( | ||||
|       appBar: AppBar( | ||||
|         title: _channel != null ? Text(_channel!.name) : Text('loading').tr(), | ||||
|       ), | ||||
| @@ -220,11 +245,7 @@ class _ChannelDetailScreenState extends State<ChannelDetailScreen> { | ||||
|               Column( | ||||
|                 crossAxisAlignment: CrossAxisAlignment.start, | ||||
|                 children: [ | ||||
|                   Text('channelDetailPersonalRegion') | ||||
|                       .bold() | ||||
|                       .fontSize(17) | ||||
|                       .tr() | ||||
|                       .padding(horizontal: 20, bottom: 4), | ||||
|                   Text('channelDetailPersonalRegion').bold().fontSize(17).tr().padding(horizontal: 20, bottom: 4), | ||||
|                   ListTile( | ||||
|                     leading: const Icon(Symbols.notifications), | ||||
|                     trailing: DropdownButtonHideUnderline( | ||||
| @@ -263,8 +284,7 @@ class _ChannelDetailScreenState extends State<ChannelDetailScreen> { | ||||
|                   ), | ||||
|                   ListTile( | ||||
|                     leading: AccountImage( | ||||
|                       content: | ||||
|                           ud.getAccountFromCache(_profile!.accountId)?.avatar, | ||||
|                       content: ud.getAccountFromCache(_profile!.accountId)?.avatar, | ||||
|                       radius: 18, | ||||
|                     ), | ||||
|                     trailing: const Icon(Symbols.chevron_right), | ||||
| @@ -283,8 +303,7 @@ class _ChannelDetailScreenState extends State<ChannelDetailScreen> { | ||||
|                       trailing: const Icon(Symbols.chevron_right), | ||||
|                       title: Text('channelActionLeave').tr(), | ||||
|                       subtitle: Text('channelActionLeaveDescription').tr(), | ||||
|                       contentPadding: | ||||
|                           const EdgeInsets.symmetric(horizontal: 24), | ||||
|                       contentPadding: const EdgeInsets.symmetric(horizontal: 24), | ||||
|                       onTap: _leaveChannel, | ||||
|                     ), | ||||
|                 ], | ||||
| @@ -292,11 +311,7 @@ class _ChannelDetailScreenState extends State<ChannelDetailScreen> { | ||||
|             Column( | ||||
|               crossAxisAlignment: CrossAxisAlignment.start, | ||||
|               children: [ | ||||
|                 Text('channelDetailMemberRegion') | ||||
|                     .bold() | ||||
|                     .fontSize(17) | ||||
|                     .tr() | ||||
|                     .padding(horizontal: 20, bottom: 4), | ||||
|                 Text('channelDetailMemberRegion').bold().fontSize(17).tr().padding(horizontal: 20, bottom: 4), | ||||
|                 ListTile( | ||||
|                   leading: const Icon(Symbols.group), | ||||
|                   trailing: const Icon(Symbols.chevron_right), | ||||
| @@ -318,11 +333,7 @@ class _ChannelDetailScreenState extends State<ChannelDetailScreen> { | ||||
|             Column( | ||||
|               crossAxisAlignment: CrossAxisAlignment.start, | ||||
|               children: [ | ||||
|                 Text('channelDetailAdminRegion') | ||||
|                     .bold() | ||||
|                     .fontSize(17) | ||||
|                     .tr() | ||||
|                     .padding(horizontal: 20, bottom: 4), | ||||
|                 Text('channelDetailAdminRegion').bold().fontSize(17).tr().padding(horizontal: 20, bottom: 4), | ||||
|                 ListTile( | ||||
|                   leading: const Icon(Symbols.edit), | ||||
|                   trailing: const Icon(Symbols.chevron_right), | ||||
| @@ -361,18 +372,17 @@ class _ChannelDetailScreenState extends State<ChannelDetailScreen> { | ||||
| class _ChannelProfileDetailDialog extends StatefulWidget { | ||||
|   final SnChannel channel; | ||||
|   final SnChannelMember current; | ||||
|  | ||||
|   const _ChannelProfileDetailDialog({ | ||||
|     required this.channel, | ||||
|     required this.current, | ||||
|   }); | ||||
|  | ||||
|   @override | ||||
|   State<_ChannelProfileDetailDialog> createState() => | ||||
|       _ChannelProfileDetailDialogState(); | ||||
|   State<_ChannelProfileDetailDialog> createState() => _ChannelProfileDetailDialogState(); | ||||
| } | ||||
|  | ||||
| class _ChannelProfileDetailDialogState | ||||
|     extends State<_ChannelProfileDetailDialog> { | ||||
| class _ChannelProfileDetailDialogState extends State<_ChannelProfileDetailDialog> { | ||||
|   bool _isBusy = false; | ||||
|  | ||||
|   final TextEditingController _nickController = TextEditingController(); | ||||
| @@ -443,11 +453,11 @@ class _ChannelProfileDetailDialogState | ||||
|  | ||||
| class _ChannelMemberListWidget extends StatefulWidget { | ||||
|   final SnChannel channel; | ||||
|   const _ChannelMemberListWidget({super.key, required this.channel}); | ||||
|  | ||||
|   const _ChannelMemberListWidget({required this.channel}); | ||||
|  | ||||
|   @override | ||||
|   State<_ChannelMemberListWidget> createState() => | ||||
|       _ChannelMemberListWidgetState(); | ||||
|   State<_ChannelMemberListWidget> createState() => _ChannelMemberListWidgetState(); | ||||
| } | ||||
|  | ||||
| class _ChannelMemberListWidgetState extends State<_ChannelMemberListWidget> { | ||||
| @@ -462,12 +472,10 @@ class _ChannelMemberListWidgetState extends State<_ChannelMemberListWidget> { | ||||
|     try { | ||||
|       final ud = context.read<UserDirectoryProvider>(); | ||||
|       final sn = context.read<SnNetworkProvider>(); | ||||
|       final resp = await sn.client.get( | ||||
|           '/cgi/im/channels/${widget.channel.keyPath}/members', | ||||
|           queryParameters: { | ||||
|             'take': 10, | ||||
|             'offset': 0, | ||||
|           }); | ||||
|       final resp = await sn.client.get('/cgi/im/channels/${widget.channel.keyPath}/members', queryParameters: { | ||||
|         'take': 10, | ||||
|         'offset': 0, | ||||
|       }); | ||||
|       final out = List<SnChannelMember>.from( | ||||
|         resp.data['data']?.map((e) => SnChannelMember.fromJson(e)) ?? [], | ||||
|       ); | ||||
| @@ -525,9 +533,7 @@ class _ChannelMemberListWidgetState extends State<_ChannelMemberListWidget> { | ||||
|           children: [ | ||||
|             const Icon(Symbols.group, size: 24), | ||||
|             const Gap(16), | ||||
|             Text('channelMemberManage') | ||||
|                 .tr() | ||||
|                 .textStyle(Theme.of(context).textTheme.titleLarge!), | ||||
|             Text('channelMemberManage').tr().textStyle(Theme.of(context).textTheme.titleLarge!), | ||||
|           ], | ||||
|         ).padding(horizontal: 20, top: 16, bottom: 12), | ||||
|         Expanded( | ||||
| @@ -538,8 +544,7 @@ class _ChannelMemberListWidgetState extends State<_ChannelMemberListWidget> { | ||||
|             }, | ||||
|             child: InfiniteList( | ||||
|               itemCount: _members.length, | ||||
|               hasReachedMax: | ||||
|                   _totalCount != null && _members.length >= _totalCount!, | ||||
|               hasReachedMax: _totalCount != null && _members.length >= _totalCount!, | ||||
|               isLoading: _isBusy, | ||||
|               onFetchData: _fetchMembers, | ||||
|               itemBuilder: (context, index) { | ||||
| @@ -550,8 +555,7 @@ class _ChannelMemberListWidgetState extends State<_ChannelMemberListWidget> { | ||||
|                     content: ud.getAccountFromCache(member.accountId)?.avatar, | ||||
|                   ), | ||||
|                   title: Text( | ||||
|                     ud.getAccountFromCache(member.accountId)?.name ?? | ||||
|                         'unknown'.tr(), | ||||
|                     ud.getAccountFromCache(member.accountId)?.name ?? 'unknown'.tr(), | ||||
|                   ), | ||||
|                   subtitle: Text(member.nick ?? 'unknown'.tr()), | ||||
|                   trailing: SizedBox( | ||||
| @@ -561,8 +565,7 @@ class _ChannelMemberListWidgetState extends State<_ChannelMemberListWidget> { | ||||
|                       mainAxisAlignment: MainAxisAlignment.end, | ||||
|                       children: [ | ||||
|                         IconButton( | ||||
|                           onPressed: | ||||
|                               _isUpdating ? null : () => _deleteMember(member), | ||||
|                           onPressed: _isUpdating ? null : () => _deleteMember(member), | ||||
|                           icon: const Icon(Symbols.person_remove), | ||||
|                         ), | ||||
|                       ], | ||||
| @@ -577,83 +580,3 @@ class _ChannelMemberListWidgetState extends State<_ChannelMemberListWidget> { | ||||
|     ); | ||||
|   } | ||||
| } | ||||
|  | ||||
| class _NewChannelMemberWidget extends StatefulWidget { | ||||
|   final SnChannel channel; | ||||
|   const _NewChannelMemberWidget({super.key, required this.channel}); | ||||
|  | ||||
|   @override | ||||
|   State<_NewChannelMemberWidget> createState() => | ||||
|       _NewChannelMemberWidgetState(); | ||||
| } | ||||
|  | ||||
| class _NewChannelMemberWidgetState extends State<_NewChannelMemberWidget> { | ||||
|   bool _isBusy = false; | ||||
|  | ||||
|   final TextEditingController _relatedController = TextEditingController(); | ||||
|  | ||||
|   Future<void> _performAction() async { | ||||
|     if (_relatedController.text.isEmpty) return; | ||||
|  | ||||
|     setState(() => _isBusy = true); | ||||
|  | ||||
|     try { | ||||
|       final sn = context.read<SnNetworkProvider>(); | ||||
|       await sn.client.post( | ||||
|         '/cgi/im/channels/${widget.channel.keyPath}/members', | ||||
|         data: { | ||||
|           'related': _relatedController.text, | ||||
|         }, | ||||
|       ); | ||||
|       if (!mounted) return; | ||||
|       Navigator.pop(context, true); | ||||
|       context.showSnackbar('channelMemberAdded'.tr()); | ||||
|     } catch (err) { | ||||
|       if (!mounted) return; | ||||
|       context.showErrorDialog(err); | ||||
|     } finally { | ||||
|       setState(() => _isBusy = false); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   void dispose() { | ||||
|     super.dispose(); | ||||
|     _relatedController.dispose(); | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     return StyledWidget(Column( | ||||
|       crossAxisAlignment: CrossAxisAlignment.start, | ||||
|       children: [ | ||||
|         Text( | ||||
|           'channelMemberAdd', | ||||
|           style: Theme.of(context).textTheme.titleLarge, | ||||
|         ).tr(), | ||||
|         const Gap(12), | ||||
|         TextField( | ||||
|           controller: _relatedController, | ||||
|           readOnly: _isBusy, | ||||
|           autocorrect: false, | ||||
|           autofocus: true, | ||||
|           textCapitalization: TextCapitalization.none, | ||||
|           decoration: InputDecoration( | ||||
|             labelText: 'fieldMemberRelatedName'.tr(), | ||||
|             suffix: SizedBox( | ||||
|               height: 24, | ||||
|               child: IconButton( | ||||
|                 onPressed: _isBusy ? null : () => _performAction(), | ||||
|                 icon: Icon(Symbols.send), | ||||
|                 visualDensity: | ||||
|                     const VisualDensity(horizontal: -4, vertical: -4), | ||||
|                 padding: EdgeInsets.zero, | ||||
|               ), | ||||
|             ), | ||||
|           ), | ||||
|           onTapOutside: (_) => FocusManager.instance.primaryFocus?.unfocus(), | ||||
|         ) | ||||
|       ], | ||||
|     )).padding(all: 24); | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -12,6 +12,7 @@ import 'package:surface/types/realm.dart'; | ||||
| import 'package:surface/widgets/account/account_image.dart'; | ||||
| import 'package:surface/widgets/dialog.dart'; | ||||
| import 'package:surface/widgets/loading_indicator.dart'; | ||||
| import 'package:surface/widgets/navigation/app_scaffold.dart'; | ||||
| import 'package:uuid/uuid.dart'; | ||||
|  | ||||
| class ChatManageScreen extends StatefulWidget { | ||||
| @@ -87,7 +88,7 @@ class _ChatManageScreenState extends State<ChatManageScreen> { | ||||
|     try { | ||||
|       final resp = await sn.client.request( | ||||
|         widget.editingChannelAlias != null | ||||
|             ? '/cgi/im/channels/$scope/${widget.editingChannelAlias}' | ||||
|             ? '/cgi/im/channels/$scope/${_editingChannel!.id}' | ||||
|             : '/cgi/im/channels/$scope', | ||||
|         data: payload, | ||||
|         options: Options( | ||||
| @@ -121,7 +122,7 @@ class _ChatManageScreenState extends State<ChatManageScreen> { | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     return Scaffold( | ||||
|     return AppScaffold( | ||||
|       appBar: AppBar( | ||||
|         title: widget.editingChannelAlias != null | ||||
|             ? Text('screenChatManage').tr() | ||||
|   | ||||
| @@ -1,4 +1,5 @@ | ||||
| import 'dart:async'; | ||||
| import 'dart:developer'; | ||||
|  | ||||
| import 'package:dio/dio.dart'; | ||||
| import 'package:easy_localization/easy_localization.dart'; | ||||
| @@ -9,26 +10,36 @@ import 'package:material_symbols_icons/symbols.dart'; | ||||
| import 'package:provider/provider.dart'; | ||||
| import 'package:styled_widget/styled_widget.dart'; | ||||
| import 'package:surface/controllers/chat_message_controller.dart'; | ||||
| import 'package:surface/controllers/post_write_controller.dart'; | ||||
| import 'package:surface/providers/channel.dart'; | ||||
| import 'package:surface/providers/chat_call.dart'; | ||||
| import 'package:surface/providers/sn_network.dart'; | ||||
| import 'package:surface/providers/user_directory.dart'; | ||||
| import 'package:surface/providers/userinfo.dart'; | ||||
| import 'package:surface/providers/websocket.dart'; | ||||
| import 'package:surface/types/chat.dart'; | ||||
| import 'package:surface/widgets/chat/call/call_prejoin.dart'; | ||||
| import 'package:surface/widgets/chat/chat_message.dart'; | ||||
| import 'package:surface/widgets/chat/chat_message_input.dart'; | ||||
| import 'package:surface/widgets/chat/chat_typing_indicator.dart'; | ||||
| import 'package:surface/widgets/dialog.dart'; | ||||
| import 'package:surface/widgets/loading_indicator.dart'; | ||||
| import 'package:surface/widgets/navigation/app_scaffold.dart'; | ||||
| import 'package:very_good_infinite_list/very_good_infinite_list.dart'; | ||||
|  | ||||
| import '../../providers/user_directory.dart'; | ||||
| import '../../providers/userinfo.dart'; | ||||
| class ChatRoomScreenExtra { | ||||
|   final String? initialText; | ||||
|   final List<PostWriteMedia>? initialAttachments; | ||||
|  | ||||
|   ChatRoomScreenExtra({this.initialText, this.initialAttachments}); | ||||
| } | ||||
|  | ||||
| class ChatRoomScreen extends StatefulWidget { | ||||
|   final String scope; | ||||
|   final String alias; | ||||
|   final ChatRoomScreenExtra? extra; | ||||
|  | ||||
|   const ChatRoomScreen({super.key, required this.scope, required this.alias}); | ||||
|   const ChatRoomScreen({super.key, required this.scope, required this.alias, this.extra}); | ||||
|  | ||||
|   @override | ||||
|   State<ChatRoomScreen> createState() => _ChatRoomScreenState(); | ||||
| @@ -97,7 +108,6 @@ class _ChatRoomScreenState extends State<ChatRoomScreen> { | ||||
|       } | ||||
|     } catch (err) { | ||||
|       if (!mounted) return; | ||||
|       print((err as DioException).response?.data); | ||||
|       context.showErrorDialog(err); | ||||
|     } finally { | ||||
|       setState(() => _isCalling = false); | ||||
| @@ -176,8 +186,23 @@ class _ChatRoomScreenState extends State<ChatRoomScreen> { | ||||
|     _messageController = ChatMessageController(context); | ||||
|     _fetchChannel().then((_) async { | ||||
|       await _messageController.initialize(_channel!); | ||||
|       await _messageController.checkUpdate(); | ||||
|       await _fetchOngoingCall(); | ||||
|  | ||||
|       if (widget.extra != null) { | ||||
|         WidgetsBinding.instance.addPostFrameCallback((_) { | ||||
|           log('[ChatInput] Setting initial text and attachments...'); | ||||
|           if (widget.extra!.initialText != null) { | ||||
|             _inputGlobalKey.currentState?.setInitialText(widget.extra!.initialText!); | ||||
|           } | ||||
|           if (widget.extra!.initialAttachments != null) { | ||||
|             _inputGlobalKey.currentState?.setInitialAttachments(widget.extra!.initialAttachments!); | ||||
|           } | ||||
|         }); | ||||
|       } | ||||
|  | ||||
|       await Future.wait([ | ||||
|         _messageController.checkUpdate(), | ||||
|         _fetchOngoingCall(), | ||||
|       ]); | ||||
|     }); | ||||
|  | ||||
|     final ws = context.read<WebSocketProvider>(); | ||||
| @@ -211,7 +236,7 @@ class _ChatRoomScreenState extends State<ChatRoomScreen> { | ||||
|     final call = context.watch<ChatCallProvider>(); | ||||
|     final ud = context.read<UserDirectoryProvider>(); | ||||
|  | ||||
|     return Scaffold( | ||||
|     return AppScaffold( | ||||
|       appBar: AppBar( | ||||
|         title: Text( | ||||
|           _channel?.type == 1 | ||||
| @@ -281,11 +306,7 @@ class _ChatRoomScreenState extends State<ChatRoomScreen> { | ||||
|                 Expanded( | ||||
|                   child: InfiniteList( | ||||
|                     reverse: true, | ||||
|                     padding: const EdgeInsets.only( | ||||
|                       left: 12, | ||||
|                       right: 12, | ||||
|                       top: 12, | ||||
|                     ), | ||||
|                     padding: const EdgeInsets.only(top: 12), | ||||
|                     hasReachedMax: _messageController.isAllLoaded, | ||||
|                     itemCount: _messageController.messages.length, | ||||
|                     isLoading: _messageController.isLoading, | ||||
| @@ -311,23 +332,20 @@ class _ChatRoomScreenState extends State<ChatRoomScreen> { | ||||
|  | ||||
|                       return Align( | ||||
|                         alignment: Alignment.centerLeft, | ||||
|                         child: Container( | ||||
|                           constraints: BoxConstraints(maxWidth: 480), | ||||
|                           child: ChatMessage( | ||||
|                             data: message, | ||||
|                             isMerged: canMerge, | ||||
|                             hasMerged: canMergePrevious, | ||||
|                             isPending: _messageController.unconfirmedMessages.contains(message.uuid), | ||||
|                             onReply: (value) { | ||||
|                               _inputGlobalKey.currentState?.setReply(value); | ||||
|                             }, | ||||
|                             onEdit: (value) { | ||||
|                               _inputGlobalKey.currentState?.setEdit(value); | ||||
|                             }, | ||||
|                             onDelete: (value) { | ||||
|                               _inputGlobalKey.currentState?.deleteMessage(value); | ||||
|                             }, | ||||
|                           ), | ||||
|                         child: ChatMessage( | ||||
|                           data: message, | ||||
|                           isMerged: canMerge, | ||||
|                           hasMerged: canMergePrevious, | ||||
|                           isPending: _messageController.unconfirmedMessages.contains(message.uuid), | ||||
|                           onReply: (value) { | ||||
|                             _inputGlobalKey.currentState?.setReply(value); | ||||
|                           }, | ||||
|                           onEdit: (value) { | ||||
|                             _inputGlobalKey.currentState?.setEdit(value); | ||||
|                           }, | ||||
|                           onDelete: (value) { | ||||
|                             _inputGlobalKey.currentState?.deleteMessage(value); | ||||
|                           }, | ||||
|                         ), | ||||
|                       ); | ||||
|                     }, | ||||
| @@ -336,11 +354,17 @@ class _ChatRoomScreenState extends State<ChatRoomScreen> { | ||||
|               if (!_messageController.isPending) | ||||
|                 Material( | ||||
|                   elevation: 2, | ||||
|                   child: ChatMessageInput( | ||||
|                     key: _inputGlobalKey, | ||||
|                     otherMember: _otherMember, | ||||
|                     controller: _messageController, | ||||
|                   ).padding(bottom: MediaQuery.of(context).padding.bottom), | ||||
|                   child: Column( | ||||
|                     children: [ | ||||
|                       ChatTypingIndicator(controller: _messageController), | ||||
|                       ChatMessageInput( | ||||
|                         key: _inputGlobalKey, | ||||
|                         otherMember: _otherMember, | ||||
|                         controller: _messageController, | ||||
|                       ), | ||||
|                       Gap(MediaQuery.of(context).padding.bottom), | ||||
|                     ], | ||||
|                   ), | ||||
|                 ), | ||||
|             ], | ||||
|           ); | ||||
|   | ||||
| @@ -1,3 +1,4 @@ | ||||
| import 'package:animations/animations.dart'; | ||||
| import 'package:easy_localization/easy_localization.dart'; | ||||
| import 'package:flutter/material.dart'; | ||||
| import 'package:flutter_expandable_fab/flutter_expandable_fab.dart'; | ||||
| @@ -5,12 +6,31 @@ import 'package:gap/gap.dart'; | ||||
| import 'package:go_router/go_router.dart'; | ||||
| import 'package:material_symbols_icons/symbols.dart'; | ||||
| import 'package:provider/provider.dart'; | ||||
| import 'package:styled_widget/styled_widget.dart'; | ||||
| import 'package:surface/providers/config.dart'; | ||||
| import 'package:surface/providers/post.dart'; | ||||
| import 'package:surface/providers/sn_network.dart'; | ||||
| import 'package:surface/screens/post/post_detail.dart'; | ||||
| import 'package:surface/types/post.dart'; | ||||
| import 'package:surface/widgets/app_bar_leading.dart'; | ||||
| import 'package:surface/widgets/dialog.dart'; | ||||
| import 'package:surface/widgets/navigation/app_scaffold.dart'; | ||||
| import 'package:surface/widgets/post/post_item.dart'; | ||||
| import 'package:very_good_infinite_list/very_good_infinite_list.dart'; | ||||
|  | ||||
| const Map<String, IconData> kCategoryIcons = { | ||||
|   'technology': Symbols.tools_wrench, | ||||
|   'gaming': Symbols.gamepad, | ||||
|   'life': Symbols.nightlife, | ||||
|   'arts': Symbols.format_paint, | ||||
|   'sports': Symbols.sports_soccer, | ||||
|   'music': Symbols.music_note, | ||||
|   'news': Symbols.newspaper, | ||||
|   'knowledge': Symbols.library_books, | ||||
|   'literature': Symbols.book, | ||||
|   'funny': Symbols.attractions, | ||||
| }; | ||||
|  | ||||
| class ExploreScreen extends StatefulWidget { | ||||
|   const ExploreScreen({super.key}); | ||||
|  | ||||
| @@ -24,15 +44,34 @@ class _ExploreScreenState extends State<ExploreScreen> { | ||||
|   bool _isBusy = true; | ||||
|  | ||||
|   final List<SnPost> _posts = List.empty(growable: true); | ||||
|   final List<SnPostCategory> _categories = List.empty(growable: true); | ||||
|   int? _postCount; | ||||
|  | ||||
|   String? _selectedCategory; | ||||
|  | ||||
|   Future<void> _fetchCategories() async { | ||||
|     _categories.clear(); | ||||
|     try { | ||||
|       final sn = context.read<SnNetworkProvider>(); | ||||
|       final resp = await sn.client.get('/cgi/co/categories?take=100'); | ||||
|       _categories.addAll(resp.data.map((e) => SnPostCategory.fromJson(e)).cast<SnPostCategory>() ?? []); | ||||
|     } catch (err) { | ||||
|       if (!mounted) return; | ||||
|       context.showErrorDialog(err); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   Future<void> _fetchPosts() async { | ||||
|     if (_postCount != null && _posts.length >= _postCount!) return; | ||||
|  | ||||
|     setState(() => _isBusy = true); | ||||
|  | ||||
|     final pt = context.read<SnPostContentProvider>(); | ||||
|     final result = await pt.listPosts(take: 10, offset: _posts.length); | ||||
|     final result = await pt.listPosts( | ||||
|       take: 10, | ||||
|       offset: _posts.length, | ||||
|       categories: _selectedCategory != null ? [_selectedCategory!] : null, | ||||
|     ); | ||||
|     final out = result.$1; | ||||
|  | ||||
|     if (!mounted) return; | ||||
| @@ -43,15 +82,24 @@ class _ExploreScreenState extends State<ExploreScreen> { | ||||
|     if (mounted) setState(() => _isBusy = false); | ||||
|   } | ||||
|  | ||||
|   Future<void> _refreshPosts() { | ||||
|     _postCount = null; | ||||
|     _posts.clear(); | ||||
|     return _fetchPosts(); | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   void initState() { | ||||
|     super.initState(); | ||||
|     _fetchPosts(); | ||||
|     _fetchCategories(); | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     return Scaffold( | ||||
|     final cfg = context.read<ConfigProvider>(); | ||||
|  | ||||
|     return AppScaffold( | ||||
|       floatingActionButtonLocation: ExpandableFab.location, | ||||
|       floatingActionButton: ExpandableFab( | ||||
|         key: _fabKey, | ||||
| @@ -59,27 +107,20 @@ class _ExploreScreenState extends State<ExploreScreen> { | ||||
|         type: ExpandableFabType.up, | ||||
|         childrenAnimation: ExpandableFabAnimation.none, | ||||
|         overlayStyle: ExpandableFabOverlayStyle( | ||||
|           color: Theme.of(context) | ||||
|               .colorScheme | ||||
|               .surface | ||||
|               .withAlpha((255 * 0.5).round()), | ||||
|           color: Theme.of(context).colorScheme.surface.withAlpha((255 * 0.5).round()), | ||||
|         ), | ||||
|         openButtonBuilder: RotateFloatingActionButtonBuilder( | ||||
|           child: const Icon(Symbols.add, size: 28), | ||||
|           fabSize: ExpandableFabSize.regular, | ||||
|           foregroundColor: | ||||
|               Theme.of(context).floatingActionButtonTheme.foregroundColor, | ||||
|           backgroundColor: | ||||
|               Theme.of(context).floatingActionButtonTheme.backgroundColor, | ||||
|           foregroundColor: Theme.of(context).floatingActionButtonTheme.foregroundColor, | ||||
|           backgroundColor: Theme.of(context).floatingActionButtonTheme.backgroundColor, | ||||
|           shape: const CircleBorder(), | ||||
|         ), | ||||
|         closeButtonBuilder: DefaultFloatingActionButtonBuilder( | ||||
|           child: const Icon(Symbols.close, size: 28), | ||||
|           fabSize: ExpandableFabSize.regular, | ||||
|           foregroundColor: | ||||
|               Theme.of(context).floatingActionButtonTheme.foregroundColor, | ||||
|           backgroundColor: | ||||
|               Theme.of(context).floatingActionButtonTheme.backgroundColor, | ||||
|           foregroundColor: Theme.of(context).floatingActionButtonTheme.foregroundColor, | ||||
|           backgroundColor: Theme.of(context).floatingActionButtonTheme.backgroundColor, | ||||
|           shape: const CircleBorder(), | ||||
|         ), | ||||
|         children: [ | ||||
| @@ -95,8 +136,7 @@ class _ExploreScreenState extends State<ExploreScreen> { | ||||
|                     'mode': 'stories', | ||||
|                   }).then((value) { | ||||
|                     if (value == true) { | ||||
|                       _posts.clear(); | ||||
|                       _fetchPosts(); | ||||
|                       _refreshPosts(); | ||||
|                     } | ||||
|                   }); | ||||
|                   _fabKey.currentState!.toggle(); | ||||
| @@ -117,8 +157,7 @@ class _ExploreScreenState extends State<ExploreScreen> { | ||||
|                     'mode': 'articles', | ||||
|                   }).then((value) { | ||||
|                     if (value == true) { | ||||
|                       _posts.clear(); | ||||
|                       _fetchPosts(); | ||||
|                       _refreshPosts(); | ||||
|                     } | ||||
|                   }); | ||||
|                   _fabKey.currentState!.toggle(); | ||||
| @@ -131,10 +170,7 @@ class _ExploreScreenState extends State<ExploreScreen> { | ||||
|       ), | ||||
|       body: RefreshIndicator( | ||||
|         displacement: 40 + MediaQuery.of(context).padding.top, | ||||
|         onRefresh: () { | ||||
|           _posts.clear(); | ||||
|           return _fetchPosts(); | ||||
|         }, | ||||
|         onRefresh: () => _refreshPosts(), | ||||
|         child: CustomScrollView( | ||||
|           slivers: [ | ||||
|             SliverAppBar( | ||||
| @@ -151,7 +187,36 @@ class _ExploreScreenState extends State<ExploreScreen> { | ||||
|                 ), | ||||
|                 const Gap(8), | ||||
|               ], | ||||
|               bottom: PreferredSize( | ||||
|                 preferredSize: const Size.fromHeight(50), | ||||
|                 child: SizedBox( | ||||
|                   height: 50, | ||||
|                   child: SingleChildScrollView( | ||||
|                     scrollDirection: Axis.horizontal, | ||||
|                     padding: const EdgeInsets.only(left: 8, right: 8, bottom: 12), | ||||
|                     child: Row( | ||||
|                       mainAxisAlignment: MainAxisAlignment.center, | ||||
|                       children: _categories.map((ele) { | ||||
|                         return StyledWidget(ChoiceChip( | ||||
|                           avatar: Icon(kCategoryIcons[ele.alias] ?? Symbols.question_mark), | ||||
|                           label: Text( | ||||
|                             'postCategory${ele.alias.capitalize()}'.trExists() | ||||
|                                 ? 'postCategory${ele.alias.capitalize()}'.tr() | ||||
|                                 : ele.name, | ||||
|                           ), | ||||
|                           selected: _selectedCategory == ele.alias, | ||||
|                           onSelected: (value) { | ||||
|                             _selectedCategory = value ? ele.alias : null; | ||||
|                             _refreshPosts(); | ||||
|                           }, | ||||
|                         )).padding(horizontal: 4); | ||||
|                       }).toList(), | ||||
|                     ), | ||||
|                   ), | ||||
|                 ), | ||||
|               ), | ||||
|             ), | ||||
|             const SliverGap(12), | ||||
|             SliverInfiniteList( | ||||
|               itemCount: _posts.length, | ||||
|               isLoading: _isBusy, | ||||
| @@ -159,28 +224,39 @@ class _ExploreScreenState extends State<ExploreScreen> { | ||||
|               hasReachedMax: _postCount != null && _posts.length >= _postCount!, | ||||
|               onFetchData: _fetchPosts, | ||||
|               itemBuilder: (context, idx) { | ||||
|                 return GestureDetector( | ||||
|                   child: PostItem( | ||||
|                     data: _posts[idx], | ||||
|                     maxWidth: 640, | ||||
|                     onChanged: (data) { | ||||
|                       setState(() => _posts[idx] = data); | ||||
|                     }, | ||||
|                     onDeleted: () { | ||||
|                       _posts.clear(); | ||||
|                       _fetchPosts(); | ||||
|                     }, | ||||
|                 return Center( | ||||
|                   child: OpenContainer( | ||||
|                     closedBuilder: (_, __) => Container( | ||||
|                       constraints: const BoxConstraints(maxWidth: 640), | ||||
|                       child: PostItem( | ||||
|                         data: _posts[idx], | ||||
|                         maxWidth: 640, | ||||
|                         onChanged: (data) { | ||||
|                           setState(() => _posts[idx] = data); | ||||
|                         }, | ||||
|                         onDeleted: () { | ||||
|                           _refreshPosts(); | ||||
|                         }, | ||||
|                       ), | ||||
|                     ), | ||||
|                     openBuilder: (_, close) => PostDetailScreen( | ||||
|                       slug: _posts[idx].id.toString(), | ||||
|                       preload: _posts[idx], | ||||
|                       onBack: close, | ||||
|                     ), | ||||
|                     openColor: Colors.transparent, | ||||
|                     openElevation: 0, | ||||
|                     transitionType: ContainerTransitionType.fade, | ||||
|                     closedColor: Theme.of(context).colorScheme.surfaceContainerLow.withOpacity( | ||||
|                           cfg.prefs.getBool(kAppBackgroundStoreKey) == true ? 0.75 : 1, | ||||
|                         ), | ||||
|                     closedShape: const RoundedRectangleBorder( | ||||
|                       borderRadius: BorderRadius.all(Radius.circular(16)), | ||||
|                     ), | ||||
|                   ), | ||||
|                   onTap: () { | ||||
|                     GoRouter.of(context).pushNamed( | ||||
|                       'postDetail', | ||||
|                       pathParameters: {'slug': _posts[idx].id.toString()}, | ||||
|                       extra: _posts[idx], | ||||
|                     ); | ||||
|                   }, | ||||
|                 ); | ||||
|               }, | ||||
|               separatorBuilder: (context, index) => const Divider(height: 1), | ||||
|               separatorBuilder: (_, __) => const Gap(8), | ||||
|             ), | ||||
|           ], | ||||
|         ), | ||||
|   | ||||
| @@ -6,14 +6,15 @@ import 'package:provider/provider.dart'; | ||||
| import 'package:styled_widget/styled_widget.dart'; | ||||
| import 'package:surface/providers/relationship.dart'; | ||||
| import 'package:surface/providers/sn_network.dart'; | ||||
| import 'package:surface/providers/userinfo.dart'; | ||||
| import 'package:surface/types/account.dart'; | ||||
| import 'package:surface/widgets/account/account_image.dart'; | ||||
| import 'package:surface/widgets/account/account_select.dart'; | ||||
| import 'package:surface/widgets/app_bar_leading.dart'; | ||||
| import 'package:surface/widgets/dialog.dart'; | ||||
| import 'package:surface/widgets/loading_indicator.dart'; | ||||
|  | ||||
| import '../providers/userinfo.dart'; | ||||
| import '../widgets/unauthorized_hint.dart'; | ||||
| import 'package:surface/widgets/navigation/app_scaffold.dart'; | ||||
| import 'package:surface/widgets/unauthorized_hint.dart'; | ||||
|  | ||||
| const kFriendStatus = { | ||||
|   0: 'friendStatusPending', | ||||
| @@ -167,6 +168,24 @@ class _FriendScreenState extends State<FriendScreen> { | ||||
|     }); | ||||
|   } | ||||
|  | ||||
|   Future<void> _sendRequest(SnAccount user) async { | ||||
|     setState(() => _isBusy = true); | ||||
|  | ||||
|     try { | ||||
|       final sn = context.read<SnNetworkProvider>(); | ||||
|       await sn.client.post('/cgi/id/users/me/relations', data: { | ||||
|         'related': user.name, | ||||
|       }); | ||||
|       if (!mounted) return; | ||||
|       context.showSnackbar('friendRequestSent'.tr()); | ||||
|     } catch (err) { | ||||
|       if (!mounted) return; | ||||
|       context.showErrorDialog(err); | ||||
|     } finally { | ||||
|       setState(() => _isBusy = false); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   void initState() { | ||||
|     super.initState(); | ||||
| @@ -180,7 +199,7 @@ class _FriendScreenState extends State<FriendScreen> { | ||||
|     final ua = context.read<UserProvider>(); | ||||
|  | ||||
|     if (!ua.isAuthorized) { | ||||
|       return Scaffold( | ||||
|       return AppScaffold( | ||||
|         appBar: AppBar( | ||||
|           leading: AutoAppBarLeading(), | ||||
|           title: Text('screenFriend').tr(), | ||||
| @@ -191,18 +210,23 @@ class _FriendScreenState extends State<FriendScreen> { | ||||
|       ); | ||||
|     } | ||||
|  | ||||
|     return Scaffold( | ||||
|     return AppScaffold( | ||||
|       appBar: AppBar( | ||||
|         leading: AutoAppBarLeading(), | ||||
|         title: Text('screenFriend').tr(), | ||||
|       ), | ||||
|       floatingActionButton: FloatingActionButton( | ||||
|         child: const Icon(Symbols.add), | ||||
|         onPressed: () { | ||||
|           showModalBottomSheet( | ||||
|         onPressed: () async { | ||||
|           final user = await showModalBottomSheet<SnAccount?>( | ||||
|             context: context, | ||||
|             builder: (context) => _NewFriendWidget(), | ||||
|             builder: (context) => AccountSelect( | ||||
|               title: 'friendNew'.tr(), | ||||
|             ), | ||||
|           ); | ||||
|           if (!mounted) return; | ||||
|           if (user == null) return; | ||||
|           _sendRequest(user); | ||||
|         }, | ||||
|       ), | ||||
|       body: Column( | ||||
| @@ -230,55 +254,54 @@ class _FriendScreenState extends State<FriendScreen> { | ||||
|               trailing: const Icon(Symbols.chevron_right), | ||||
|               onTap: _showBlocks, | ||||
|             ), | ||||
|           if (_requests.isNotEmpty || _blocks.isNotEmpty) | ||||
|             const Divider(height: 1), | ||||
|           if (_requests.isNotEmpty || _blocks.isNotEmpty) const Divider(height: 1), | ||||
|           Expanded( | ||||
|             child: RefreshIndicator( | ||||
|               onRefresh: () => Future.wait([ | ||||
|                 _fetchRelations(), | ||||
|                 _fetchRequests(), | ||||
|               ]), | ||||
|               child: ListView.builder( | ||||
|                 itemCount: _relations.length, | ||||
|                 itemBuilder: (context, index) { | ||||
|                   final relation = _relations[index]; | ||||
|                   final other = relation.related; | ||||
|                   return ListTile( | ||||
|                     contentPadding: const EdgeInsets.only(right: 24, left: 16), | ||||
|                     leading: AccountImage(content: other?.avatar), | ||||
|                     title: Text(other?.nick ?? 'unknown'), | ||||
|                     subtitle: Text(other?.nick ?? 'unknown'), | ||||
|                     trailing: SizedBox( | ||||
|                       height: 48, | ||||
|                       width: 120, | ||||
|                       child: Column( | ||||
|                         mainAxisSize: MainAxisSize.min, | ||||
|                         mainAxisAlignment: MainAxisAlignment.center, | ||||
|                         crossAxisAlignment: CrossAxisAlignment.end, | ||||
|                         children: [ | ||||
|                           Row( | ||||
|                             mainAxisAlignment: MainAxisAlignment.end, | ||||
|                             children: [ | ||||
|                               InkWell( | ||||
|                                 onTap: _isUpdating | ||||
|                                     ? null | ||||
|                                     : () => _changeRelation(relation, 2), | ||||
|                                 child: Text('friendBlock').tr(), | ||||
|                               ), | ||||
|                               const Gap(8), | ||||
|                               InkWell( | ||||
|                                 onTap: _isUpdating | ||||
|                                     ? null | ||||
|                                     : () => _deleteRelation(relation), | ||||
|                                 child: Text('friendDeleteAction').tr(), | ||||
|                               ), | ||||
|                             ], | ||||
|                           ), | ||||
|                         ], | ||||
|             child: MediaQuery.removePadding( | ||||
|               context: context, | ||||
|               removeTop: true, | ||||
|               child: RefreshIndicator( | ||||
|                 onRefresh: () => Future.wait([ | ||||
|                   _fetchRelations(), | ||||
|                   _fetchRequests(), | ||||
|                 ]), | ||||
|                 child: ListView.builder( | ||||
|                   itemCount: _relations.length, | ||||
|                   itemBuilder: (context, index) { | ||||
|                     final relation = _relations[index]; | ||||
|                     final other = relation.related; | ||||
|                     return ListTile( | ||||
|                       contentPadding: const EdgeInsets.only(right: 24, left: 16), | ||||
|                       leading: AccountImage(content: other?.avatar), | ||||
|                       title: Text(other?.nick ?? 'unknown'), | ||||
|                       subtitle: Text(other?.nick ?? 'unknown'), | ||||
|                       trailing: SizedBox( | ||||
|                         height: 48, | ||||
|                         width: 120, | ||||
|                         child: Column( | ||||
|                           mainAxisSize: MainAxisSize.min, | ||||
|                           mainAxisAlignment: MainAxisAlignment.center, | ||||
|                           crossAxisAlignment: CrossAxisAlignment.end, | ||||
|                           children: [ | ||||
|                             Row( | ||||
|                               mainAxisAlignment: MainAxisAlignment.end, | ||||
|                               children: [ | ||||
|                                 InkWell( | ||||
|                                   onTap: _isUpdating ? null : () => _changeRelation(relation, 2), | ||||
|                                   child: Text('friendBlock').tr(), | ||||
|                                 ), | ||||
|                                 const Gap(8), | ||||
|                                 InkWell( | ||||
|                                   onTap: _isUpdating ? null : () => _deleteRelation(relation), | ||||
|                                   child: Text('friendDeleteAction').tr(), | ||||
|                                 ), | ||||
|                               ], | ||||
|                             ), | ||||
|                           ], | ||||
|                         ), | ||||
|                       ), | ||||
|                     ), | ||||
|                   ); | ||||
|                 }, | ||||
|                     ); | ||||
|                   }, | ||||
|                 ), | ||||
|               ), | ||||
|             ), | ||||
|           ), | ||||
| @@ -288,84 +311,10 @@ class _FriendScreenState extends State<FriendScreen> { | ||||
|   } | ||||
| } | ||||
|  | ||||
| class _NewFriendWidget extends StatefulWidget { | ||||
|   const _NewFriendWidget({super.key}); | ||||
|  | ||||
|   @override | ||||
|   State<_NewFriendWidget> createState() => _NewFriendWidgetState(); | ||||
| } | ||||
|  | ||||
| class _NewFriendWidgetState extends State<_NewFriendWidget> { | ||||
|   bool _isBusy = false; | ||||
|  | ||||
|   final TextEditingController _relatedController = TextEditingController(); | ||||
|  | ||||
|   Future<void> _sendRequest() async { | ||||
|     if (_relatedController.text.isEmpty) return; | ||||
|  | ||||
|     setState(() => _isBusy = true); | ||||
|  | ||||
|     try { | ||||
|       final sn = context.read<SnNetworkProvider>(); | ||||
|       await sn.client.post('/cgi/id/users/me/relations', data: { | ||||
|         'related': _relatedController.text, | ||||
|       }); | ||||
|       if (!mounted) return; | ||||
|       Navigator.pop(context, true); | ||||
|       context.showSnackbar('friendRequestSent'.tr()); | ||||
|     } catch (err) { | ||||
|       if (!mounted) return; | ||||
|       context.showErrorDialog(err); | ||||
|     } finally { | ||||
|       setState(() => _isBusy = false); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   void dispose() { | ||||
|     super.dispose(); | ||||
|     _relatedController.dispose(); | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     return StyledWidget(Column( | ||||
|       crossAxisAlignment: CrossAxisAlignment.start, | ||||
|       children: [ | ||||
|         Text( | ||||
|           'friendNew', | ||||
|           style: Theme.of(context).textTheme.titleLarge, | ||||
|         ).tr(), | ||||
|         const Gap(12), | ||||
|         TextField( | ||||
|           controller: _relatedController, | ||||
|           readOnly: _isBusy, | ||||
|           autocorrect: false, | ||||
|           autofocus: true, | ||||
|           textCapitalization: TextCapitalization.none, | ||||
|           decoration: InputDecoration( | ||||
|             labelText: 'fieldFriendRelatedName'.tr(), | ||||
|             suffix: SizedBox( | ||||
|               height: 24, | ||||
|               child: IconButton( | ||||
|                 onPressed: _isBusy ? null : () => _sendRequest(), | ||||
|                 icon: Icon(Symbols.send), | ||||
|                 visualDensity: | ||||
|                     const VisualDensity(horizontal: -4, vertical: -4), | ||||
|                 padding: EdgeInsets.zero, | ||||
|               ), | ||||
|             ), | ||||
|           ), | ||||
|           onTapOutside: (_) => FocusManager.instance.primaryFocus?.unfocus(), | ||||
|         ) | ||||
|       ], | ||||
|     )).padding(all: 24); | ||||
|   } | ||||
| } | ||||
|  | ||||
| class _FriendshipListWidget extends StatefulWidget { | ||||
|   final List<SnRelationship> relations; | ||||
|   const _FriendshipListWidget({super.key, required this.relations}); | ||||
|  | ||||
|   const _FriendshipListWidget({required this.relations}); | ||||
|  | ||||
|   @override | ||||
|   State<_FriendshipListWidget> createState() => _FriendshipListWidgetState(); | ||||
| @@ -471,9 +420,7 @@ class _FriendshipListWidgetState extends State<_FriendshipListWidget> { | ||||
|               mainAxisAlignment: MainAxisAlignment.center, | ||||
|               crossAxisAlignment: CrossAxisAlignment.end, | ||||
|               children: [ | ||||
|                 Text(kFriendStatus[relation.status] ?? 'unknown') | ||||
|                     .tr() | ||||
|                     .opacity(0.75), | ||||
|                 Text(kFriendStatus[relation.status] ?? 'unknown').tr().opacity(0.75), | ||||
|                 if (relation.status == 0) | ||||
|                   Row( | ||||
|                     mainAxisAlignment: MainAxisAlignment.end, | ||||
| @@ -494,8 +441,7 @@ class _FriendshipListWidgetState extends State<_FriendshipListWidget> { | ||||
|                     mainAxisAlignment: MainAxisAlignment.end, | ||||
|                     children: [ | ||||
|                       InkWell( | ||||
|                         onTap: | ||||
|                             _isBusy ? null : () => _changeRelation(relation, 1), | ||||
|                         onTap: _isBusy ? null : () => _changeRelation(relation, 1), | ||||
|                         child: Text('friendUnblock').tr(), | ||||
|                       ), | ||||
|                       const Gap(8), | ||||
|   | ||||
| @@ -9,22 +9,27 @@ import 'package:flutter_staggered_grid_view/flutter_staggered_grid_view.dart'; | ||||
| import 'package:gap/gap.dart'; | ||||
| import 'package:go_router/go_router.dart'; | ||||
| import 'package:google_fonts/google_fonts.dart'; | ||||
| import 'package:html/parser.dart'; | ||||
| import 'package:material_symbols_icons/symbols.dart'; | ||||
| import 'package:provider/provider.dart'; | ||||
| import 'package:relative_time/relative_time.dart'; | ||||
| import 'package:slide_countdown/slide_countdown.dart'; | ||||
| import 'package:styled_widget/styled_widget.dart'; | ||||
| import 'package:flutter/material.dart'; | ||||
| import 'package:surface/providers/config.dart'; | ||||
| import 'package:surface/providers/post.dart'; | ||||
| import 'package:surface/providers/sn_network.dart'; | ||||
| import 'package:surface/providers/special_day.dart'; | ||||
| import 'package:surface/providers/userinfo.dart'; | ||||
| import 'package:surface/providers/widget.dart'; | ||||
| import 'package:surface/types/check_in.dart'; | ||||
| import 'package:surface/types/news.dart'; | ||||
| import 'package:surface/types/post.dart'; | ||||
| import 'package:surface/widgets/app_bar_leading.dart'; | ||||
| import 'package:surface/widgets/dialog.dart'; | ||||
| import 'package:surface/widgets/navigation/app_scaffold.dart'; | ||||
| import 'package:surface/widgets/post/post_item.dart'; | ||||
|  | ||||
| import '../providers/widget.dart'; | ||||
|  | ||||
| class HomeScreenDashEntry { | ||||
|   final String name; | ||||
|   final Widget child; | ||||
| @@ -46,12 +51,12 @@ class HomeScreen extends StatefulWidget { | ||||
| } | ||||
|  | ||||
| class _HomeScreenState extends State<HomeScreen> { | ||||
|   static const List<HomeScreenDashEntry> kCards = [ | ||||
|   late final List<HomeScreenDashEntry> kCards = [ | ||||
|     HomeScreenDashEntry( | ||||
|       name: 'dashEntryRecommendation', | ||||
|       cols: 2, | ||||
|       rows: 2, | ||||
|       child: _HomeDashRecommendationPostWidget(), | ||||
|       rows: 2, | ||||
|       cols: 2, | ||||
|     ), | ||||
|     HomeScreenDashEntry( | ||||
|       name: 'dashEntryCheckIn', | ||||
| @@ -61,11 +66,16 @@ class _HomeScreenState extends State<HomeScreen> { | ||||
|       name: 'dashEntryNotification', | ||||
|       child: _HomeDashNotificationWidget(), | ||||
|     ), | ||||
|     HomeScreenDashEntry( | ||||
|       name: 'dashEntryTodayNews', | ||||
|       child: _HomeDashTodayNews(), | ||||
|       cols: MediaQuery.of(context).size.width >= 640 ? 3 : 2, | ||||
|     ), | ||||
|   ]; | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     return Scaffold( | ||||
|     return AppScaffold( | ||||
|       appBar: AppBar( | ||||
|         leading: AutoAppBarLeading(), | ||||
|         title: Text("screenHome").tr(), | ||||
| @@ -80,8 +90,8 @@ class _HomeScreenState extends State<HomeScreen> { | ||||
|                 child: Column( | ||||
|                   mainAxisAlignment: constraints.maxWidth > 640 ? MainAxisAlignment.center : MainAxisAlignment.start, | ||||
|                   children: [ | ||||
|                     _HomeDashSpecialDayWidget().padding(bottom: 8, horizontal: 8), | ||||
|                     _HomeDashUpdateWidget(padding: const EdgeInsets.only(bottom: 8, left: 8, right: 8)), | ||||
|                     _HomeDashSpecialDayWidget().padding(horizontal: 8), | ||||
|                     StaggeredGrid.extent( | ||||
|                       maxCrossAxisExtent: 280, | ||||
|                       mainAxisSpacing: 8, | ||||
| @@ -108,7 +118,7 @@ class _HomeScreenState extends State<HomeScreen> { | ||||
| class _HomeDashUpdateWidget extends StatelessWidget { | ||||
|   final EdgeInsets? padding; | ||||
|  | ||||
|   const _HomeDashUpdateWidget({super.key, this.padding}); | ||||
|   const _HomeDashUpdateWidget({this.padding}); | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
| @@ -133,7 +143,7 @@ class _HomeDashUpdateWidget extends StatelessWidget { | ||||
|                           final model = UpdateModel( | ||||
|                             'https://files.solsynth.dev/d/production01/solian/app-arm64-v8a-release.apk', | ||||
|                             'solian-app-release-${config.updatableVersion!}.apk', | ||||
|                             'ic_notification', | ||||
|                             'ic_launcher', | ||||
|                             'https://apps.apple.com/us/app/solian/id6499032345', | ||||
|                           ); | ||||
|                           AzhonAppUpdate.update(model); | ||||
| @@ -151,47 +161,184 @@ class _HomeDashUpdateWidget extends StatelessWidget { | ||||
|   } | ||||
| } | ||||
|  | ||||
| class _HomeDashSpecialDayWidget extends StatelessWidget { | ||||
|   const _HomeDashSpecialDayWidget({super.key}); | ||||
| class _HomeDashSpecialDayWidget extends StatefulWidget { | ||||
|   const _HomeDashSpecialDayWidget(); | ||||
|  | ||||
|   @override | ||||
|   State<_HomeDashSpecialDayWidget> createState() => _HomeDashSpecialDayWidgetState(); | ||||
| } | ||||
|  | ||||
| class _HomeDashSpecialDayWidgetState extends State<_HomeDashSpecialDayWidget> { | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     final ua = context.watch<UserProvider>(); | ||||
|     final today = DateTime.now(); | ||||
|     final birthday = ua.user?.profile?.birthday?.toLocal(); | ||||
|     final isBirthday = birthday != null && birthday.day == today.day && birthday.month == today.month; | ||||
|     final dayz = context.watch<SpecialDayProvider>(); | ||||
|  | ||||
|     return Column( | ||||
|       spacing: 8, | ||||
|       children: [ | ||||
|         if (isBirthday) | ||||
|           Card( | ||||
|             child: ListTile( | ||||
|               leading: Text('🎂').fontSize(24), | ||||
|               title: Text('happyBirthday').tr(args: [ua.user?.nick ?? 'user']), | ||||
|             ), | ||||
|           ).padding(bottom: 8), | ||||
|         if (today.month == 12 && today.day == 25) | ||||
|           Card( | ||||
|             child: ListTile( | ||||
|               leading: Text('🎄').fontSize(24), | ||||
|               title: Text('celebrateMerryXmas').tr(args: [ua.user?.nick ?? 'user']), | ||||
|     final days = dayz.getSpecialDays(); | ||||
|  | ||||
|     if (days.isNotEmpty) { | ||||
|       return Column( | ||||
|           children: days.map((ele) { | ||||
|         return Card( | ||||
|           child: ListTile( | ||||
|             leading: Text(kSpecialDaysSymbol[ele] ?? '🎉').fontSize(24), | ||||
|             title: Text('celebrate$ele').tr(args: [ua.user?.nick ?? 'user']), | ||||
|             subtitle: Text( | ||||
|               DateFormat('y/M/d').format(DateTime.now().copyWith( | ||||
|                 month: kSpecialDays[ele]?.$1, | ||||
|                 day: kSpecialDays[ele]?.$2, | ||||
|               )), | ||||
|             ), | ||||
|           ), | ||||
|         if (today.month == 1 && today.day == 1) | ||||
|           Card( | ||||
|             child: ListTile( | ||||
|               leading: Text('🎉').fontSize(24), | ||||
|               title: Text('celebrateNewYear').tr(args: [ua.user?.nick ?? 'user']), | ||||
|             ), | ||||
|         ).padding(bottom: 8); | ||||
|       }).toList()); | ||||
|     } | ||||
|  | ||||
|     final nextOne = dayz.getNextSpecialDay(); | ||||
|     final lastOne = dayz.getLastSpecialDay(); | ||||
|  | ||||
|     if (nextOne != null && lastOne != null) { | ||||
|       var (name, date) = nextOne; | ||||
|       date = date.add(Duration(days: 1)); | ||||
|       final progress = dayz.getSpecialDayProgress(lastOne.$2, date); | ||||
|       final diff = nextOne.$2.difference(DateTime.now()); | ||||
|       return Card( | ||||
|         child: ListTile( | ||||
|           leading: Text(kSpecialDaysSymbol[name] ?? '🎉').fontSize(24), | ||||
|           title: Text('pending$name').tr(args: [RelativeTime(context).format(date).replaceFirst('in', '').trim()]), | ||||
|           subtitle: Row( | ||||
|             crossAxisAlignment: CrossAxisAlignment.center, | ||||
|             children: [ | ||||
|               SlideCountdown( | ||||
|                 duration: diff, | ||||
|                 style: GoogleFonts.robotoMono(fontSize: 13), | ||||
|                 separatorStyle: GoogleFonts.robotoMono(fontSize: 13), | ||||
|                 separatorType: SeparatorType.symbol, | ||||
|                 decoration: BoxDecoration(), | ||||
|                 padding: EdgeInsets.zero, | ||||
|                 onDone: () { | ||||
|                   setState(() {}); | ||||
|                 }, | ||||
|               ), | ||||
|               const Gap(12), | ||||
|               Expanded( | ||||
|                 child: LinearProgressIndicator( | ||||
|                   value: progress, | ||||
|                   borderRadius: BorderRadius.circular(8), | ||||
|                 ), | ||||
|               ), | ||||
|             ], | ||||
|           ), | ||||
|       ], | ||||
|         ), | ||||
|       ).padding(bottom: 8); | ||||
|     } | ||||
|  | ||||
|     return const SizedBox.shrink(); | ||||
|   } | ||||
| } | ||||
|  | ||||
| class _HomeDashTodayNews extends StatefulWidget { | ||||
|   const _HomeDashTodayNews(); | ||||
|  | ||||
|   @override | ||||
|   State<_HomeDashTodayNews> createState() => _HomeDashTodayNewsState(); | ||||
| } | ||||
|  | ||||
| class _HomeDashTodayNewsState extends State<_HomeDashTodayNews> { | ||||
|   SnNewsArticle? _article; | ||||
|  | ||||
|   Future<void> _fetchArticle() async { | ||||
|     try { | ||||
|       final sn = context.read<SnNetworkProvider>(); | ||||
|       final resp = await sn.client.get('/cgi/re/news/today'); | ||||
|       _article = SnNewsArticle.fromJson(resp.data['data']); | ||||
|     } catch (err) { | ||||
|       if (!mounted) return; | ||||
|       context.showErrorDialog(err); | ||||
|       rethrow; | ||||
|     } finally { | ||||
|       setState(() {}); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   initState() { | ||||
|     super.initState(); | ||||
|     _fetchArticle(); | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     return Card( | ||||
|       child: Column( | ||||
|         crossAxisAlignment: CrossAxisAlignment.start, | ||||
|         children: [ | ||||
|           Row( | ||||
|             children: [ | ||||
|               const Icon(Symbols.newspaper), | ||||
|               const Gap(8), | ||||
|               Text( | ||||
|                 'newsToday', | ||||
|                 style: Theme.of(context).textTheme.titleLarge, | ||||
|               ).tr() | ||||
|             ], | ||||
|           ).padding(horizontal: 18, top: 12, bottom: 8), | ||||
|           if (_article != null) | ||||
|             Expanded( | ||||
|               child: InkWell( | ||||
|                 borderRadius: BorderRadius.all(Radius.circular(8)), | ||||
|                 child: Column( | ||||
|                   crossAxisAlignment: CrossAxisAlignment.start, | ||||
|                   spacing: 4, | ||||
|                   children: [ | ||||
|                     Text( | ||||
|                       _article!.title, | ||||
|                       style: Theme.of(context).textTheme.titleMedium!.copyWith(fontSize: 18), | ||||
|                       maxLines: MediaQuery.of(context).size.width >= 640 ? 2 : 1, | ||||
|                       overflow: TextOverflow.ellipsis, | ||||
|                     ), | ||||
|                     Text( | ||||
|                       parse(_article!.description).children.map((e) => e.text.trim()).join(), | ||||
|                       maxLines: 3, | ||||
|                       overflow: TextOverflow.ellipsis, | ||||
|                       style: Theme.of(context).textTheme.bodyMedium, | ||||
|                     ), | ||||
|                     Builder(builder: (context) { | ||||
|                       final date = _article!.publishedAt ?? _article!.createdAt; | ||||
|                       return Row( | ||||
|                         crossAxisAlignment: CrossAxisAlignment.center, | ||||
|                         spacing: 2, | ||||
|                         children: [ | ||||
|                           Text(DateFormat().format(date)).textStyle(Theme.of(context).textTheme.bodySmall!), | ||||
|                           Text(' · ').textStyle(Theme.of(context).textTheme.bodySmall!).bold(), | ||||
|                           Text(RelativeTime(context).format(date)).textStyle(Theme.of(context).textTheme.bodySmall!), | ||||
|                         ], | ||||
|                       ).opacity(0.75); | ||||
|                     }), | ||||
|                   ], | ||||
|                 ).padding(horizontal: 16), | ||||
|                 onTap: () { | ||||
|                   GoRouter.of(context).pushNamed( | ||||
|                     'newsDetail', | ||||
|                     pathParameters: {'hash': _article!.hash}, | ||||
|                   ); | ||||
|                 }, | ||||
|               ), | ||||
|             ) | ||||
|           else | ||||
|             Expanded( | ||||
|               child: Center( | ||||
|                 child: CircularProgressIndicator(), | ||||
|               ), | ||||
|             ) | ||||
|         ], | ||||
|       ), | ||||
|     ); | ||||
|   } | ||||
| } | ||||
|  | ||||
| class _HomeDashCheckInWidget extends StatefulWidget { | ||||
|   const _HomeDashCheckInWidget({super.key}); | ||||
|   const _HomeDashCheckInWidget(); | ||||
|  | ||||
|   @override | ||||
|   State<_HomeDashCheckInWidget> createState() => _HomeDashCheckInWidgetState(); | ||||
| @@ -212,7 +359,7 @@ class _HomeDashCheckInWidgetState extends State<_HomeDashCheckInWidget> { | ||||
|       final home = context.read<HomeWidgetProvider>(); | ||||
|       final resp = await sn.client.get('/cgi/id/check-in/today'); | ||||
|       _todayRecord = SnCheckInRecord.fromJson(resp.data); | ||||
|       home.saveWidgetData('pas_check_in_record', _todayRecord!.toJson()); | ||||
|       await home.saveWidgetData('pas_check_in_record', _todayRecord!.toJson()); | ||||
|     } finally { | ||||
|       setState(() => _isBusy = false); | ||||
|     } | ||||
| @@ -225,7 +372,7 @@ class _HomeDashCheckInWidgetState extends State<_HomeDashCheckInWidget> { | ||||
|       final home = context.read<HomeWidgetProvider>(); | ||||
|       final resp = await sn.client.post('/cgi/id/check-in'); | ||||
|       _todayRecord = SnCheckInRecord.fromJson(resp.data); | ||||
|       home.saveWidgetData('pas_check_in_record', _todayRecord!.toJson()); | ||||
|       await home.saveWidgetData('pas_check_in_record', _todayRecord!.toJson()); | ||||
|     } catch (err) { | ||||
|       if (!mounted) return; | ||||
|       context.showErrorDialog(err); | ||||
| @@ -348,6 +495,8 @@ class _HomeDashCheckInWidgetState extends State<_HomeDashCheckInWidget> { | ||||
|                         Text( | ||||
|                           'dailyCheckInNone', | ||||
|                           style: Theme.of(context).textTheme.bodyLarge, | ||||
|                           maxLines: 2, | ||||
|                           overflow: TextOverflow.ellipsis, | ||||
|                         ).tr(), | ||||
|                       ], | ||||
|                     ) | ||||
| @@ -365,6 +514,11 @@ class _HomeDashCheckInWidgetState extends State<_HomeDashCheckInWidget> { | ||||
|                           '+${_todayRecord!.resultExperience} EXP', | ||||
|                           style: Theme.of(context).textTheme.bodyLarge, | ||||
|                         ), | ||||
|                         if (_todayRecord!.resultCoin >= 0) | ||||
|                           Text( | ||||
|                             '+${_todayRecord!.resultCoin} ${'walletCurrencyShort'.tr()}', | ||||
|                             style: Theme.of(context).textTheme.bodyLarge, | ||||
|                           ) | ||||
|                       ], | ||||
|                     ), | ||||
|             ), | ||||
| @@ -409,7 +563,7 @@ class _HomeDashCheckInWidgetState extends State<_HomeDashCheckInWidget> { | ||||
| } | ||||
|  | ||||
| class _HomeDashNotificationWidget extends StatefulWidget { | ||||
|   const _HomeDashNotificationWidget({super.key}); | ||||
|   const _HomeDashNotificationWidget(); | ||||
|  | ||||
|   @override | ||||
|   State<_HomeDashNotificationWidget> createState() => _HomeDashNotificationWidgetState(); | ||||
| @@ -480,7 +634,7 @@ class _HomeDashNotificationWidgetState extends State<_HomeDashNotificationWidget | ||||
| } | ||||
|  | ||||
| class _HomeDashRecommendationPostWidget extends StatefulWidget { | ||||
|   const _HomeDashRecommendationPostWidget({super.key}); | ||||
|   const _HomeDashRecommendationPostWidget(); | ||||
|  | ||||
|   @override | ||||
|   State<_HomeDashRecommendationPostWidget> createState() => _HomeDashRecommendationPostWidgetState(); | ||||
| @@ -494,9 +648,7 @@ class _HomeDashRecommendationPostWidgetState extends State<_HomeDashRecommendati | ||||
|     setState(() => _isBusy = true); | ||||
|     try { | ||||
|       final pt = context.read<SnPostContentProvider>(); | ||||
|       final home = context.read<HomeWidgetProvider>(); | ||||
|       _posts = await pt.listRecommendations(); | ||||
|       home.saveWidgetData('post_featured', _posts!.first.toJson()); | ||||
|     } catch (err) { | ||||
|       if (!mounted) return; | ||||
|       context.showErrorDialog(err); | ||||
|   | ||||
							
								
								
									
										241
									
								
								lib/screens/news/news_detail.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,241 @@ | ||||
| import 'package:easy_localization/easy_localization.dart'; | ||||
| import 'package:flutter/gestures.dart'; | ||||
| import 'package:flutter/material.dart'; | ||||
| import 'package:gap/gap.dart'; | ||||
| import 'package:html/dom.dart' as dom; | ||||
| import 'package:html/parser.dart'; | ||||
| import 'package:provider/provider.dart'; | ||||
| import 'package:relative_time/relative_time.dart'; | ||||
| import 'package:styled_widget/styled_widget.dart'; | ||||
| import 'package:surface/providers/sn_network.dart'; | ||||
| import 'package:surface/types/news.dart'; | ||||
| import 'package:surface/widgets/dialog.dart'; | ||||
| import 'package:surface/widgets/navigation/app_scaffold.dart'; | ||||
| import 'package:flutter_inappwebview/flutter_inappwebview.dart'; | ||||
| import 'package:surface/widgets/universal_image.dart'; | ||||
| import 'package:url_launcher/url_launcher_string.dart'; | ||||
|  | ||||
| class NewsDetailScreen extends StatefulWidget { | ||||
|   final String hash; | ||||
|  | ||||
|   const NewsDetailScreen({super.key, required this.hash}); | ||||
|  | ||||
|   @override | ||||
|   State<NewsDetailScreen> createState() => _NewsDetailScreenState(); | ||||
| } | ||||
|  | ||||
| class _NewsDetailScreenState extends State<NewsDetailScreen> { | ||||
|   SnNewsArticle? _article; | ||||
|   dom.Document? _articleFragment; | ||||
|  | ||||
|   Future<void> _fetchArticle() async { | ||||
|     try { | ||||
|       final sn = context.read<SnNetworkProvider>(); | ||||
|       final resp = await sn.client.get('/cgi/re/news/${widget.hash}'); | ||||
|       _article = SnNewsArticle.fromJson(resp.data); | ||||
|       _articleFragment = parse(_article!.content); | ||||
|     } catch (err) { | ||||
|       if (!mounted) return; | ||||
|       context.showErrorDialog(err).then((_) { | ||||
|         if (!mounted) return; | ||||
|         Navigator.pop(context); | ||||
|       }); | ||||
|     } finally { | ||||
|       setState(() {}); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   List<Widget> _parseHtmlToWidgets(Iterable<dom.Element>? elements) { | ||||
|     if (elements == null) return []; | ||||
|  | ||||
|     final List<Widget> widgets = []; | ||||
|  | ||||
|     for (final node in elements) { | ||||
|       switch (node.localName) { | ||||
|         case 'h1': | ||||
|         case 'h2': | ||||
|         case 'h3': | ||||
|         case 'h4': | ||||
|         case 'h5': | ||||
|         case 'h6': | ||||
|           widgets.add(Text(node.text.trim(), style: Theme.of(context).textTheme.titleMedium)); | ||||
|           break; | ||||
|         case 'p': | ||||
|           if (node.text.trim().isEmpty) continue; | ||||
|           widgets.add( | ||||
|             Text.rich( | ||||
|               TextSpan( | ||||
|                 text: node.text.trim(), | ||||
|                 children: [ | ||||
|                   for (final child in node.children) | ||||
|                     switch (child.localName) { | ||||
|                       'a' => TextSpan( | ||||
|                           text: child.text.trim(), | ||||
|                           style: const TextStyle(decoration: TextDecoration.underline), | ||||
|                           recognizer: TapGestureRecognizer() | ||||
|                             ..onTap = () { | ||||
|                               launchUrlString(child.attributes['href']!); | ||||
|                             }, | ||||
|                         ), | ||||
|                       _ => TextSpan(text: child.text.trim()), | ||||
|                     }, | ||||
|                 ], | ||||
|               ), | ||||
|               style: Theme.of(context).textTheme.bodyLarge, | ||||
|             ), | ||||
|           ); | ||||
|           break; | ||||
|         case 'a': | ||||
|           // drop single link | ||||
|           break; | ||||
|         case 'div': | ||||
|           // ignore div text, normally it is not meaningful | ||||
|           widgets.addAll(_parseHtmlToWidgets(node.children)); | ||||
|           break; | ||||
|         case 'hr': | ||||
|           widgets.add(const Divider()); | ||||
|           break; | ||||
|         case 'img': | ||||
|           var src = node.attributes['src']; | ||||
|           if (src == null) break; | ||||
|           final width = double.tryParse(node.attributes['width'] ?? 'null'); | ||||
|           final height = double.tryParse(node.attributes['height'] ?? 'null'); | ||||
|           final ratio = width != null && height != null ? width / height : 1.0; | ||||
|           if (src.startsWith('//')) { | ||||
|             src = 'https:$src'; | ||||
|           } else if (!src.startsWith('http')) { | ||||
|             final baseUri = Uri.parse(_article!.url); | ||||
|             final baseUrl = '${baseUri.scheme}://${baseUri.host}'; | ||||
|             src = '$baseUrl/$src'; | ||||
|           } | ||||
|           widgets.add( | ||||
|             AspectRatio( | ||||
|               aspectRatio: ratio, | ||||
|               child: Container( | ||||
|                 decoration: BoxDecoration( | ||||
|                   borderRadius: BorderRadius.all(Radius.circular(8)), | ||||
|                   border: Border.all( | ||||
|                     color: Theme.of(context).dividerColor, | ||||
|                     width: 1, | ||||
|                   ), | ||||
|                 ), | ||||
|                 height: height ?? double.infinity, | ||||
|                 child: ClipRRect( | ||||
|                   borderRadius: BorderRadius.all(Radius.circular(8)), | ||||
|                   child: Container( | ||||
|                     color: Theme.of(context).colorScheme.surfaceContainer, | ||||
|                     child: AutoResizeUniversalImage( | ||||
|                       src, | ||||
|                       fit: width != null && height != null ? BoxFit.cover : BoxFit.contain, | ||||
|                     ), | ||||
|                   ), | ||||
|                 ), | ||||
|               ), | ||||
|             ), | ||||
|           ); | ||||
|           break; | ||||
|         default: | ||||
|           widgets.addAll(_parseHtmlToWidgets(node.children)); | ||||
|           break; | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     return widgets; | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   void initState() { | ||||
|     super.initState(); | ||||
|     _fetchArticle(); | ||||
|   } | ||||
|  | ||||
|   bool _isReadingFromReader = true; | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     return AppScaffold( | ||||
|       appBar: AppBar( | ||||
|         leading: const PageBackButton(), | ||||
|         title: Text(_article?.title ?? 'loading'.tr()), | ||||
|       ), | ||||
|       body: Column( | ||||
|         children: [ | ||||
|           MaterialBanner( | ||||
|             dividerColor: Colors.transparent, | ||||
|             leading: const Icon(Icons.info), | ||||
|             content: Text(_isReadingFromReader ? 'newsReadingFromReader'.tr() : 'newsReadingFromOriginal'.tr()), | ||||
|             actions: [ | ||||
|               TextButton( | ||||
|                 child: Text('newsReadingProviderSwap').tr(), | ||||
|                 onPressed: () { | ||||
|                   setState(() => _isReadingFromReader = !_isReadingFromReader); | ||||
|                 }, | ||||
|               ), | ||||
|             ], | ||||
|           ), | ||||
|           if (_articleFragment != null && _isReadingFromReader) | ||||
|             Expanded( | ||||
|               child: Container( | ||||
|                 constraints: BoxConstraints(maxWidth: 640), | ||||
|                 child: SingleChildScrollView( | ||||
|                   child: Column( | ||||
|                     crossAxisAlignment: CrossAxisAlignment.start, | ||||
|                     spacing: 8, | ||||
|                     children: [ | ||||
|                       Text(_article!.title, style: Theme.of(context).textTheme.titleLarge), | ||||
|                       Builder(builder: (context) { | ||||
|                         final htmlDescription = parse(_article!.description); | ||||
|                         return Text( | ||||
|                           htmlDescription.children.map((ele) => ele.text.trim()).join(), | ||||
|                           style: Theme.of(context).textTheme.bodyMedium, | ||||
|                         ); | ||||
|                       }), | ||||
|                       Builder(builder: (context) { | ||||
|                         final date = _article!.publishedAt ?? _article!.createdAt; | ||||
|                         return Row( | ||||
|                           spacing: 2, | ||||
|                           children: [ | ||||
|                             Text(DateFormat().format(date)).textStyle(Theme.of(context).textTheme.bodySmall!), | ||||
|                             Text(' · ').textStyle(Theme.of(context).textTheme.bodySmall!).bold(), | ||||
|                             Text(RelativeTime(context).format(date)).textStyle(Theme.of(context).textTheme.bodySmall!), | ||||
|                           ], | ||||
|                         ).opacity(0.75); | ||||
|                       }), | ||||
|                       Text('newsDisclaimer').tr().textStyle(Theme.of(context).textTheme.bodySmall!).opacity(0.75), | ||||
|                       const Divider(), | ||||
|                       ..._parseHtmlToWidgets(_articleFragment!.children), | ||||
|                       const Divider(), | ||||
|                       InkWell( | ||||
|                         child: Row( | ||||
|                           mainAxisSize: MainAxisSize.min, | ||||
|                           children: [ | ||||
|                             Text( | ||||
|                               'Reference from original website', | ||||
|                               style: TextStyle(decoration: TextDecoration.underline), | ||||
|                             ), | ||||
|                             const Gap(4), | ||||
|                             Icon(Icons.launch, size: 16), | ||||
|                           ], | ||||
|                         ).opacity(0.85), | ||||
|                         onTap: () { | ||||
|                           launchUrlString(_article!.url); | ||||
|                         }, | ||||
|                       ), | ||||
|                       Gap(MediaQuery.of(context).padding.bottom), | ||||
|                     ], | ||||
|                   ).padding(horizontal: 12, vertical: 16), | ||||
|                 ), | ||||
|               ).center(), | ||||
|             ) | ||||
|           else if (_article != null) | ||||
|             Expanded( | ||||
|               child: InAppWebView( | ||||
|                 key: GlobalKey(), | ||||
|                 initialUrlRequest: URLRequest(url: WebUri(_article!.url)), | ||||
|               ), | ||||
|             ), | ||||
|         ], | ||||
|       ), | ||||
|     ); | ||||
|   } | ||||
| } | ||||
							
								
								
									
										239
									
								
								lib/screens/news/news_list.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,239 @@ | ||||
| import 'package:easy_localization/easy_localization.dart'; | ||||
| import 'package:flutter/material.dart'; | ||||
| import 'package:gap/gap.dart'; | ||||
| import 'package:go_router/go_router.dart'; | ||||
| import 'package:html/parser.dart'; | ||||
| import 'package:provider/provider.dart'; | ||||
| import 'package:relative_time/relative_time.dart'; | ||||
| import 'package:styled_widget/styled_widget.dart'; | ||||
| import 'package:surface/providers/sn_network.dart'; | ||||
| import 'package:surface/types/news.dart'; | ||||
| import 'package:surface/widgets/app_bar_leading.dart'; | ||||
| import 'package:surface/widgets/dialog.dart'; | ||||
| import 'package:surface/widgets/navigation/app_scaffold.dart'; | ||||
| import 'package:surface/widgets/universal_image.dart'; | ||||
| import 'package:very_good_infinite_list/very_good_infinite_list.dart'; | ||||
|  | ||||
| class NewsScreen extends StatefulWidget { | ||||
|   const NewsScreen({super.key}); | ||||
|  | ||||
|   @override | ||||
|   State<NewsScreen> createState() => _NewsScreenState(); | ||||
| } | ||||
|  | ||||
| class _NewsScreenState extends State<NewsScreen> { | ||||
|   List<SnNewsSource>? _sources; | ||||
|  | ||||
|   @override | ||||
|   initState() { | ||||
|     super.initState(); | ||||
|     _fetchSources(); | ||||
|   } | ||||
|  | ||||
|   Future<void> _fetchSources() async { | ||||
|     try { | ||||
|       final sn = context.read<SnNetworkProvider>(); | ||||
|       final resp = await sn.client.get('/cgi/re/well-known/sources'); | ||||
|       _sources = List<SnNewsSource>.from( | ||||
|         resp.data?.map((e) => SnNewsSource.fromJson(e)) ?? [], | ||||
|       ); | ||||
|     } catch (err) { | ||||
|       if (!mounted) return; | ||||
|       context.showErrorDialog(err); | ||||
|     } finally { | ||||
|       setState(() {}); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     if (_sources == null) { | ||||
|       return AppScaffold( | ||||
|         appBar: AppBar( | ||||
|           leading: AutoAppBarLeading(), | ||||
|           title: Text('screenNews').tr(), | ||||
|         ), | ||||
|         body: Center( | ||||
|           child: CircularProgressIndicator(), | ||||
|         ), | ||||
|       ); | ||||
|     } | ||||
|  | ||||
|     return DefaultTabController( | ||||
|       length: _sources!.length + 1, | ||||
|       child: AppScaffold( | ||||
|         body: NestedScrollView( | ||||
|           headerSliverBuilder: (BuildContext context, bool innerBoxIsScrolled) { | ||||
|             return <Widget>[ | ||||
|               SliverOverlapAbsorber( | ||||
|                 handle: NestedScrollView.sliverOverlapAbsorberHandleFor(context), | ||||
|                 sliver: SliverAppBar( | ||||
|                   leading: AutoAppBarLeading(), | ||||
|                   title: Text('screenNews').tr(), | ||||
|                   floating: true, | ||||
|                   snap: true, | ||||
|                   bottom: TabBar( | ||||
|                     isScrollable: true, | ||||
|                     tabs: [ | ||||
|                       Tab(child: Text('newsAllSources'.tr()).textColor(Theme.of(context).appBarTheme.foregroundColor)), | ||||
|                       for (final source in _sources!) | ||||
|                         Tab( | ||||
|                           child: Text(source.label).textColor(Theme.of(context).appBarTheme.foregroundColor), | ||||
|                         ), | ||||
|                     ], | ||||
|                   ), | ||||
|                 ), | ||||
|               ), | ||||
|             ]; | ||||
|           }, | ||||
|           body: TabBarView( | ||||
|             children: [ | ||||
|               _NewsArticleListWidget(allSources: _sources!), | ||||
|               for (final source in _sources!) | ||||
|                 _NewsArticleListWidget( | ||||
|                   source: source.id, | ||||
|                   allSources: _sources!, | ||||
|                 ), | ||||
|             ], | ||||
|           ), | ||||
|         ), | ||||
|       ), | ||||
|     ); | ||||
|   } | ||||
| } | ||||
|  | ||||
| class _NewsArticleListWidget extends StatefulWidget { | ||||
|   final String? source; | ||||
|   final List<SnNewsSource> allSources; | ||||
|  | ||||
|   const _NewsArticleListWidget({this.source, required this.allSources}); | ||||
|  | ||||
|   @override | ||||
|   State<_NewsArticleListWidget> createState() => _NewsArticleListWidgetState(); | ||||
| } | ||||
|  | ||||
| class _NewsArticleListWidgetState extends State<_NewsArticleListWidget> { | ||||
|   bool _isBusy = false; | ||||
|  | ||||
|   int? _totalCount; | ||||
|   final List<SnNewsArticle> _articles = List.empty(growable: true); | ||||
|  | ||||
|   Future<void> _fetchArticles() async { | ||||
|     setState(() => _isBusy = true); | ||||
|  | ||||
|     try { | ||||
|       final sn = context.read<SnNetworkProvider>(); | ||||
|       final resp = await sn.client.get('/cgi/re/news', queryParameters: { | ||||
|         'take': 10, | ||||
|         'offset': _articles.length, | ||||
|         if (widget.source != null) 'source': widget.source, | ||||
|       }); | ||||
|       _totalCount = resp.data['count']; | ||||
|       _articles.addAll(List<SnNewsArticle>.from( | ||||
|         resp.data['data']?.map((e) => SnNewsArticle.fromJson(e)) ?? [], | ||||
|       )); | ||||
|     } catch (err) { | ||||
|       if (!mounted) return; | ||||
|       context.showErrorDialog(err); | ||||
|     } finally { | ||||
|       setState(() => _isBusy = false); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   void initState() { | ||||
|     super.initState(); | ||||
|     _fetchArticles(); | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     return MediaQuery.removePadding( | ||||
|       context: context, | ||||
|       removeTop: true, | ||||
|       child: Center( | ||||
|         child: Container( | ||||
|           constraints: BoxConstraints(maxWidth: 640), | ||||
|           child: RefreshIndicator( | ||||
|             onRefresh: _fetchArticles, | ||||
|             child: InfiniteList( | ||||
|               isLoading: _isBusy, | ||||
|               itemCount: _articles.length, | ||||
|               hasReachedMax: _totalCount != null && _articles.length >= _totalCount!, | ||||
|               onFetchData: () { | ||||
|                 _fetchArticles(); | ||||
|               }, | ||||
|               itemBuilder: (context, index) { | ||||
|                 final article = _articles[index]; | ||||
|  | ||||
|                 final baseUri = Uri.parse(article.url); | ||||
|                 final baseUrl = '${baseUri.scheme}://${baseUri.host}'; | ||||
|  | ||||
|                 final htmlDescription = parse(article.description); | ||||
|                 final date = article.publishedAt ?? article.createdAt; | ||||
|  | ||||
|                 return Card( | ||||
|                   child: InkWell( | ||||
|                     radius: 8, | ||||
|                     onTap: () { | ||||
|                       GoRouter.of(context).pushNamed( | ||||
|                         'newsDetail', | ||||
|                         pathParameters: {'hash': article.hash}, | ||||
|                       ); | ||||
|                     }, | ||||
|                     child: Column( | ||||
|                       crossAxisAlignment: CrossAxisAlignment.start, | ||||
|                       children: [ | ||||
|                         if (article.thumbnail.isNotEmpty && !article.thumbnail.endsWith('.svg')) | ||||
|                           ClipRRect( | ||||
|                             borderRadius: BorderRadius.only( | ||||
|                               topRight: Radius.circular(8), | ||||
|                               topLeft: Radius.circular(8), | ||||
|                             ), | ||||
|                             child: AspectRatio( | ||||
|                               aspectRatio: 16 / 9, | ||||
|                               child: Container( | ||||
|                                 color: Theme.of(context).colorScheme.surfaceContainer, | ||||
|                                 child: AutoResizeUniversalImage( | ||||
|                                   article.thumbnail.startsWith('http') | ||||
|                                       ? article.thumbnail | ||||
|                                       : '$baseUrl/${article.thumbnail}', | ||||
|                                 ), | ||||
|                               ), | ||||
|                             ), | ||||
|                           ), | ||||
|                         const Gap(16), | ||||
|                         Text(article.title).textStyle(Theme.of(context).textTheme.titleLarge!).padding(horizontal: 16), | ||||
|                         const Gap(8), | ||||
|                         Text(htmlDescription.children.map((ele) => ele.text.trim()).join()) | ||||
|                             .textStyle(Theme.of(context).textTheme.bodyMedium!) | ||||
|                             .padding(horizontal: 16), | ||||
|                         const Gap(8), | ||||
|                         Row( | ||||
|                           spacing: 2, | ||||
|                           children: [ | ||||
|                             Text(widget.allSources.where((x) => x.id == article.source).first.label) | ||||
|                                 .textStyle(Theme.of(context).textTheme.bodySmall!), | ||||
|                           ], | ||||
|                         ).opacity(0.75).padding(horizontal: 16), | ||||
|                         Row( | ||||
|                           spacing: 2, | ||||
|                           children: [ | ||||
|                             Text(DateFormat().format(date)).textStyle(Theme.of(context).textTheme.bodySmall!), | ||||
|                             Text(' · ').textStyle(Theme.of(context).textTheme.bodySmall!).bold(), | ||||
|                             Text(RelativeTime(context).format(date)).textStyle(Theme.of(context).textTheme.bodySmall!), | ||||
|                           ], | ||||
|                         ).opacity(0.75).padding(horizontal: 16), | ||||
|                         const Gap(16), | ||||
|                       ], | ||||
|                     ), | ||||
|                   ), | ||||
|                 ); | ||||
|               }, | ||||
|             ), | ||||
|           ), | ||||
|         ), | ||||
|       ), | ||||
|     ); | ||||
|   } | ||||
| } | ||||
| @@ -7,6 +7,7 @@ import 'package:material_symbols_icons/symbols.dart'; | ||||
| import 'package:provider/provider.dart'; | ||||
| import 'package:relative_time/relative_time.dart'; | ||||
| import 'package:styled_widget/styled_widget.dart'; | ||||
| import 'package:surface/providers/notification.dart'; | ||||
| import 'package:surface/providers/sn_network.dart'; | ||||
| import 'package:surface/types/notification.dart'; | ||||
| import 'package:surface/types/post.dart'; | ||||
| @@ -14,12 +15,23 @@ import 'package:surface/widgets/app_bar_leading.dart'; | ||||
| import 'package:surface/widgets/dialog.dart'; | ||||
| import 'package:surface/widgets/loading_indicator.dart'; | ||||
| import 'package:surface/widgets/markdown_content.dart'; | ||||
| import 'package:surface/widgets/navigation/app_scaffold.dart'; | ||||
| import 'package:surface/widgets/post/post_item.dart'; | ||||
| import 'package:very_good_infinite_list/very_good_infinite_list.dart'; | ||||
|  | ||||
| import '../providers/userinfo.dart'; | ||||
| import '../widgets/unauthorized_hint.dart'; | ||||
|  | ||||
| const Map<String, IconData> kNotificationTopicIcons = { | ||||
|   'general': Symbols.notifications, | ||||
|   'passport.security.alert': Symbols.gpp_maybe, | ||||
|   'passport.security.otp': Symbols.password, | ||||
|   'interactive.subscription': Symbols.subscriptions, | ||||
|   'interactive.feedback': Symbols.add_reaction, | ||||
|   'messaging.callStart': Symbols.call_received, | ||||
|   'wallet.transaction.new': Symbols.receipt, | ||||
| }; | ||||
|  | ||||
| class NotificationScreen extends StatefulWidget { | ||||
|   const NotificationScreen({super.key}); | ||||
|  | ||||
| @@ -35,13 +47,6 @@ class _NotificationScreenState extends State<NotificationScreen> { | ||||
|   final List<SnNotification> _notifications = List.empty(growable: true); | ||||
|   int? _totalCount; | ||||
|  | ||||
|   static const Map<String, IconData> kNotificationTopicIcons = { | ||||
|     'passport.security.alert': Symbols.gpp_maybe, | ||||
|     'interactive.subscription': Symbols.subscriptions, | ||||
|     'interactive.feedback': Symbols.add_reaction, | ||||
|     'messaging.callStart': Symbols.call_received, | ||||
|   }; | ||||
|  | ||||
|   Future<void> _fetchNotifications() async { | ||||
|     final ua = context.read<UserProvider>(); | ||||
|     if (!ua.isAuthorized) return; | ||||
| @@ -50,6 +55,7 @@ class _NotificationScreenState extends State<NotificationScreen> { | ||||
|  | ||||
|     try { | ||||
|       final sn = context.read<SnNetworkProvider>(); | ||||
|       final nty = context.read<NotificationProvider>(); | ||||
|       final resp = await sn.client.get('/cgi/id/notifications?take=10'); | ||||
|       _totalCount = resp.data['count']; | ||||
|       _notifications.addAll( | ||||
| @@ -58,6 +64,7 @@ class _NotificationScreenState extends State<NotificationScreen> { | ||||
|                 .cast<SnNotification>() ?? | ||||
|             [], | ||||
|       ); | ||||
|       nty.updateTray(); | ||||
|     } catch (err) { | ||||
|       if (!mounted) return; | ||||
|       context.showErrorDialog(err); | ||||
| @@ -82,24 +89,17 @@ class _NotificationScreenState extends State<NotificationScreen> { | ||||
|     if (!mounted) return; | ||||
|     setState(() => _isSubmitting = true); | ||||
|  | ||||
|     List<int> markList = List.empty(growable: true); | ||||
|     for (final element in _notifications) { | ||||
|       if (element.id <= 0) continue; | ||||
|       if (element.readAt != null) continue; | ||||
|       markList.add(element.id); | ||||
|     } | ||||
|  | ||||
|     try { | ||||
|       final sn = context.read<SnNetworkProvider>(); | ||||
|       await sn.client.put('/cgi/id/notifications/read', data: { | ||||
|         'messages': markList, | ||||
|       }); | ||||
|       final nty = context.read<NotificationProvider>(); | ||||
|       final resp = await sn.client.put('/cgi/id/notifications/read/all'); | ||||
|       _notifications.clear(); | ||||
|       _fetchNotifications(); | ||||
|       nty.clear(); | ||||
|  | ||||
|       if (!mounted) return; | ||||
|       context.showSnackbar( | ||||
|         'notificationMarkAllReadPrompt'.plural(markList.length), | ||||
|         'notificationMarkAllReadPrompt'.plural(resp.data['count']), | ||||
|       ); | ||||
|     } catch (err) { | ||||
|       if (!mounted) return; | ||||
| @@ -146,7 +146,7 @@ class _NotificationScreenState extends State<NotificationScreen> { | ||||
|     final ua = context.read<UserProvider>(); | ||||
|  | ||||
|     if (!ua.isAuthorized) { | ||||
|       return Scaffold( | ||||
|       return AppScaffold( | ||||
|         appBar: AppBar( | ||||
|           leading: AutoAppBarLeading(), | ||||
|           title: Text('screenNotification').tr(), | ||||
| @@ -157,7 +157,7 @@ class _NotificationScreenState extends State<NotificationScreen> { | ||||
|       ); | ||||
|     } | ||||
|  | ||||
|     return Scaffold( | ||||
|     return AppScaffold( | ||||
|       appBar: AppBar( | ||||
|         leading: AutoAppBarLeading(), | ||||
|         title: Text('screenNotification').tr(), | ||||
| @@ -215,10 +215,11 @@ class _NotificationScreenState extends State<NotificationScreen> { | ||||
|                                 style: Theme.of(context).textTheme.titleSmall, | ||||
|                               ), | ||||
|                             if (nty.subtitle != null) const Gap(4), | ||||
|                             MarkdownTextContent( | ||||
|                               content: nty.body, | ||||
|                               isAutoWarp: true, | ||||
|                               isSelectable: true, | ||||
|                             SelectionArea( | ||||
|                               child: MarkdownTextContent( | ||||
|                                 content: nty.body, | ||||
|                                 isAutoWarp: true, | ||||
|                               ), | ||||
|                             ), | ||||
|                             if ([ | ||||
|                                   'interactive.feedback', | ||||
|   | ||||
| @@ -13,6 +13,8 @@ import 'package:surface/providers/userinfo.dart'; | ||||
| import 'package:surface/types/post.dart'; | ||||
| import 'package:surface/widgets/dialog.dart'; | ||||
| import 'package:surface/widgets/loading_indicator.dart'; | ||||
| import 'package:surface/widgets/navigation/app_background.dart'; | ||||
| import 'package:surface/widgets/navigation/app_scaffold.dart'; | ||||
| import 'package:surface/widgets/post/post_comment_list.dart'; | ||||
| import 'package:surface/widgets/post/post_item.dart'; | ||||
| import 'package:surface/widgets/post/post_mini_editor.dart'; | ||||
| @@ -20,12 +22,9 @@ import 'package:surface/widgets/post/post_mini_editor.dart'; | ||||
| class PostDetailScreen extends StatefulWidget { | ||||
|   final String slug; | ||||
|   final SnPost? preload; | ||||
|   final Function? onBack; | ||||
|  | ||||
|   const PostDetailScreen({ | ||||
|     super.key, | ||||
|     required this.slug, | ||||
|     this.preload, | ||||
|   }); | ||||
|   const PostDetailScreen({super.key, required this.slug, this.preload, this.onBack}); | ||||
|  | ||||
|   @override | ||||
|   State<PostDetailScreen> createState() => _PostDetailScreenState(); | ||||
| @@ -67,121 +66,129 @@ class _PostDetailScreenState extends State<PostDetailScreen> { | ||||
|     final ua = context.watch<UserProvider>(); | ||||
|     final devicePixelRatio = MediaQuery.of(context).devicePixelRatio; | ||||
|  | ||||
|     return Scaffold( | ||||
|       appBar: AppBar( | ||||
|         leading: BackButton( | ||||
|           onPressed: () { | ||||
|             if (GoRouter.of(context).canPop()) { | ||||
|               GoRouter.of(context).pop(context); | ||||
|               return; | ||||
|             } | ||||
|             GoRouter.of(context).replaceNamed('explore'); | ||||
|           }, | ||||
|         ), | ||||
|         title: _data?.body['title'] != null | ||||
|             ? RichText( | ||||
|                 textAlign: TextAlign.center, | ||||
|                 text: TextSpan(children: [ | ||||
|                   TextSpan( | ||||
|                     text: _data?.body['title'] ?? 'postNoun'.tr(), | ||||
|                     style: Theme.of(context).textTheme.titleLarge!.copyWith( | ||||
|                           color: Theme.of(context).appBarTheme.foregroundColor!, | ||||
|                         ), | ||||
|                   ), | ||||
|                   const TextSpan(text: '\n'), | ||||
|                   TextSpan( | ||||
|                     text: 'postDetail'.tr(), | ||||
|                     style: Theme.of(context).textTheme.bodySmall!.copyWith( | ||||
|                           color: Theme.of(context).appBarTheme.foregroundColor!, | ||||
|                         ), | ||||
|                   ), | ||||
|                 ]), | ||||
|               ) | ||||
|             : Text('postDetail').tr(), | ||||
|       ), | ||||
|       body: CustomScrollView( | ||||
|         slivers: [ | ||||
|           SliverToBoxAdapter( | ||||
|             child: LoadingIndicator(isActive: _isBusy), | ||||
|     return AppBackground( | ||||
|       isRoot: widget.onBack != null, | ||||
|       child: AppScaffold( | ||||
|         appBar: AppBar( | ||||
|           leading: BackButton( | ||||
|             onPressed: () { | ||||
|               if (widget.onBack != null) { | ||||
|                 widget.onBack!.call(); | ||||
|               } | ||||
|               if (GoRouter.of(context).canPop()) { | ||||
|                 GoRouter.of(context).pop(context); | ||||
|                 return; | ||||
|               } | ||||
|               GoRouter.of(context).replaceNamed('explore'); | ||||
|             }, | ||||
|           ), | ||||
|           if (_data != null) | ||||
|             SliverToBoxAdapter( | ||||
|               child: PostItem( | ||||
|                 data: _data!, | ||||
|                 maxWidth: 640, | ||||
|                 showComments: false, | ||||
|                 showFullPost: true, | ||||
|                 onChanged: (data) { | ||||
|                   setState(() => _data = data); | ||||
|                 }, | ||||
|                 onDeleted: () { | ||||
|                   Navigator.pop(context); | ||||
|                 }, | ||||
|               ), | ||||
|             ), | ||||
|           const SliverToBoxAdapter(child: Divider(height: 1)), | ||||
|           if (_data != null) | ||||
|             SliverToBoxAdapter( | ||||
|               child: Container( | ||||
|                 constraints: const BoxConstraints(maxWidth: 640), | ||||
|                 child: Row( | ||||
|                   crossAxisAlignment: CrossAxisAlignment.center, | ||||
|                   children: [ | ||||
|                     const Icon(Symbols.comment, size: 24), | ||||
|                     const Gap(16), | ||||
|                     Text('postCommentsDetailed') | ||||
|                         .plural(_data!.metric.replyCount) | ||||
|                         .textStyle(Theme.of(context).textTheme.titleLarge!), | ||||
|                   ], | ||||
|                 ).padding(horizontal: 20, vertical: 12).center(), | ||||
|               ), | ||||
|             ), | ||||
|           if (_data != null && ua.isAuthorized) | ||||
|             SliverToBoxAdapter( | ||||
|               child: Container( | ||||
|                 height: 240, | ||||
|                 constraints: const BoxConstraints(maxWidth: 640), | ||||
|                 margin: | ||||
|                     ResponsiveBreakpoints.of(context).largerThan(MOBILE) ? const EdgeInsets.all(8) : EdgeInsets.zero, | ||||
|                 decoration: BoxDecoration( | ||||
|                   borderRadius: ResponsiveBreakpoints.of(context).largerThan(MOBILE) | ||||
|                       ? const BorderRadius.all(Radius.circular(8)) | ||||
|                       : BorderRadius.zero, | ||||
|                   border: ResponsiveBreakpoints.of(context).largerThan(MOBILE) | ||||
|                       ? Border.all( | ||||
|                           color: Theme.of(context).dividerColor, | ||||
|                           width: 1 / devicePixelRatio, | ||||
|                         ) | ||||
|                       : Border.symmetric( | ||||
|                           horizontal: BorderSide( | ||||
|                             color: Theme.of(context).dividerColor, | ||||
|                             width: 1 / devicePixelRatio, | ||||
|           title: _data?.body['title'] != null | ||||
|               ? RichText( | ||||
|                   textAlign: TextAlign.center, | ||||
|                   text: TextSpan(children: [ | ||||
|                     TextSpan( | ||||
|                       text: _data?.body['title'] ?? 'postNoun'.tr(), | ||||
|                       style: Theme.of(context).textTheme.titleLarge!.copyWith( | ||||
|                             color: Theme.of(context).appBarTheme.foregroundColor!, | ||||
|                           ), | ||||
|                         ), | ||||
|                 ), | ||||
|                 child: PostMiniEditor( | ||||
|                   postReplyId: _data!.id, | ||||
|                   onPost: () { | ||||
|                     setState(() { | ||||
|                       _data = _data!.copyWith( | ||||
|                         metric: _data!.metric.copyWith( | ||||
|                           replyCount: _data!.metric.replyCount + 1, | ||||
|                         ), | ||||
|                       ); | ||||
|                     }); | ||||
|                     _childListKey.currentState!.refresh(); | ||||
|                     ), | ||||
|                     const TextSpan(text: '\n'), | ||||
|                     TextSpan( | ||||
|                       text: 'postDetail'.tr(), | ||||
|                       style: Theme.of(context).textTheme.bodySmall!.copyWith( | ||||
|                             color: Theme.of(context).appBarTheme.foregroundColor!, | ||||
|                           ), | ||||
|                     ), | ||||
|                   ]), | ||||
|                   maxLines: 2, | ||||
|                   overflow: TextOverflow.ellipsis, | ||||
|                 ) | ||||
|               : Text('postDetail').tr(), | ||||
|         ), | ||||
|         body: CustomScrollView( | ||||
|           slivers: [ | ||||
|             SliverToBoxAdapter( | ||||
|               child: LoadingIndicator(isActive: _isBusy), | ||||
|             ), | ||||
|             if (_data != null) | ||||
|               SliverToBoxAdapter( | ||||
|                 child: PostItem( | ||||
|                   data: _data!, | ||||
|                   maxWidth: 640, | ||||
|                   showComments: false, | ||||
|                   showFullPost: true, | ||||
|                   onChanged: (data) { | ||||
|                     setState(() => _data = data); | ||||
|                   }, | ||||
|                   onDeleted: () { | ||||
|                     Navigator.pop(context); | ||||
|                   }, | ||||
|                 ), | ||||
|               ).center(), | ||||
|             ), | ||||
|           if (_data != null) | ||||
|             PostCommentSliverList( | ||||
|               key: _childListKey, | ||||
|               parentPostId: _data!.id, | ||||
|               maxWidth: 640, | ||||
|             ), | ||||
|           SliverGap(math.max(MediaQuery.of(context).padding.bottom, 16)), | ||||
|         ], | ||||
|               ), | ||||
|             const SliverToBoxAdapter(child: Divider(height: 1)), | ||||
|             if (_data != null) | ||||
|               SliverToBoxAdapter( | ||||
|                 child: Container( | ||||
|                   constraints: const BoxConstraints(maxWidth: 640), | ||||
|                   child: Row( | ||||
|                     crossAxisAlignment: CrossAxisAlignment.center, | ||||
|                     children: [ | ||||
|                       const Icon(Symbols.comment, size: 24), | ||||
|                       const Gap(16), | ||||
|                       Text('postCommentsDetailed') | ||||
|                           .plural(_data!.metric.replyCount) | ||||
|                           .textStyle(Theme.of(context).textTheme.titleLarge!), | ||||
|                     ], | ||||
|                   ).padding(horizontal: 20, vertical: 12).center(), | ||||
|                 ), | ||||
|               ), | ||||
|             if (_data != null && ua.isAuthorized) | ||||
|               SliverToBoxAdapter( | ||||
|                 child: Container( | ||||
|                   height: 240, | ||||
|                   constraints: const BoxConstraints(maxWidth: 640), | ||||
|                   margin: | ||||
|                       ResponsiveBreakpoints.of(context).largerThan(MOBILE) ? const EdgeInsets.all(8) : EdgeInsets.zero, | ||||
|                   decoration: BoxDecoration( | ||||
|                     borderRadius: ResponsiveBreakpoints.of(context).largerThan(MOBILE) | ||||
|                         ? const BorderRadius.all(Radius.circular(8)) | ||||
|                         : BorderRadius.zero, | ||||
|                     border: ResponsiveBreakpoints.of(context).largerThan(MOBILE) | ||||
|                         ? Border.all( | ||||
|                             color: Theme.of(context).dividerColor, | ||||
|                             width: 1 / devicePixelRatio, | ||||
|                           ) | ||||
|                         : Border.symmetric( | ||||
|                             horizontal: BorderSide( | ||||
|                               color: Theme.of(context).dividerColor, | ||||
|                               width: 1 / devicePixelRatio, | ||||
|                             ), | ||||
|                           ), | ||||
|                   ), | ||||
|                   child: PostMiniEditor( | ||||
|                     postReplyId: _data!.id, | ||||
|                     onPost: () { | ||||
|                       setState(() { | ||||
|                         _data = _data!.copyWith( | ||||
|                           metric: _data!.metric.copyWith( | ||||
|                             replyCount: _data!.metric.replyCount + 1, | ||||
|                           ), | ||||
|                         ); | ||||
|                       }); | ||||
|                       _childListKey.currentState!.refresh(); | ||||
|                     }, | ||||
|                   ), | ||||
|                 ).center(), | ||||
|               ), | ||||
|             if (_data != null) | ||||
|               PostCommentSliverList( | ||||
|                 key: _childListKey, | ||||
|                 parentPostId: _data!.id, | ||||
|                 maxWidth: 640, | ||||
|               ), | ||||
|             SliverGap(math.max(MediaQuery.of(context).padding.bottom, 16)), | ||||
|           ], | ||||
|         ), | ||||
|       ), | ||||
|     ); | ||||
|   } | ||||
|   | ||||
| @@ -6,30 +6,35 @@ import 'package:easy_localization/easy_localization.dart'; | ||||
| import 'package:flutter/foundation.dart'; | ||||
| import 'package:flutter/gestures.dart'; | ||||
| import 'package:flutter/material.dart'; | ||||
| import 'package:flutter/services.dart'; | ||||
| import 'package:gap/gap.dart'; | ||||
| import 'package:go_router/go_router.dart'; | ||||
| import 'package:image_picker/image_picker.dart'; | ||||
| import 'package:hotkey_manager/hotkey_manager.dart'; | ||||
| import 'package:material_symbols_icons/symbols.dart'; | ||||
| import 'package:pasteboard/pasteboard.dart'; | ||||
| import 'package:styled_widget/styled_widget.dart'; | ||||
| import 'package:surface/controllers/post_write_controller.dart'; | ||||
| import 'package:surface/providers/config.dart'; | ||||
| import 'package:surface/providers/sn_network.dart'; | ||||
| import 'package:surface/types/post.dart'; | ||||
| import 'package:surface/widgets/account/account_image.dart'; | ||||
| import 'package:surface/widgets/loading_indicator.dart'; | ||||
| import 'package:surface/widgets/navigation/app_scaffold.dart'; | ||||
| import 'package:surface/widgets/post/post_item.dart'; | ||||
| import 'package:surface/widgets/post/post_media_pending_list.dart'; | ||||
| import 'package:surface/widgets/post/post_meta_editor.dart'; | ||||
| import 'package:surface/widgets/dialog.dart'; | ||||
| import 'package:provider/provider.dart'; | ||||
|  | ||||
| class PostEditorExtraProps { | ||||
| import '../../types/attachment.dart'; | ||||
|  | ||||
| class PostEditorExtra { | ||||
|   final String? text; | ||||
|   final String? title; | ||||
|   final String? description; | ||||
|   final List<PostWriteMedia>? attachments; | ||||
|  | ||||
|   const PostEditorExtraProps({ | ||||
|   const PostEditorExtra({ | ||||
|     this.text, | ||||
|     this.title, | ||||
|     this.description, | ||||
| @@ -42,7 +47,7 @@ class PostEditorScreen extends StatefulWidget { | ||||
|   final int? postEditId; | ||||
|   final int? postReplyId; | ||||
|   final int? postRepostId; | ||||
|   final PostEditorExtraProps? extraProps; | ||||
|   final PostEditorExtra? extraProps; | ||||
|  | ||||
|   const PostEditorScreen({ | ||||
|     super.key, | ||||
| @@ -58,7 +63,9 @@ class PostEditorScreen extends StatefulWidget { | ||||
| } | ||||
|  | ||||
| class _PostEditorScreenState extends State<PostEditorScreen> { | ||||
|   final PostWriteController _writeController = PostWriteController(); | ||||
|   late final PostWriteController _writeController = PostWriteController( | ||||
|     doLoadFromTemporary: widget.postEditId == null, | ||||
|   ); | ||||
|  | ||||
|   bool _isFetching = false; | ||||
|  | ||||
| @@ -71,11 +78,14 @@ class _PostEditorScreenState extends State<PostEditorScreen> { | ||||
|  | ||||
|     try { | ||||
|       final sn = context.read<SnNetworkProvider>(); | ||||
|       final config = context.read<ConfigProvider>(); | ||||
|       final resp = await sn.client.get('/cgi/co/publishers/me'); | ||||
|       _publishers = List<SnPublisher>.from( | ||||
|         resp.data?.map((e) => SnPublisher.fromJson(e)) ?? [], | ||||
|       ); | ||||
|       _writeController.setPublisher(_publishers?.firstOrNull); | ||||
|       final beforeId = config.prefs.getInt('int_last_publisher_id'); | ||||
|       _writeController | ||||
|           .setPublisher(_publishers?.where((ele) => ele.id == beforeId).firstOrNull ?? _publishers?.firstOrNull); | ||||
|     } catch (err) { | ||||
|       if (!mounted) return; | ||||
|       context.showErrorDialog(err); | ||||
| @@ -92,47 +102,39 @@ class _PostEditorScreenState extends State<PostEditorScreen> { | ||||
|     ); | ||||
|   } | ||||
|  | ||||
|   final _imagePicker = ImagePicker(); | ||||
|   final HotKey _pasteHotKey = HotKey( | ||||
|     key: PhysicalKeyboardKey.keyV, | ||||
|     modifiers: [Platform.isMacOS ? HotKeyModifier.meta : HotKeyModifier.control], | ||||
|     scope: HotKeyScope.inapp, | ||||
|   ); | ||||
|  | ||||
|   void _takeMedia(bool isVideo) async { | ||||
|     final result = isVideo | ||||
|         ? await _imagePicker.pickVideo(source: ImageSource.camera) | ||||
|         : await _imagePicker.pickImage(source: ImageSource.camera); | ||||
|     if (result == null) return; | ||||
|     _writeController.addAttachments([ | ||||
|       PostWriteMedia.fromFile(result), | ||||
|     ]); | ||||
|   } | ||||
|  | ||||
|   void _selectMedia() async { | ||||
|     final result = await _imagePicker.pickMultipleMedia(); | ||||
|     if (result.isEmpty) return; | ||||
|     _writeController.addAttachments( | ||||
|       result.map((e) => PostWriteMedia.fromFile(e)), | ||||
|     ); | ||||
|   } | ||||
|  | ||||
|   void _pasteMedia() async { | ||||
|     final imageBytes = await Pasteboard.image; | ||||
|     if (imageBytes == null) return; | ||||
|     _writeController.addAttachments([ | ||||
|       PostWriteMedia.fromBytes( | ||||
|         imageBytes, | ||||
|         'attachmentPastedImage'.tr(), | ||||
|         PostWriteMediaType.image, | ||||
|       ), | ||||
|     ]); | ||||
|   void _registerHotKey() { | ||||
|     if (kIsWeb || Platform.isAndroid || Platform.isIOS) return; | ||||
|     hotKeyManager.register(_pasteHotKey, keyDownHandler: (_) async { | ||||
|       final imageBytes = await Pasteboard.image; | ||||
|       if (imageBytes == null) return; | ||||
|       _writeController.addAttachments([ | ||||
|         PostWriteMedia.fromBytes( | ||||
|           imageBytes, | ||||
|           'attachmentPastedImage'.tr(), | ||||
|           SnMediaType.image, | ||||
|         ), | ||||
|       ]); | ||||
|       setState(() {}); | ||||
|     }); | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   void dispose() { | ||||
|     _writeController.dispose(); | ||||
|     if (!kIsWeb && !(Platform.isAndroid || Platform.isIOS)) hotKeyManager.unregister(_pasteHotKey); | ||||
|     super.dispose(); | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   void initState() { | ||||
|     super.initState(); | ||||
|     _registerHotKey(); | ||||
|     if (!PostWriteController.kTitleMap.keys.contains(widget.mode)) { | ||||
|       context.showErrorDialog('Unknown post type'); | ||||
|       Navigator.pop(context); | ||||
| @@ -159,7 +161,7 @@ class _PostEditorScreenState extends State<PostEditorScreen> { | ||||
|     return ListenableBuilder( | ||||
|       listenable: _writeController, | ||||
|       builder: (context, _) { | ||||
|         return Scaffold( | ||||
|         return AppScaffold( | ||||
|           appBar: AppBar( | ||||
|             leading: BackButton( | ||||
|               onPressed: () { | ||||
| @@ -183,6 +185,7 @@ class _PostEditorScreenState extends State<PostEditorScreen> { | ||||
|                       ), | ||||
|                 ), | ||||
|               ]), | ||||
|               maxLines: 2, | ||||
|             ), | ||||
|             actions: [ | ||||
|               IconButton( | ||||
| @@ -265,6 +268,8 @@ class _PostEditorScreenState extends State<PostEditorScreen> { | ||||
|                       }); | ||||
|                     } else { | ||||
|                       _writeController.setPublisher(value); | ||||
|                       final config = context.read<ConfigProvider>(); | ||||
|                       config.prefs.setInt('int_last_publisher_id', value.id); | ||||
|                     } | ||||
|                   }, | ||||
|                   buttonStyleData: const ButtonStyleData( | ||||
| @@ -278,133 +283,130 @@ class _PostEditorScreenState extends State<PostEditorScreen> { | ||||
|               ), | ||||
|               const Divider(height: 1), | ||||
|               Expanded( | ||||
|                 child: SingleChildScrollView( | ||||
|                   padding: EdgeInsets.only(bottom: 8), | ||||
|                   child: Column( | ||||
|                     children: [ | ||||
|                       // Replying Notice | ||||
|                       if (_writeController.replyingPost != null) | ||||
|                         Column( | ||||
|                           children: [ | ||||
|                             Theme( | ||||
|                               data: Theme.of(context).copyWith( | ||||
|                                 dividerColor: Colors.transparent, | ||||
|                               ), | ||||
|                               child: ExpansionTile( | ||||
|                                 minTileHeight: 48, | ||||
|                                 leading: const Icon(Symbols.reply).padding(left: 4), | ||||
|                                 title: Text('postReplyingNotice') | ||||
|                                     .fontSize(15) | ||||
|                                     .tr(args: ['@${_writeController.replyingPost!.publisher.name}']), | ||||
|                                 children: <Widget>[PostItem(data: _writeController.replyingPost!)], | ||||
|                               ), | ||||
|                 child: Stack( | ||||
|                   children: [ | ||||
|                     SingleChildScrollView( | ||||
|                       padding: EdgeInsets.only(bottom: 160), | ||||
|                       child: Column( | ||||
|                         children: [ | ||||
|                           // Replying Notice | ||||
|                           if (_writeController.replyingPost != null) | ||||
|                             Column( | ||||
|                               children: [ | ||||
|                                 ExpansionTile( | ||||
|                                   minTileHeight: 48, | ||||
|                                   leading: const Icon(Symbols.reply).padding(left: 4), | ||||
|                                   title: Text('postReplyingNotice') | ||||
|                                       .fontSize(15) | ||||
|                                       .tr(args: ['@${_writeController.replyingPost!.publisher.name}']), | ||||
|                                   children: <Widget>[PostItem(data: _writeController.replyingPost!)], | ||||
|                                 ), | ||||
|                                 const Divider(height: 1), | ||||
|                               ], | ||||
|                             ), | ||||
|                             const Divider(height: 1), | ||||
|                           ], | ||||
|                         ), | ||||
|                       // Reposting Notice | ||||
|                       if (_writeController.repostingPost != null) | ||||
|                         Column( | ||||
|                           children: [ | ||||
|                             Theme( | ||||
|                               data: Theme.of(context).copyWith( | ||||
|                                 dividerColor: Colors.transparent, | ||||
|                               ), | ||||
|                               child: ExpansionTile( | ||||
|                                 minTileHeight: 48, | ||||
|                                 leading: const Icon(Symbols.forward).padding(left: 4), | ||||
|                                 title: Text('postRepostingNotice') | ||||
|                                     .fontSize(15) | ||||
|                                     .tr(args: ['@${_writeController.repostingPost!.publisher.name}']), | ||||
|                                 children: <Widget>[ | ||||
|                                   PostItem( | ||||
|                                     data: _writeController.repostingPost!, | ||||
|                                   ) | ||||
|                                 ], | ||||
|                               ), | ||||
|                           // Reposting Notice | ||||
|                           if (_writeController.repostingPost != null) | ||||
|                             Column( | ||||
|                               children: [ | ||||
|                                 ExpansionTile( | ||||
|                                   minTileHeight: 48, | ||||
|                                   leading: const Icon(Symbols.forward).padding(left: 4), | ||||
|                                   title: Text('postRepostingNotice') | ||||
|                                       .fontSize(15) | ||||
|                                       .tr(args: ['@${_writeController.repostingPost!.publisher.name}']), | ||||
|                                   children: <Widget>[ | ||||
|                                     PostItem( | ||||
|                                       data: _writeController.repostingPost!, | ||||
|                                     ) | ||||
|                                   ], | ||||
|                                 ), | ||||
|                                 const Divider(height: 1), | ||||
|                               ], | ||||
|                             ), | ||||
|                             const Divider(height: 1), | ||||
|                           ], | ||||
|                         ), | ||||
|                       // Editing Notice | ||||
|                       if (_writeController.editingPost != null) | ||||
|                         Column( | ||||
|                           children: [ | ||||
|                             Theme( | ||||
|                               data: Theme.of(context).copyWith( | ||||
|                                 dividerColor: Colors.transparent, | ||||
|                               ), | ||||
|                               child: ExpansionTile( | ||||
|                                 minTileHeight: 48, | ||||
|                                 leading: const Icon(Symbols.edit_note).padding(left: 4), | ||||
|                                 title: Text('postEditingNotice') | ||||
|                                     .fontSize(15) | ||||
|                                     .tr(args: ['@${_writeController.editingPost!.publisher.name}']), | ||||
|                                 children: <Widget>[PostItem(data: _writeController.editingPost!)], | ||||
|                               ), | ||||
|                           // Editing Notice | ||||
|                           if (_writeController.editingPost != null) | ||||
|                             Column( | ||||
|                               children: [ | ||||
|                                 ExpansionTile( | ||||
|                                   minTileHeight: 48, | ||||
|                                   leading: const Icon(Symbols.edit_note).padding(left: 4), | ||||
|                                   title: Text('postEditingNotice') | ||||
|                                       .fontSize(15) | ||||
|                                       .tr(args: ['@${_writeController.editingPost!.publisher.name}']), | ||||
|                                   children: <Widget>[PostItem(data: _writeController.editingPost!)], | ||||
|                                 ), | ||||
|                                 const Divider(height: 1), | ||||
|                               ], | ||||
|                             ), | ||||
|                           // Content Input Area | ||||
|                           Container( | ||||
|                             constraints: const BoxConstraints(maxWidth: 640), | ||||
|                             child: TextField( | ||||
|                               controller: _writeController.contentController, | ||||
|                               maxLines: null, | ||||
|                               decoration: InputDecoration( | ||||
|                                 hintText: 'fieldPostContent'.tr(), | ||||
|                                 hintStyle: TextStyle(fontSize: 14), | ||||
|                                 isCollapsed: true, | ||||
|                                 contentPadding: const EdgeInsets.symmetric( | ||||
|                                   horizontal: 16, | ||||
|                                 ), | ||||
|                                 border: InputBorder.none, | ||||
|                               ), | ||||
|                               onTapOutside: (_) => FocusManager.instance.primaryFocus?.unfocus(), | ||||
|                             ), | ||||
|                             const Divider(height: 1), | ||||
|                           ], | ||||
|                         ), | ||||
|                       // Content Input Area | ||||
|                       TextField( | ||||
|                         controller: _writeController.contentController, | ||||
|                         maxLines: null, | ||||
|                         decoration: InputDecoration( | ||||
|                           hintText: 'fieldPostContent'.tr(), | ||||
|                           hintStyle: TextStyle(fontSize: 14), | ||||
|                           isCollapsed: true, | ||||
|                           contentPadding: const EdgeInsets.symmetric( | ||||
|                             horizontal: 16, | ||||
|                           ), | ||||
|                           border: InputBorder.none, | ||||
|                         ), | ||||
|                         onTapOutside: (_) => FocusManager.instance.primaryFocus?.unfocus(), | ||||
|                         ] | ||||
|                             .expandIndexed( | ||||
|                               (idx, ele) => [ | ||||
|                                 if (idx != 0 || _writeController.isRelatedNull) const Gap(8), | ||||
|                                 ele, | ||||
|                               ], | ||||
|                             ) | ||||
|                             .toList(), | ||||
|                       ), | ||||
|                     ] | ||||
|                         .expandIndexed( | ||||
|                           (idx, ele) => [ | ||||
|                             if (idx != 0 || _writeController.isRelatedNull) const Gap(8), | ||||
|                             ele, | ||||
|                           ], | ||||
|                         ) | ||||
|                         .toList(), | ||||
|                   ), | ||||
|                     ), | ||||
|                     if (_writeController.attachments.isNotEmpty || _writeController.thumbnail != null) | ||||
|                       Positioned( | ||||
|                         bottom: 0, | ||||
|                         left: 0, | ||||
|                         right: 0, | ||||
|                         child: PostMediaPendingList( | ||||
|                           thumbnail: _writeController.thumbnail, | ||||
|                           attachments: _writeController.attachments, | ||||
|                           isBusy: _writeController.isBusy, | ||||
|                           onUpload: (int idx) async { | ||||
|                             await _writeController.uploadSingleAttachment(context, idx); | ||||
|                           }, | ||||
|                           onPostSetThumbnail: (int? idx) { | ||||
|                             _writeController.setThumbnail(idx); | ||||
|                           }, | ||||
|                           onInsertLink: (int idx) async { | ||||
|                             _writeController.contentController.text += | ||||
|                                 '\n'; | ||||
|                           }, | ||||
|                           onUpdate: (int idx, PostWriteMedia updatedMedia) async { | ||||
|                             _writeController.setIsBusy(true); | ||||
|                             try { | ||||
|                               _writeController.setAttachmentAt(idx, updatedMedia); | ||||
|                             } finally { | ||||
|                               _writeController.setIsBusy(false); | ||||
|                             } | ||||
|                           }, | ||||
|                           onRemove: (int idx) async { | ||||
|                             _writeController.setIsBusy(true); | ||||
|                             try { | ||||
|                               _writeController.removeAttachmentAt(idx); | ||||
|                             } finally { | ||||
|                               _writeController.setIsBusy(false); | ||||
|                             } | ||||
|                           }, | ||||
|                           onUpdateBusy: (state) => _writeController.setIsBusy(state), | ||||
|                         ).padding(bottom: 8), | ||||
|                       ), | ||||
|                   ], | ||||
|                 ), | ||||
|               ), | ||||
|               if (_writeController.attachments.isNotEmpty || _writeController.thumbnail != null) | ||||
|                 PostMediaPendingList( | ||||
|                   thumbnail: _writeController.thumbnail, | ||||
|                   attachments: _writeController.attachments, | ||||
|                   isBusy: _writeController.isBusy, | ||||
|                   onUpload: (int idx) async { | ||||
|                     await _writeController.uploadSingleAttachment(context, idx); | ||||
|                   }, | ||||
|                   onPostSetThumbnail: (int? idx) { | ||||
|                     _writeController.setThumbnail(idx); | ||||
|                   }, | ||||
|                   onInsertLink: (int idx) async { | ||||
|                     _writeController.contentController.text += | ||||
|                         '\n'; | ||||
|                   }, | ||||
|                   onUpdate: (int idx, PostWriteMedia updatedMedia) async { | ||||
|                     _writeController.setIsBusy(true); | ||||
|                     try { | ||||
|                       _writeController.setAttachmentAt(idx, updatedMedia); | ||||
|                     } finally { | ||||
|                       _writeController.setIsBusy(false); | ||||
|                     } | ||||
|                   }, | ||||
|                   onRemove: (int idx) async { | ||||
|                     _writeController.setIsBusy(true); | ||||
|                     try { | ||||
|                       _writeController.removeAttachmentAt(idx); | ||||
|                     } finally { | ||||
|                       _writeController.setIsBusy(false); | ||||
|                     } | ||||
|                   }, | ||||
|                   onUpdateBusy: (state) => _writeController.setIsBusy(state), | ||||
|                 ).padding(bottom: 8), | ||||
|               Material( | ||||
|                 elevation: 2, | ||||
|                 child: Column( | ||||
| @@ -419,6 +421,36 @@ class _PostEditorScreenState extends State<PostEditorScreen> { | ||||
|                       ) | ||||
|                     else if (_writeController.isBusy) | ||||
|                       const LinearProgressIndicator(value: null, minHeight: 2), | ||||
|                     Container( | ||||
|                       child: _writeController.temporaryRestored | ||||
|                           ? Container( | ||||
|                               padding: const EdgeInsets.only(top: 4, bottom: 4, left: 28, right: 22), | ||||
|                               decoration: BoxDecoration( | ||||
|                                 border: Border( | ||||
|                                   bottom: BorderSide( | ||||
|                                     color: Theme.of(context).dividerColor, | ||||
|                                     width: 1 / MediaQuery.of(context).devicePixelRatio, | ||||
|                                   ), | ||||
|                                 ), | ||||
|                               ), | ||||
|                               child: Row( | ||||
|                                 crossAxisAlignment: CrossAxisAlignment.center, | ||||
|                                 children: [ | ||||
|                                   const Icon(Icons.restore, size: 20), | ||||
|                                   const Gap(8), | ||||
|                                   Expanded(child: Text('postLocalDraftRestored').tr()), | ||||
|                                   InkWell( | ||||
|                                     child: Text('dialogDismiss').tr(), | ||||
|                                     onTap: () { | ||||
|                                       _writeController.reset(); | ||||
|                                     }, | ||||
|                                   ), | ||||
|                                 ], | ||||
|                               )) | ||||
|                           : const SizedBox.shrink(), | ||||
|                     ) | ||||
|                         .height(_writeController.temporaryRestored ? 32 : 0, animate: true) | ||||
|                         .animate(const Duration(milliseconds: 300), Curves.fastLinearToSlowEaseIn), | ||||
|                     Row( | ||||
|                       mainAxisAlignment: MainAxisAlignment.spaceBetween, | ||||
|                       children: [ | ||||
| @@ -429,63 +461,12 @@ class _PostEditorScreenState extends State<PostEditorScreen> { | ||||
|                               scrollDirection: Axis.vertical, | ||||
|                               child: Row( | ||||
|                                 children: [ | ||||
|                                   PopupMenuButton( | ||||
|                                     icon: Icon( | ||||
|                                       Symbols.add_photo_alternate, | ||||
|                                       color: Theme.of(context).colorScheme.primary, | ||||
|                                     ), | ||||
|                                     itemBuilder: (context) => [ | ||||
|                                       if (!kIsWeb && !Platform.isLinux && !Platform.isMacOS && !Platform.isWindows) | ||||
|                                         PopupMenuItem( | ||||
|                                           child: Row( | ||||
|                                             children: [ | ||||
|                                               const Icon(Symbols.photo_camera), | ||||
|                                               const Gap(16), | ||||
|                                               Text('addAttachmentFromCameraPhoto').tr(), | ||||
|                                             ], | ||||
|                                           ), | ||||
|                                           onTap: () { | ||||
|                                             _takeMedia(false); | ||||
|                                           }, | ||||
|                                         ), | ||||
|                                       if (!kIsWeb && !Platform.isLinux && !Platform.isMacOS && !Platform.isWindows) | ||||
|                                         PopupMenuItem( | ||||
|                                           child: Row( | ||||
|                                             children: [ | ||||
|                                               const Icon(Symbols.videocam), | ||||
|                                               const Gap(16), | ||||
|                                               Text('addAttachmentFromCameraVideo').tr(), | ||||
|                                             ], | ||||
|                                           ), | ||||
|                                           onTap: () { | ||||
|                                             _takeMedia(true); | ||||
|                                           }, | ||||
|                                         ), | ||||
|                                       PopupMenuItem( | ||||
|                                         child: Row( | ||||
|                                           children: [ | ||||
|                                             const Icon(Symbols.photo_library), | ||||
|                                             const Gap(16), | ||||
|                                             Text('addAttachmentFromAlbum').tr(), | ||||
|                                           ], | ||||
|                                         ), | ||||
|                                         onTap: () { | ||||
|                                           _selectMedia(); | ||||
|                                         }, | ||||
|                                       ), | ||||
|                                       PopupMenuItem( | ||||
|                                         child: Row( | ||||
|                                           children: [ | ||||
|                                             const Icon(Symbols.content_paste), | ||||
|                                             const Gap(16), | ||||
|                                             Text('addAttachmentFromClipboard').tr(), | ||||
|                                           ], | ||||
|                                         ), | ||||
|                                         onTap: () { | ||||
|                                           _pasteMedia(); | ||||
|                                         }, | ||||
|                                       ), | ||||
|                                     ], | ||||
|                                   AddPostMediaButton( | ||||
|                                     onAdd: (items) { | ||||
|                                       setState(() { | ||||
|                                         _writeController.addAttachments(items); | ||||
|                                       }); | ||||
|                                     }, | ||||
|                                   ), | ||||
|                                 ], | ||||
|                               ), | ||||
| @@ -496,7 +477,7 @@ class _PostEditorScreenState extends State<PostEditorScreen> { | ||||
|                           onPressed: (_writeController.isBusy || _writeController.publisher == null) | ||||
|                               ? null | ||||
|                               : () { | ||||
|                                   _writeController.post(context).then((_) { | ||||
|                                   _writeController.sendPost(context).then((_) { | ||||
|                                     if (!context.mounted) return; | ||||
|                                     Navigator.pop(context, true); | ||||
|                                   }); | ||||
|   | ||||
| @@ -8,12 +8,16 @@ import 'package:styled_widget/styled_widget.dart'; | ||||
| import 'package:surface/providers/post.dart'; | ||||
| import 'package:surface/types/post.dart'; | ||||
| import 'package:surface/widgets/dialog.dart'; | ||||
| import 'package:surface/widgets/navigation/app_scaffold.dart'; | ||||
| import 'package:surface/widgets/post/post_item.dart'; | ||||
| import 'package:surface/widgets/post/post_tags_field.dart'; | ||||
| import 'package:very_good_infinite_list/very_good_infinite_list.dart'; | ||||
|  | ||||
| class PostSearchScreen extends StatefulWidget { | ||||
|   const PostSearchScreen({super.key}); | ||||
|   final Iterable<String>? initialTags; | ||||
|   final Iterable<String>? initialCategories; | ||||
|  | ||||
|   const PostSearchScreen({super.key, this.initialTags, this.initialCategories}); | ||||
|  | ||||
|   @override | ||||
|   State<PostSearchScreen> createState() => _PostSearchScreenState(); | ||||
| @@ -23,6 +27,7 @@ class _PostSearchScreenState extends State<PostSearchScreen> { | ||||
|   bool _isBusy = false; | ||||
|  | ||||
|   List<String> _searchTags = List.empty(growable: true); | ||||
|   List<String> _searchCategories = List.empty(growable: true); | ||||
|  | ||||
|   final List<SnPost> _posts = List.empty(growable: true); | ||||
|   int? _postCount; | ||||
| @@ -30,8 +35,18 @@ class _PostSearchScreenState extends State<PostSearchScreen> { | ||||
|   String _searchTerm = ''; | ||||
|   Duration? _lastTook; | ||||
|  | ||||
|   @override | ||||
|   void initState() { | ||||
|     super.initState(); | ||||
|     _searchTags.addAll(widget.initialTags ?? []); | ||||
|     _searchCategories.addAll(widget.initialCategories ?? []); | ||||
|     if (_searchTags.isNotEmpty || _searchCategories.isNotEmpty) { | ||||
|       _fetchPosts(); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   Future<void> _fetchPosts() async { | ||||
|     if (_searchTerm.isEmpty && _searchTags.isEmpty) return; | ||||
|     if (_searchTerm.isEmpty && _searchCategories.isEmpty && _searchTags.isEmpty) return; | ||||
|     if (_postCount != null && _posts.length >= _postCount!) return; | ||||
|  | ||||
|     setState(() => _isBusy = true); | ||||
| @@ -45,6 +60,7 @@ class _PostSearchScreenState extends State<PostSearchScreen> { | ||||
|         take: 10, | ||||
|         offset: _posts.length, | ||||
|         tags: _searchTags, | ||||
|         categories: _searchCategories, | ||||
|       ); | ||||
|       final List<SnPost> out = result.$1; | ||||
|       _postCount = result.$2; | ||||
| @@ -73,9 +89,25 @@ class _PostSearchScreenState extends State<PostSearchScreen> { | ||||
|               setState(() => _searchTags = value); | ||||
|             }, | ||||
|           ), | ||||
|           const Gap(4), | ||||
|           PostCategoriesField( | ||||
|             labelText: 'fieldPostCategories'.tr(), | ||||
|             initialCategories: _searchCategories, | ||||
|             onUpdate: (value) { | ||||
|               setState(() => _searchCategories = value); | ||||
|             }, | ||||
|           ), | ||||
|         ], | ||||
|       ).padding(horizontal: 24, vertical: 16), | ||||
|     ); | ||||
|     ).then((_) { | ||||
|       _refreshPosts(); | ||||
|     }); | ||||
|   } | ||||
|  | ||||
|   Future<void> _refreshPosts() { | ||||
|     _postCount = null; | ||||
|     _posts.clear(); | ||||
|     return _fetchPosts(); | ||||
|   } | ||||
|  | ||||
|   @override | ||||
| @@ -88,7 +120,7 @@ class _PostSearchScreenState extends State<PostSearchScreen> { | ||||
|       ), | ||||
|     ]; | ||||
|  | ||||
|     return Scaffold( | ||||
|     return AppScaffold( | ||||
|       appBar: AppBar( | ||||
|         title: Text('screenPostSearch').tr(), | ||||
|         actions: [ | ||||
| @@ -118,8 +150,7 @@ class _PostSearchScreenState extends State<PostSearchScreen> { | ||||
|                     setState(() => _posts[idx] = data); | ||||
|                   }, | ||||
|                   onDeleted: () { | ||||
|                     _posts.clear(); | ||||
|                     _fetchPosts(); | ||||
|                     _refreshPosts(); | ||||
|                   }, | ||||
|                 ), | ||||
|                 onTap: () { | ||||
| @@ -150,10 +181,8 @@ class _PostSearchScreenState extends State<PostSearchScreen> { | ||||
|                     _searchTerm = value; | ||||
|                   }, | ||||
|                   onSubmitted: (value) { | ||||
|                     setState(() => _posts.clear()); | ||||
|  | ||||
|                     _searchTerm = value; | ||||
|                     _fetchPosts(); | ||||
|                     _refreshPosts(); | ||||
|                   }, | ||||
|                 ), | ||||
|                 if (_lastTook != null) | ||||
|   | ||||
| @@ -17,6 +17,7 @@ import 'package:surface/types/post.dart'; | ||||
| import 'package:surface/types/realm.dart'; | ||||
| import 'package:surface/widgets/account/account_image.dart'; | ||||
| import 'package:surface/widgets/dialog.dart'; | ||||
| import 'package:surface/widgets/navigation/app_scaffold.dart'; | ||||
| import 'package:surface/widgets/post/post_item.dart'; | ||||
| import 'package:surface/widgets/universal_image.dart'; | ||||
| import 'package:very_good_infinite_list/very_good_infinite_list.dart'; | ||||
| @@ -45,17 +46,9 @@ class _PostPublisherScreenState extends State<PostPublisherScreen> with SingleTi | ||||
|   Future<void> _fetchPublisher() async { | ||||
|     try { | ||||
|       final sn = context.read<SnNetworkProvider>(); | ||||
|       final ud = context.read<UserDirectoryProvider>(); | ||||
|       final rel = context.read<SnRelationshipProvider>(); | ||||
|       final resp = await sn.client.get('/cgi/co/publishers/${widget.name}'); | ||||
|       if (!mounted) return; | ||||
|       _publisher = SnPublisher.fromJson(resp.data); | ||||
|       _account = await ud.getAccount(_publisher?.accountId); | ||||
|       _accountRelationship = await rel.getRelationship(_account!.id); | ||||
|       if (_publisher?.realmId != null && _publisher!.realmId != 0) { | ||||
|         final resp = await sn.client.get('/cgi/id/realms/${_publisher!.realmId}'); | ||||
|         _realm = SnRealm.fromJson(resp.data); | ||||
|       } | ||||
|     } catch (err) { | ||||
|       if (!mounted) return; | ||||
|       context.showErrorDialog(err).then((_) { | ||||
| @@ -65,6 +58,20 @@ class _PostPublisherScreenState extends State<PostPublisherScreen> with SingleTi | ||||
|     } finally { | ||||
|       setState(() {}); | ||||
|     } | ||||
|  | ||||
|     try { | ||||
|       final sn = context.read<SnNetworkProvider>(); | ||||
|       final ud = context.read<UserDirectoryProvider>(); | ||||
|       final rel = context.read<SnRelationshipProvider>(); | ||||
|       _account = await ud.getAccount(_publisher?.accountId); | ||||
|       _accountRelationship = await rel.getRelationship(_account!.id); | ||||
|       if (_publisher?.realmId != null && _publisher!.realmId != 0) { | ||||
|         final resp = await sn.client.get('/cgi/id/realms/${_publisher!.realmId}'); | ||||
|         _realm = SnRealm.fromJson(resp.data); | ||||
|       } | ||||
|     } catch (_) { | ||||
|       // ignore | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   bool _isSubscribing = false; | ||||
| @@ -268,7 +275,7 @@ class _PostPublisherScreenState extends State<PostPublisherScreen> with SingleTi | ||||
|  | ||||
|     final sn = context.read<SnNetworkProvider>(); | ||||
|  | ||||
|     return Scaffold( | ||||
|     return AppScaffold( | ||||
|       body: NestedScrollView( | ||||
|         controller: _scrollController, | ||||
|         headerSliverBuilder: (BuildContext context, bool innerBoxIsScrolled) { | ||||
| @@ -277,70 +284,77 @@ class _PostPublisherScreenState extends State<PostPublisherScreen> with SingleTi | ||||
|               handle: NestedScrollView.sliverOverlapAbsorberHandleFor(context), | ||||
|               sliver: MultiSliver( | ||||
|                 children: [ | ||||
|                   SliverAppBar( | ||||
|                     expandedHeight: _appBarHeight, | ||||
|                     title: _publisher == null | ||||
|                         ? Text('loading').tr() | ||||
|                         : RichText( | ||||
|                             textAlign: TextAlign.center, | ||||
|                             text: TextSpan(children: [ | ||||
|                               TextSpan( | ||||
|                                 text: _publisher!.nick, | ||||
|                                 style: Theme.of(context).textTheme.titleLarge!.copyWith( | ||||
|                                       color: Theme.of(context).appBarTheme.foregroundColor!, | ||||
|                                       shadows: labelShadows, | ||||
|                                     ), | ||||
|                               ), | ||||
|                               const TextSpan(text: '\n'), | ||||
|                               TextSpan( | ||||
|                                 text: '@${_publisher!.name}', | ||||
|                                 style: Theme.of(context).textTheme.bodySmall!.copyWith( | ||||
|                                       color: Colors.white, | ||||
|                                       shadows: labelShadows, | ||||
|                                     ), | ||||
|                               ), | ||||
|                             ]), | ||||
|                           ), | ||||
|                     pinned: true, | ||||
|                     flexibleSpace: _publisher != null | ||||
|                         ? Stack( | ||||
|                             fit: StackFit.expand, | ||||
|                             children: [ | ||||
|                               if (_publisher!.banner.isNotEmpty) | ||||
|                                 UniversalImage( | ||||
|                                   sn.getAttachmentUrl(_publisher!.banner), | ||||
|                                   fit: BoxFit.cover, | ||||
|                                   height: imageHeight, | ||||
|                                   width: _appBarWidth, | ||||
|                                   cacheHeight: imageHeight, | ||||
|                                   cacheWidth: _appBarWidth, | ||||
|                                 ) | ||||
|                               else | ||||
|                                 Container( | ||||
|                                   color: Theme.of(context).colorScheme.surfaceContainer, | ||||
|                   Theme( | ||||
|                     data: Theme.of(context).copyWith( | ||||
|                       appBarTheme: Theme.of(context).appBarTheme.copyWith( | ||||
|                         foregroundColor: Colors.white, | ||||
|                       ), | ||||
|                     ), | ||||
|                     child: SliverAppBar( | ||||
|                       expandedHeight: _appBarHeight, | ||||
|                       title: _publisher == null | ||||
|                           ? Text('loading').tr() | ||||
|                           : RichText( | ||||
|                               textAlign: TextAlign.center, | ||||
|                               text: TextSpan(children: [ | ||||
|                                 TextSpan( | ||||
|                                   text: _publisher!.nick, | ||||
|                                   style: Theme.of(context).textTheme.titleLarge!.copyWith( | ||||
|                                         color: Colors.white, | ||||
|                                         shadows: labelShadows, | ||||
|                                       ), | ||||
|                                 ), | ||||
|                               Positioned( | ||||
|                                 top: 0, | ||||
|                                 left: 0, | ||||
|                                 right: 0, | ||||
|                                 height: 56 + MediaQuery.of(context).padding.top, | ||||
|                                 child: ClipRect( | ||||
|                                   child: BackdropFilter( | ||||
|                                     filter: ImageFilter.blur( | ||||
|                                       sigmaX: _appBarBlur, | ||||
|                                       sigmaY: _appBarBlur, | ||||
|                                     ), | ||||
|                                     child: Container( | ||||
|                                       color: Colors.black.withOpacity( | ||||
|                                         clampDouble(_appBarBlur * 0.1, 0, 0.5), | ||||
|                                 const TextSpan(text: '\n'), | ||||
|                                 TextSpan( | ||||
|                                   text: '@${_publisher!.name}', | ||||
|                                   style: Theme.of(context).textTheme.bodySmall!.copyWith( | ||||
|                                         color: Colors.white, | ||||
|                                         shadows: labelShadows, | ||||
|                                       ), | ||||
|                                 ), | ||||
|                               ]), | ||||
|                             ), | ||||
|                       pinned: true, | ||||
|                       flexibleSpace: _publisher != null | ||||
|                           ? Stack( | ||||
|                               fit: StackFit.expand, | ||||
|                               children: [ | ||||
|                                 if (_publisher!.banner.isNotEmpty) | ||||
|                                   UniversalImage( | ||||
|                                     sn.getAttachmentUrl(_publisher!.banner), | ||||
|                                     fit: BoxFit.cover, | ||||
|                                     height: imageHeight, | ||||
|                                     width: _appBarWidth, | ||||
|                                     cacheHeight: imageHeight, | ||||
|                                     cacheWidth: _appBarWidth, | ||||
|                                   ) | ||||
|                                 else | ||||
|                                   Container( | ||||
|                                     color: Theme.of(context).colorScheme.surfaceContainer, | ||||
|                                   ), | ||||
|                                 Positioned( | ||||
|                                   top: 0, | ||||
|                                   left: 0, | ||||
|                                   right: 0, | ||||
|                                   height: 56 + MediaQuery.of(context).padding.top, | ||||
|                                   child: ClipRect( | ||||
|                                     child: BackdropFilter( | ||||
|                                       filter: ImageFilter.blur( | ||||
|                                         sigmaX: _appBarBlur, | ||||
|                                         sigmaY: _appBarBlur, | ||||
|                                       ), | ||||
|                                       child: Container( | ||||
|                                         color: Colors.black.withOpacity( | ||||
|                                           clampDouble(_appBarBlur * 0.1, 0, 0.5), | ||||
|                                         ), | ||||
|                                       ), | ||||
|                                     ), | ||||
|                                   ), | ||||
|                                 ), | ||||
|                               ), | ||||
|                             ], | ||||
|                           ) | ||||
|                         : null, | ||||
|                               ], | ||||
|                             ) | ||||
|                           : null, | ||||
|                     ), | ||||
|                   ), | ||||
|                   if (_publisher != null) | ||||
|                     SliverToBoxAdapter( | ||||
| @@ -567,7 +581,6 @@ class _PublisherPostList extends StatelessWidget { | ||||
|   final void Function() onDeleted; | ||||
|  | ||||
|   const _PublisherPostList({ | ||||
|     super.key, | ||||
|     required this.isBusy, | ||||
|     required this.postCount, | ||||
|     required this.posts, | ||||
|   | ||||
| @@ -12,6 +12,7 @@ import 'package:surface/widgets/account/account_image.dart'; | ||||
| import 'package:surface/widgets/app_bar_leading.dart'; | ||||
| import 'package:surface/widgets/dialog.dart'; | ||||
| import 'package:surface/widgets/loading_indicator.dart'; | ||||
| import 'package:surface/widgets/navigation/app_scaffold.dart'; | ||||
| import 'package:surface/widgets/unauthorized_hint.dart'; | ||||
| import 'package:surface/widgets/universal_image.dart'; | ||||
|  | ||||
| @@ -83,7 +84,7 @@ class _RealmScreenState extends State<RealmScreen> { | ||||
|     final ua = context.read<UserProvider>(); | ||||
|  | ||||
|     if (!ua.isAuthorized) { | ||||
|       return Scaffold( | ||||
|       return AppScaffold( | ||||
|         appBar: AppBar( | ||||
|           leading: AutoAppBarLeading(), | ||||
|           title: Text('screenRealm').tr(), | ||||
| @@ -94,7 +95,7 @@ class _RealmScreenState extends State<RealmScreen> { | ||||
|       ); | ||||
|     } | ||||
|  | ||||
|     return Scaffold( | ||||
|     return AppScaffold( | ||||
|       appBar: AppBar( | ||||
|         leading: AutoAppBarLeading(), | ||||
|         title: Text('screenRealm').tr(), | ||||
| @@ -118,113 +119,61 @@ class _RealmScreenState extends State<RealmScreen> { | ||||
|         children: [ | ||||
|           LoadingIndicator(isActive: _isBusy), | ||||
|           Expanded( | ||||
|             child: RefreshIndicator( | ||||
|               onRefresh: _fetchRealms, | ||||
|               child: ListView.builder( | ||||
|                 itemCount: _realms?.length ?? 0, | ||||
|                 itemBuilder: (context, idx) { | ||||
|                   final realm = _realms![idx]; | ||||
|                   if (_isCompactView) { | ||||
|                     return ListTile( | ||||
|                       contentPadding: const EdgeInsets.symmetric(horizontal: 16), | ||||
|                       leading: AccountImage( | ||||
|                         content: realm.avatar, | ||||
|                         fallbackWidget: const Icon(Symbols.group, size: 20), | ||||
|                       ), | ||||
|                       title: Text(realm.name), | ||||
|                       subtitle: Text( | ||||
|                         realm.description, | ||||
|                         maxLines: 1, | ||||
|                         overflow: TextOverflow.ellipsis, | ||||
|                       ), | ||||
|                       trailing: PopupMenuButton( | ||||
|                         itemBuilder: (BuildContext context) => [ | ||||
|                           PopupMenuItem( | ||||
|                             child: Row( | ||||
|                               children: [ | ||||
|                                 const Icon(Symbols.edit), | ||||
|                                 const Gap(16), | ||||
|                                 Text('edit').tr(), | ||||
|                               ], | ||||
|                             ), | ||||
|                             onTap: () { | ||||
|                               GoRouter.of(context).pushNamed( | ||||
|                                 'realmManage', | ||||
|                                 queryParameters: {'editing': realm.alias}, | ||||
|                               ).then((value) { | ||||
|                                 if (value != null) { | ||||
|                                   _fetchRealms(); | ||||
|                                 } | ||||
|                               }); | ||||
|                             }, | ||||
|                           ), | ||||
|                           PopupMenuItem( | ||||
|                             child: Row( | ||||
|                               children: [ | ||||
|                                 const Icon(Symbols.delete), | ||||
|                                 const Gap(16), | ||||
|                                 Text('delete').tr(), | ||||
|                               ], | ||||
|                             ), | ||||
|                             onTap: () { | ||||
|                               _deleteRealm(realm); | ||||
|                             }, | ||||
|                           ), | ||||
|                         ], | ||||
|                       ), | ||||
|                       onTap: () { | ||||
|                         GoRouter.of(context).pushNamed( | ||||
|                           'realmDetail', | ||||
|                           pathParameters: {'alias': realm.alias}, | ||||
|                         ); | ||||
|                       }, | ||||
|                     ); | ||||
|                   } | ||||
|  | ||||
|                   return Container( | ||||
|                     constraints: BoxConstraints(maxWidth: 640), | ||||
|                     child: Card( | ||||
|                       margin: const EdgeInsets.all(12), | ||||
|                       child: InkWell( | ||||
|                         borderRadius: const BorderRadius.all(Radius.circular(8)), | ||||
|                         child: Column( | ||||
|                           crossAxisAlignment: CrossAxisAlignment.start, | ||||
|                           children: [ | ||||
|                             AspectRatio( | ||||
|                               aspectRatio: 16 / 7, | ||||
|                               child: Stack( | ||||
|                                 clipBehavior: Clip.none, | ||||
|                                 fit: StackFit.expand, | ||||
|             child: MediaQuery.removePadding( | ||||
|               context: context, | ||||
|               removeTop: true, | ||||
|               child: RefreshIndicator( | ||||
|                 onRefresh: _fetchRealms, | ||||
|                 child: ListView.builder( | ||||
|                   itemCount: _realms?.length ?? 0, | ||||
|                   itemBuilder: (context, idx) { | ||||
|                     final realm = _realms![idx]; | ||||
|                     if (_isCompactView) { | ||||
|                       return ListTile( | ||||
|                         contentPadding: const EdgeInsets.symmetric(horizontal: 16), | ||||
|                         leading: AccountImage( | ||||
|                           content: realm.avatar, | ||||
|                           fallbackWidget: const Icon(Symbols.group, size: 20), | ||||
|                         ), | ||||
|                         title: Text(realm.name), | ||||
|                         subtitle: Text( | ||||
|                           realm.description, | ||||
|                           maxLines: 1, | ||||
|                           overflow: TextOverflow.ellipsis, | ||||
|                         ), | ||||
|                         trailing: PopupMenuButton( | ||||
|                           itemBuilder: (BuildContext context) => [ | ||||
|                             PopupMenuItem( | ||||
|                               child: Row( | ||||
|                                 children: [ | ||||
|                                   Container( | ||||
|                                     color: Theme.of(context).colorScheme.surfaceContainer, | ||||
|                                     child: (realm.banner?.isEmpty ?? true) | ||||
|                                         ? const SizedBox.shrink() | ||||
|                                         : AutoResizeUniversalImage( | ||||
|                                             sn.getAttachmentUrl(realm.banner!), | ||||
|                                             fit: BoxFit.cover, | ||||
|                                           ), | ||||
|                                   ), | ||||
|                                   Positioned( | ||||
|                                     bottom: -30, | ||||
|                                     left: 18, | ||||
|                                     child: AccountImage( | ||||
|                                       content: realm.avatar, | ||||
|                                       radius: 24, | ||||
|                                       fallbackWidget: const Icon(Symbols.group, size: 24), | ||||
|                                     ), | ||||
|                                   ), | ||||
|                                   const Icon(Symbols.edit), | ||||
|                                   const Gap(16), | ||||
|                                   Text('edit').tr(), | ||||
|                                 ], | ||||
|                               ), | ||||
|                               onTap: () { | ||||
|                                 GoRouter.of(context).pushNamed( | ||||
|                                   'realmManage', | ||||
|                                   queryParameters: {'editing': realm.alias}, | ||||
|                                 ).then((value) { | ||||
|                                   if (value != null) { | ||||
|                                     _fetchRealms(); | ||||
|                                   } | ||||
|                                 }); | ||||
|                               }, | ||||
|                             ), | ||||
|                             PopupMenuItem( | ||||
|                               child: Row( | ||||
|                                 children: [ | ||||
|                                   const Icon(Symbols.delete), | ||||
|                                   const Gap(16), | ||||
|                                   Text('delete').tr(), | ||||
|                                 ], | ||||
|                               ), | ||||
|                               onTap: () { | ||||
|                                 _deleteRealm(realm); | ||||
|                               }, | ||||
|                             ), | ||||
|                             const Gap(20 + 12), | ||||
|                             Column( | ||||
|                               crossAxisAlignment: CrossAxisAlignment.start, | ||||
|                               children: [ | ||||
|                                 Text(realm.name).textStyle(Theme.of(context).textTheme.titleMedium!), | ||||
|                                 Text(realm.description).textStyle(Theme.of(context).textTheme.bodySmall!), | ||||
|                               ], | ||||
|                             ).padding(horizontal: 24, bottom: 14), | ||||
|                           ], | ||||
|                         ), | ||||
|                         onTap: () { | ||||
| @@ -233,10 +182,69 @@ class _RealmScreenState extends State<RealmScreen> { | ||||
|                             pathParameters: {'alias': realm.alias}, | ||||
|                           ); | ||||
|                         }, | ||||
|                       ); | ||||
|                     } | ||||
|  | ||||
|                     return Container( | ||||
|                       constraints: BoxConstraints(maxWidth: 640), | ||||
|                       child: Card( | ||||
|                         margin: const EdgeInsets.all(12), | ||||
|                         child: InkWell( | ||||
|                           borderRadius: const BorderRadius.all(Radius.circular(8)), | ||||
|                           child: Column( | ||||
|                             crossAxisAlignment: CrossAxisAlignment.start, | ||||
|                             children: [ | ||||
|                               AspectRatio( | ||||
|                                 aspectRatio: 16 / 7, | ||||
|                                 child: Stack( | ||||
|                                   clipBehavior: Clip.none, | ||||
|                                   fit: StackFit.expand, | ||||
|                                   children: [ | ||||
|                                     ClipRRect( | ||||
|                                       borderRadius: const BorderRadius.all(Radius.circular(8)), | ||||
|                                       child: Container( | ||||
|                                         color: Theme.of(context).colorScheme.surfaceContainer, | ||||
|                                         child: (realm.banner?.isEmpty ?? true) | ||||
|                                             ? const SizedBox.shrink() | ||||
|                                             : AutoResizeUniversalImage( | ||||
|                                                 sn.getAttachmentUrl(realm.banner!), | ||||
|                                                 fit: BoxFit.cover, | ||||
|                                               ), | ||||
|                                       ), | ||||
|                                     ), | ||||
|                                     Positioned( | ||||
|                                       bottom: -30, | ||||
|                                       left: 18, | ||||
|                                       child: AccountImage( | ||||
|                                         content: realm.avatar, | ||||
|                                         radius: 24, | ||||
|                                         fallbackWidget: const Icon(Symbols.group, size: 24), | ||||
|                                       ), | ||||
|                                     ), | ||||
|                                   ], | ||||
|                                 ), | ||||
|                               ), | ||||
|                               const Gap(20 + 12), | ||||
|                               Column( | ||||
|                                 crossAxisAlignment: CrossAxisAlignment.start, | ||||
|                                 children: [ | ||||
|                                   Text(realm.name).textStyle(Theme.of(context).textTheme.titleMedium!), | ||||
|                                   Text(realm.description).textStyle(Theme.of(context).textTheme.bodySmall!), | ||||
|                                 ], | ||||
|                               ).padding(horizontal: 24, bottom: 14), | ||||
|                             ], | ||||
|                           ), | ||||
|                           onTap: () { | ||||
|                             GoRouter.of(context).pushNamed( | ||||
|                               'realmDetail', | ||||
|                               pathParameters: {'alias': realm.alias}, | ||||
|                             ); | ||||
|                           }, | ||||
|                         ), | ||||
|                       ), | ||||
|                     ), | ||||
|                   ).center(); | ||||
|                 }, | ||||
|                     ).center(); | ||||
|                   }, | ||||
|                 ), | ||||
|               ), | ||||
|             ), | ||||
|           ), | ||||
|   | ||||
| @@ -18,6 +18,7 @@ import 'package:surface/types/realm.dart'; | ||||
| import 'package:surface/widgets/account/account_image.dart'; | ||||
| import 'package:surface/widgets/dialog.dart'; | ||||
| import 'package:surface/widgets/loading_indicator.dart'; | ||||
| import 'package:surface/widgets/navigation/app_scaffold.dart'; | ||||
| import 'package:surface/widgets/universal_image.dart'; | ||||
| import 'package:uuid/uuid.dart'; | ||||
|  | ||||
| @@ -179,7 +180,7 @@ class _RealmManageScreenState extends State<RealmManageScreen> { | ||||
|   Widget build(BuildContext context) { | ||||
|     final sn = context.read<SnNetworkProvider>(); | ||||
|  | ||||
|     return Scaffold( | ||||
|     return AppScaffold( | ||||
|       appBar: AppBar( | ||||
|         title: widget.editingRealmAlias != null | ||||
|             ? Text('screenRealmManage').tr() | ||||
|   | ||||
| @@ -8,13 +8,15 @@ import 'package:styled_widget/styled_widget.dart'; | ||||
| import 'package:surface/providers/sn_network.dart'; | ||||
| import 'package:surface/providers/user_directory.dart'; | ||||
| import 'package:surface/providers/userinfo.dart'; | ||||
| import 'package:surface/types/account.dart'; | ||||
| import 'package:surface/types/post.dart'; | ||||
| import 'package:surface/types/realm.dart'; | ||||
| import 'package:surface/widgets/account/account_image.dart'; | ||||
| import 'package:surface/widgets/account/account_select.dart'; | ||||
| import 'package:surface/widgets/dialog.dart'; | ||||
| import 'package:surface/widgets/navigation/app_scaffold.dart'; | ||||
| import 'package:very_good_infinite_list/very_good_infinite_list.dart'; | ||||
|  | ||||
| import '../../types/post.dart'; | ||||
|  | ||||
| class RealmDetailScreen extends StatefulWidget { | ||||
|   final String alias; | ||||
|  | ||||
| @@ -70,27 +72,19 @@ class _RealmDetailScreenState extends State<RealmDetailScreen> { | ||||
|   Widget build(BuildContext context) { | ||||
|     return DefaultTabController( | ||||
|       length: 3, | ||||
|       child: Scaffold( | ||||
|       child: AppScaffold( | ||||
|         body: NestedScrollView( | ||||
|           headerSliverBuilder: (BuildContext context, bool innerBoxIsScrolled) { | ||||
|             // These are the slivers that show up in the "outer" scroll view. | ||||
|             return <Widget>[ | ||||
|               SliverOverlapAbsorber( | ||||
|                 // This widget takes the overlapping behavior of the SliverAppBar, | ||||
|                 // and redirects it to the SliverOverlapInjector below. If it is | ||||
|                 // missing, then it is possible for the nested "inner" scroll view | ||||
|                 // below to end up under the SliverAppBar even when the inner | ||||
|                 // scroll view thinks it has not been scrolled. | ||||
|                 // This is not necessary if the "headerSliverBuilder" only builds | ||||
|                 // widgets that do not overlap the next sliver. | ||||
|                 handle: NestedScrollView.sliverOverlapAbsorberHandleFor(context), | ||||
|                 sliver: SliverAppBar( | ||||
|                   title: Text(_realm?.name ?? 'loading'.tr()), | ||||
|                   bottom: TabBar( | ||||
|                     tabs: [ | ||||
|                       Tab(icon: const Icon(Symbols.home)), | ||||
|                       Tab(icon: const Icon(Symbols.group)), | ||||
|                       Tab(icon: const Icon(Symbols.settings)), | ||||
|                       Tab(icon: Icon(Symbols.home, color: Theme.of(context).appBarTheme.foregroundColor)), | ||||
|                       Tab(icon: Icon(Symbols.group, color: Theme.of(context).appBarTheme.foregroundColor)), | ||||
|                       Tab(icon: Icon(Symbols.settings, color: Theme.of(context).appBarTheme.foregroundColor)), | ||||
|                     ], | ||||
|                   ), | ||||
|                 ), | ||||
| @@ -119,7 +113,7 @@ class _RealmDetailHomeWidget extends StatelessWidget { | ||||
|   final SnRealm? realm; | ||||
|   final List<SnPublisher>? publishers; | ||||
|  | ||||
|   const _RealmDetailHomeWidget({super.key, required this.realm, this.publishers}); | ||||
|   const _RealmDetailHomeWidget({required this.realm, this.publishers}); | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
| @@ -175,7 +169,7 @@ class _RealmDetailHomeWidget extends StatelessWidget { | ||||
| class _RealmMemberListWidget extends StatefulWidget { | ||||
|   final SnRealm? realm; | ||||
|  | ||||
|   const _RealmMemberListWidget({super.key, this.realm}); | ||||
|   const _RealmMemberListWidget({this.realm}); | ||||
|  | ||||
|   @override | ||||
|   State<_RealmMemberListWidget> createState() => _RealmMemberListWidgetState(); | ||||
| @@ -237,13 +231,35 @@ class _RealmMemberListWidgetState extends State<_RealmMemberListWidget> { | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   void _showMemberAdd() { | ||||
|     showModalBottomSheet( | ||||
|   Future<void> _addMember(SnAccount related) async { | ||||
|     setState(() => _isBusy = true); | ||||
|  | ||||
|     try { | ||||
|       final sn = context.read<SnNetworkProvider>(); | ||||
|       await sn.client.post( | ||||
|         '/cgi/id/realms/${widget.realm!.alias}/members', | ||||
|         data: {'related': related.name}, | ||||
|       ); | ||||
|       if (!mounted) return; | ||||
|       context.showSnackbar('realmMemberAdded'.tr()); | ||||
|     } catch (err) { | ||||
|       if (!mounted) return; | ||||
|       context.showErrorDialog(err); | ||||
|     } finally { | ||||
|       setState(() => _isBusy = false); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   void _showMemberAdd() async { | ||||
|     final user = await showModalBottomSheet<SnAccount?>( | ||||
|       context: context, | ||||
|       builder: (context) => _NewRealmMemberWidget( | ||||
|         realm: widget.realm!, | ||||
|       builder: (context) => AccountSelect( | ||||
|         title: 'realmMemberAdd'.tr(), | ||||
|       ), | ||||
|     ); | ||||
|     if (!mounted) return; | ||||
|     if (user == null) return; | ||||
|     _addMember(user); | ||||
|   } | ||||
|  | ||||
|   @override | ||||
| @@ -301,90 +317,11 @@ class _RealmMemberListWidgetState extends State<_RealmMemberListWidget> { | ||||
|   } | ||||
| } | ||||
|  | ||||
| class _NewRealmMemberWidget extends StatefulWidget { | ||||
|   final SnRealm realm; | ||||
|  | ||||
|   const _NewRealmMemberWidget({super.key, required this.realm}); | ||||
|  | ||||
|   @override | ||||
|   State<_NewRealmMemberWidget> createState() => _NewRealmMemberWidgetState(); | ||||
| } | ||||
|  | ||||
| class _NewRealmMemberWidgetState extends State<_NewRealmMemberWidget> { | ||||
|   bool _isBusy = false; | ||||
|  | ||||
|   final TextEditingController _relatedController = TextEditingController(); | ||||
|  | ||||
|   Future<void> _performAction() async { | ||||
|     if (_relatedController.text.isEmpty) return; | ||||
|  | ||||
|     setState(() => _isBusy = true); | ||||
|  | ||||
|     try { | ||||
|       final sn = context.read<SnNetworkProvider>(); | ||||
|       await sn.client.post( | ||||
|         '/cgi/id/realms/${widget.realm.alias}/members', | ||||
|         data: { | ||||
|           'related': _relatedController.text, | ||||
|         }, | ||||
|       ); | ||||
|       if (!mounted) return; | ||||
|       Navigator.pop(context, true); | ||||
|       context.showSnackbar('channelMemberAdded'.tr()); | ||||
|     } catch (err) { | ||||
|       if (!mounted) return; | ||||
|       context.showErrorDialog(err); | ||||
|     } finally { | ||||
|       setState(() => _isBusy = false); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   void dispose() { | ||||
|     super.dispose(); | ||||
|     _relatedController.dispose(); | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     return StyledWidget(Column( | ||||
|       crossAxisAlignment: CrossAxisAlignment.start, | ||||
|       children: [ | ||||
|         Text( | ||||
|           'realmMemberAdd', | ||||
|           style: Theme.of(context).textTheme.titleLarge, | ||||
|         ).tr(), | ||||
|         const Gap(12), | ||||
|         TextField( | ||||
|           controller: _relatedController, | ||||
|           readOnly: _isBusy, | ||||
|           autocorrect: false, | ||||
|           autofocus: true, | ||||
|           textCapitalization: TextCapitalization.none, | ||||
|           decoration: InputDecoration( | ||||
|             labelText: 'fieldMemberRelatedName'.tr(), | ||||
|             suffix: SizedBox( | ||||
|               height: 24, | ||||
|               child: IconButton( | ||||
|                 onPressed: _isBusy ? null : () => _performAction(), | ||||
|                 icon: Icon(Symbols.send), | ||||
|                 visualDensity: const VisualDensity(horizontal: -4, vertical: -4), | ||||
|                 padding: EdgeInsets.zero, | ||||
|               ), | ||||
|             ), | ||||
|           ), | ||||
|           onTapOutside: (_) => FocusManager.instance.primaryFocus?.unfocus(), | ||||
|         ) | ||||
|       ], | ||||
|     )).padding(all: 24); | ||||
|   } | ||||
| } | ||||
|  | ||||
| class _RealmSettingsWidget extends StatefulWidget { | ||||
|   final SnRealm? realm; | ||||
|   final Function() onUpdate; | ||||
|  | ||||
|   const _RealmSettingsWidget({super.key, required this.realm, required this.onUpdate}); | ||||
|   const _RealmSettingsWidget({required this.realm, required this.onUpdate}); | ||||
|  | ||||
|   @override | ||||
|   State<_RealmSettingsWidget> createState() => _RealmSettingsWidgetState(); | ||||
| @@ -428,7 +365,7 @@ class _RealmSettingsWidgetState extends State<_RealmSettingsWidget> { | ||||
|  | ||||
|     return Column( | ||||
|       children: [ | ||||
|         const Gap(16), | ||||
|         const Gap(8), | ||||
|         ListTile( | ||||
|           leading: const Icon(Symbols.edit), | ||||
|           trailing: const Icon(Symbols.chevron_right), | ||||
|   | ||||
| @@ -5,6 +5,7 @@ import 'package:dropdown_button2/dropdown_button2.dart'; | ||||
| import 'package:easy_localization/easy_localization.dart'; | ||||
| import 'package:flutter/foundation.dart'; | ||||
| import 'package:flutter/material.dart'; | ||||
| import 'package:flutter_colorpicker/flutter_colorpicker.dart'; | ||||
| import 'package:go_router/go_router.dart'; | ||||
| import 'package:image_picker/image_picker.dart'; | ||||
| import 'package:material_symbols_icons/symbols.dart'; | ||||
| @@ -17,6 +18,18 @@ import 'package:surface/providers/sn_network.dart'; | ||||
| import 'package:surface/providers/theme.dart'; | ||||
| import 'package:surface/theme.dart'; | ||||
| import 'package:surface/widgets/dialog.dart'; | ||||
| import 'package:surface/widgets/navigation/app_scaffold.dart'; | ||||
|  | ||||
| const Map<String, Color> kColorSchemes = { | ||||
|   'colorSchemeIndigo': Colors.indigo, | ||||
|   'colorSchemeBlue': Colors.blue, | ||||
|   'colorSchemeGreen': Colors.green, | ||||
|   'colorSchemeYellow': Colors.yellow, | ||||
|   'colorSchemeOrange': Colors.orange, | ||||
|   'colorSchemeRed': Colors.red, | ||||
|   'colorSchemeWhite': Colors.white, | ||||
|   'colorSchemeBlack': Colors.black, | ||||
| }; | ||||
|  | ||||
| class SettingsScreen extends StatefulWidget { | ||||
|   const SettingsScreen({super.key}); | ||||
| @@ -55,7 +68,11 @@ class _SettingsScreenState extends State<SettingsScreen> { | ||||
|   Widget build(BuildContext context) { | ||||
|     final sn = context.read<SnNetworkProvider>(); | ||||
|  | ||||
|     return Scaffold( | ||||
|     return AppScaffold( | ||||
|       appBar: AppBar( | ||||
|         leading: const PageBackButton(), | ||||
|         title: Text('screenSettings').tr(), | ||||
|       ), | ||||
|       body: SingleChildScrollView( | ||||
|         child: Column( | ||||
|           spacing: 16, | ||||
| @@ -65,6 +82,48 @@ class _SettingsScreenState extends State<SettingsScreen> { | ||||
|               crossAxisAlignment: CrossAxisAlignment.start, | ||||
|               children: [ | ||||
|                 Text('settingsAppearance').bold().fontSize(17).tr().padding(horizontal: 20, bottom: 4), | ||||
|                 ListTile( | ||||
|                   title: Text('settingsDisplayLanguage').tr(), | ||||
|                   subtitle: Text('settingsDisplayLanguageDescription').tr(), | ||||
|                   contentPadding: const EdgeInsets.only(left: 24, right: 17), | ||||
|                   leading: const Icon(Symbols.translate), | ||||
|                   trailing: DropdownButtonHideUnderline( | ||||
|                     child: DropdownButton2<Locale?>( | ||||
|                       isExpanded: true, | ||||
|                       items: [ | ||||
|                         ...EasyLocalization.of(context)!.supportedLocales.mapIndexed((idx, ele) { | ||||
|                           return DropdownMenuItem<Locale?>( | ||||
|                             value: ele, | ||||
|                             child: Text('${ele.languageCode}-${ele.countryCode}').fontSize(14), | ||||
|                           ); | ||||
|                         }), | ||||
|                         DropdownMenuItem<Locale?>( | ||||
|                           value: null, | ||||
|                           child: Text('settingsDisplayLanguageSystem').tr().fontSize(14), | ||||
|                         ), | ||||
|                       ], | ||||
|                       value: EasyLocalization.of(context)!.currentLocale, | ||||
|                       onChanged: (Locale? value) { | ||||
|                         if (value != null) { | ||||
|                           EasyLocalization.of(context)!.setLocale(value); | ||||
|                         } else { | ||||
|                           EasyLocalization.of(context)!.resetLocale(); | ||||
|                         } | ||||
|                       }, | ||||
|                       buttonStyleData: const ButtonStyleData( | ||||
|                         padding: EdgeInsets.symmetric( | ||||
|                           horizontal: 16, | ||||
|                           vertical: 5, | ||||
|                         ), | ||||
|                         height: 40, | ||||
|                         width: 160, | ||||
|                       ), | ||||
|                       menuItemStyleData: const MenuItemStyleData( | ||||
|                         height: 40, | ||||
|                       ), | ||||
|                     ), | ||||
|                   ), | ||||
|                 ), | ||||
|                 if (!kIsWeb) | ||||
|                   ListTile( | ||||
|                     title: Text('settingsBackgroundImage').tr(), | ||||
| @@ -77,7 +136,7 @@ class _SettingsScreenState extends State<SettingsScreen> { | ||||
|                       if (image == null) return; | ||||
|  | ||||
|                       await File(image.path).copy('$_docBasepath/app_background_image'); | ||||
|                       _prefs.setBool('has_background_image', true); | ||||
|                       _prefs.setBool(kAppBackgroundStoreKey, true); | ||||
|  | ||||
|                       setState(() {}); | ||||
|                     }, | ||||
| @@ -98,7 +157,7 @@ class _SettingsScreenState extends State<SettingsScreen> { | ||||
|                           trailing: const Icon(Symbols.chevron_right), | ||||
|                           onTap: () { | ||||
|                             File('$_docBasepath/app_background_image').deleteSync(); | ||||
|                             _prefs.remove('has_background_image'); | ||||
|                             _prefs.remove(kAppBackgroundStoreKey); | ||||
|                             setState(() {}); | ||||
|                           }, | ||||
|                         ); | ||||
| @@ -108,7 +167,7 @@ class _SettingsScreenState extends State<SettingsScreen> { | ||||
|                   subtitle: Text('settingsThemeMaterial3Description').tr(), | ||||
|                   contentPadding: const EdgeInsets.only(left: 24, right: 17), | ||||
|                   secondary: const Icon(Symbols.new_releases), | ||||
|                   value: _prefs.getBool(kMaterialYouToggleStoreKey) ?? false, | ||||
|                   value: _prefs.getBool(kMaterialYouToggleStoreKey) ?? true, | ||||
|                   onChanged: (value) { | ||||
|                     setState(() { | ||||
|                       _prefs.setBool( | ||||
| @@ -116,10 +175,176 @@ class _SettingsScreenState extends State<SettingsScreen> { | ||||
|                         value ?? false, | ||||
|                       ); | ||||
|                     }); | ||||
|                     final th = context.watch<ThemeProvider>(); | ||||
|                     final th = context.read<ThemeProvider>(); | ||||
|                     th.reloadTheme(useMaterial3: value ?? false); | ||||
|                   }, | ||||
|                 ), | ||||
|                 ListTile( | ||||
|                   leading: const Icon(Symbols.format_paint), | ||||
|                   title: Text('settingsColorScheme').tr(), | ||||
|                   subtitle: Text('settingsColorSchemeDescription').tr(), | ||||
|                   contentPadding: const EdgeInsets.symmetric(horizontal: 24), | ||||
|                   trailing: const Icon(Symbols.chevron_right), | ||||
|                   onTap: () async { | ||||
|                     Color pickerColor = Color(_prefs.getInt(kAppColorSchemeStoreKey) ?? Colors.indigo.value); | ||||
|                     final color = await showDialog<Color?>( | ||||
|                       context: context, | ||||
|                       builder: (context) => | ||||
|                           AlertDialog( | ||||
|                             content: SingleChildScrollView( | ||||
|                               child: ColorPicker( | ||||
|                                 pickerColor: pickerColor, | ||||
|                                 onColorChanged: (color) => pickerColor = color, | ||||
|                                 enableAlpha: false, | ||||
|                                 hexInputBar: true, | ||||
|                               ), | ||||
|                             ), | ||||
|                             actions: <Widget>[ | ||||
|                               TextButton( | ||||
|                                 child: const Text('dialogDismiss').tr(), | ||||
|                                 onPressed: () { | ||||
|                                   Navigator.of(context).pop(); | ||||
|                                 }, | ||||
|                               ), | ||||
|                               TextButton( | ||||
|                                 child: const Text('dialogConfirm').tr(), | ||||
|                                 onPressed: () { | ||||
|                                   Navigator.of(context).pop(pickerColor); | ||||
|                                 }, | ||||
|                               ), | ||||
|                             ], | ||||
|                           ), | ||||
|                     ); | ||||
|  | ||||
|                     if (color == null || !context.mounted) return; | ||||
|  | ||||
|                     _prefs.setInt(kAppColorSchemeStoreKey, color.value); | ||||
|                     final th = context.read<ThemeProvider>(); | ||||
|                     th.reloadTheme(seedColorOverride: color); | ||||
|                     setState(() {}); | ||||
|  | ||||
|                     context.showSnackbar('colorSchemeApplied'.tr()); | ||||
|                   }, | ||||
|                 ), | ||||
|                 ListTile( | ||||
|                   leading: const Icon(Symbols.palette), | ||||
|                   title: Text('settingsColorSeed').tr(), | ||||
|                   subtitle: Text('settingsColorSeedDescription').tr(), | ||||
|                   contentPadding: const EdgeInsets.only(left: 24, right: 17), | ||||
|                   trailing: DropdownButtonHideUnderline( | ||||
|                     child: DropdownButton2<int?>( | ||||
|                       isExpanded: true, | ||||
|                       items: [ | ||||
|                         ...kColorSchemes.entries.mapIndexed((idx, ele) { | ||||
|                           return DropdownMenuItem<int>( | ||||
|                             value: idx, | ||||
|                             child: Text(ele.key).tr(), | ||||
|                           ); | ||||
|                         }), | ||||
|                         DropdownMenuItem<int>( | ||||
|                           value: -1, | ||||
|                           child: Text('custom').tr(), | ||||
|                         ), | ||||
|                       ], | ||||
|                       value: _prefs.getInt(kAppColorSchemeStoreKey) == null | ||||
|                           ? 1 | ||||
|                           : kColorSchemes.values | ||||
|                           .toList() | ||||
|                           .indexWhere((ele) => ele.value == _prefs.getInt(kAppColorSchemeStoreKey)), | ||||
|                       onChanged: (int? value) { | ||||
|                         if (value != null && value != -1) { | ||||
|                           _prefs.setInt(kAppColorSchemeStoreKey, kColorSchemes.values | ||||
|                               .elementAt(value) | ||||
|                               .value); | ||||
|                           final th = context.read<ThemeProvider>(); | ||||
|                           th.reloadTheme(seedColorOverride: kColorSchemes.values.elementAt(value)); | ||||
|                           setState(() {}); | ||||
|  | ||||
|                           context.showSnackbar('colorSchemeApplied'.tr()); | ||||
|                         } | ||||
|                       }, | ||||
|                       buttonStyleData: const ButtonStyleData( | ||||
|                         padding: EdgeInsets.symmetric( | ||||
|                           horizontal: 16, | ||||
|                           vertical: 5, | ||||
|                         ), | ||||
|                         height: 40, | ||||
|                         width: 160, | ||||
|                       ), | ||||
|                       menuItemStyleData: const MenuItemStyleData( | ||||
|                         height: 40, | ||||
|                       ), | ||||
|                     ), | ||||
|                   ), | ||||
|                 ), | ||||
|                 CheckboxListTile( | ||||
|                   secondary: const Icon(Symbols.blur_on), | ||||
|                   title: Text('settingsAppBarTransparent').tr(), | ||||
|                   subtitle: Text('settingsAppBarTransparentDescription').tr(), | ||||
|                   contentPadding: const EdgeInsets.only(left: 24, right: 17), | ||||
|                   value: _prefs.getBool(kAppbarTransparentStoreKey) ?? false, | ||||
|                   onChanged: (value) { | ||||
|                     _prefs.setBool(kAppbarTransparentStoreKey, value ?? false); | ||||
|                     final th = context.read<ThemeProvider>(); | ||||
|                     th.reloadTheme(); | ||||
|                     setState(() {}); | ||||
|                   }, | ||||
|                 ), | ||||
|                 CheckboxListTile( | ||||
|                   secondary: const Icon(Symbols.left_panel_close), | ||||
|                   title: Text('settingsDrawerPreferCollapse').tr(), | ||||
|                   subtitle: Text('settingsDrawerPreferCollapseDescription').tr(), | ||||
|                   contentPadding: const EdgeInsets.only(left: 24, right: 17), | ||||
|                   value: _prefs.getBool(kAppDrawerPreferCollapse) ?? false, | ||||
|                   onChanged: (value) { | ||||
|                     _prefs.setBool(kAppDrawerPreferCollapse, value ?? false); | ||||
|                     final cfg = context.read<ConfigProvider>(); | ||||
|                     cfg.calcDrawerSize(context); | ||||
|                     setState(() {}); | ||||
|                   }, | ||||
|                 ), | ||||
|               ], | ||||
|             ), | ||||
|             Column( | ||||
|               crossAxisAlignment: CrossAxisAlignment.start, | ||||
|               children: [ | ||||
|                 Text('settingsFeatures').bold().fontSize(17).tr().padding(horizontal: 20, bottom: 4), | ||||
|                 CheckboxListTile( | ||||
|                   secondary: const Icon(Symbols.vibration), | ||||
|                   contentPadding: const EdgeInsets.only(left: 24, right: 17), | ||||
|                   title: Text('settingsNotifyWithHaptic').tr(), | ||||
|                   subtitle: Text('settingsNotifyWithHapticDescription').tr(), | ||||
|                   value: _prefs.getBool(kAppNotifyWithHaptic) ?? true, | ||||
|                   onChanged: (value) { | ||||
|                     setState(() { | ||||
|                       _prefs.setBool(kAppNotifyWithHaptic, value ?? false); | ||||
|                     }); | ||||
|                   }, | ||||
|                 ), | ||||
|                 CheckboxListTile( | ||||
|                   secondary: const Icon(Symbols.link), | ||||
|                   title: Text('settingsExpandPostLink').tr(), | ||||
|                   subtitle: Text('settingsExpandPostLinkDescription').tr(), | ||||
|                   contentPadding: const EdgeInsets.only(left: 24, right: 17), | ||||
|                   value: _prefs.getBool(kAppExpandPostLink) ?? true, | ||||
|                   onChanged: (value) { | ||||
|                     setState(() { | ||||
|                       _prefs.setBool(kAppExpandPostLink, value ?? false); | ||||
|                     }); | ||||
|                   }, | ||||
|                 ), | ||||
|                 CheckboxListTile( | ||||
|                   secondary: const Icon(Symbols.chat), | ||||
|                   title: Text('settingsExpandChatLink').tr(), | ||||
|                   subtitle: Text('settingsExpandChatLinkDescription').tr(), | ||||
|                   contentPadding: const EdgeInsets.only(left: 24, right: 17), | ||||
|                   value: _prefs.getBool(kAppExpandChatLink) ?? true, | ||||
|                   onChanged: (value) { | ||||
|                     setState(() { | ||||
|                       _prefs.setBool(kAppExpandChatLink, value ?? false); | ||||
|                     }); | ||||
|                   }, | ||||
|                 ), | ||||
|               ], | ||||
|             ), | ||||
|             Column( | ||||
| @@ -162,7 +387,8 @@ class _SettingsScreenState extends State<SettingsScreen> { | ||||
|                           ('Custom', _serverUrlController.text), | ||||
|                       ] | ||||
|                           .map( | ||||
|                             (item) => DropdownMenuItem<String>( | ||||
|                             (item) => | ||||
|                             DropdownMenuItem<String>( | ||||
|                               value: item.$2, | ||||
|                               child: Column( | ||||
|                                 mainAxisSize: MainAxisSize.max, | ||||
| @@ -174,7 +400,7 @@ class _SettingsScreenState extends State<SettingsScreen> { | ||||
|                                 ], | ||||
|                               ), | ||||
|                             ), | ||||
|                           ) | ||||
|                       ) | ||||
|                           .toList(), | ||||
|                       value: _serverUrlController.text, | ||||
|                       onChanged: (String? value) { | ||||
| @@ -189,7 +415,7 @@ class _SettingsScreenState extends State<SettingsScreen> { | ||||
|                           horizontal: 16, | ||||
|                           vertical: 5, | ||||
|                         ), | ||||
|                         height: 40, | ||||
|                         height: 56, | ||||
|                         width: 160, | ||||
|                       ), | ||||
|                       menuItemStyleData: const MenuItemStyleData( | ||||
| @@ -229,11 +455,12 @@ class _SettingsScreenState extends State<SettingsScreen> { | ||||
|                       isExpanded: true, | ||||
|                       items: kImageQualityLevel.entries | ||||
|                           .map( | ||||
|                             (item) => DropdownMenuItem<FilterQuality>( | ||||
|                             (item) => | ||||
|                             DropdownMenuItem<FilterQuality>( | ||||
|                               value: item.value, | ||||
|                               child: Text(item.key).tr().fontSize(14), | ||||
|                             ), | ||||
|                           ) | ||||
|                       ) | ||||
|                           .toList(), | ||||
|                       onChanged: (FilterQuality? value) { | ||||
|                         if (value == null) return; | ||||
|   | ||||
| @@ -8,9 +8,20 @@ import 'package:flutter/foundation.dart'; | ||||
| import 'package:flutter/material.dart'; | ||||
| import 'package:gap/gap.dart'; | ||||
| import 'package:go_router/go_router.dart'; | ||||
| import 'package:material_symbols_icons/material_symbols_icons.dart'; | ||||
| import 'package:provider/provider.dart'; | ||||
| import 'package:receive_sharing_intent/receive_sharing_intent.dart'; | ||||
| import 'package:styled_widget/styled_widget.dart'; | ||||
| import 'package:surface/controllers/post_write_controller.dart'; | ||||
| import 'package:surface/providers/channel.dart'; | ||||
| import 'package:surface/providers/user_directory.dart'; | ||||
| import 'package:surface/providers/userinfo.dart'; | ||||
| import 'package:surface/screens/chat/room.dart'; | ||||
| import 'package:surface/screens/post/post_editor.dart'; | ||||
| import 'package:surface/types/chat.dart'; | ||||
| import 'package:surface/widgets/account/account_image.dart'; | ||||
| import 'package:surface/widgets/dialog.dart'; | ||||
| import 'package:surface/widgets/loading_indicator.dart'; | ||||
|  | ||||
| class AppSharingListener extends StatefulWidget { | ||||
|   final Widget child; | ||||
| @@ -51,20 +62,39 @@ class _AppSharingListenerState extends State<AppSharingListener> { | ||||
|                           pathParameters: { | ||||
|                             'mode': 'stories', | ||||
|                           }, | ||||
|                           extra: PostEditorExtraProps( | ||||
|                           extra: PostEditorExtra( | ||||
|                             text: value | ||||
|                                 .where((e) => [SharedMediaType.text, SharedMediaType.url].contains(e.type)) | ||||
|                                 .map((e) => e.path).join('\n'), | ||||
|                                 .map((e) => e.path) | ||||
|                                 .join('\n'), | ||||
|                             attachments: value | ||||
|                                 .where((e) => [SharedMediaType.video, SharedMediaType.file, SharedMediaType.image].contains(e.type)) | ||||
|                                 .map((e) => PostWriteMedia.fromFile(XFile(e.path))).toList(), | ||||
|                                 .where((e) => [SharedMediaType.video, SharedMediaType.file, SharedMediaType.image] | ||||
|                                     .contains(e.type)) | ||||
|                                 .map((e) => PostWriteMedia.fromFile(XFile(e.path))) | ||||
|                                 .toList(), | ||||
|                           ), | ||||
|                         ); | ||||
|                         Navigator.pop(context); | ||||
|                       }, | ||||
|                     ), | ||||
|                     ListTile( | ||||
|                       contentPadding: const EdgeInsets.symmetric(horizontal: 24), | ||||
|                       shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(8)), | ||||
|                       leading: Icon(Icons.chat_outlined), | ||||
|                       trailing: const Icon(Icons.chevron_right), | ||||
|                       title: Text('shareIntentSendChannel').tr(), | ||||
|                       onTap: () { | ||||
|                         showModalBottomSheet( | ||||
|                           context: context, | ||||
|                           builder: (context) => _ShareIntentChannelSelect(value: value), | ||||
|                         ).then((val) { | ||||
|                           if (!context.mounted) return; | ||||
|                           if (val == true) Navigator.pop(context); | ||||
|                         }); | ||||
|                       }, | ||||
|                     ), | ||||
|                   ], | ||||
|                 ), | ||||
|                 ).width(280), | ||||
|               ) | ||||
|             ], | ||||
|           ), | ||||
| @@ -103,7 +133,7 @@ class _AppSharingListenerState extends State<AppSharingListener> { | ||||
|   @override | ||||
|   void initState() { | ||||
|     super.initState(); | ||||
|     if(!kIsWeb && (Platform.isAndroid || Platform.isIOS)) { | ||||
|     if (!kIsWeb && (Platform.isAndroid || Platform.isIOS)) { | ||||
|       _initialize(); | ||||
|       _initialHandle(); | ||||
|     } | ||||
| @@ -120,3 +150,193 @@ class _AppSharingListenerState extends State<AppSharingListener> { | ||||
|     return widget.child; | ||||
|   } | ||||
| } | ||||
|  | ||||
| class _ShareIntentChannelSelect extends StatefulWidget { | ||||
|   final Iterable<SharedMediaFile> value; | ||||
|  | ||||
|   const _ShareIntentChannelSelect({required this.value}); | ||||
|  | ||||
|   @override | ||||
|   State<_ShareIntentChannelSelect> createState() => _ShareIntentChannelSelectState(); | ||||
| } | ||||
|  | ||||
| class _ShareIntentChannelSelectState extends State<_ShareIntentChannelSelect> { | ||||
|   bool _isBusy = true; | ||||
|  | ||||
|   List<SnChannel>? _channels; | ||||
|   Map<int, SnChatMessage>? _lastMessages; | ||||
|  | ||||
|   void _refreshChannels() { | ||||
|     final ua = context.read<UserProvider>(); | ||||
|     if (!ua.isAuthorized) { | ||||
|       setState(() => _isBusy = false); | ||||
|       return; | ||||
|     } | ||||
|  | ||||
|     final chan = context.read<ChatChannelProvider>(); | ||||
|     chan.fetchChannels().listen((channels) async { | ||||
|       final lastMessages = await chan.getLastMessages(channels); | ||||
|       _lastMessages = {for (final val in lastMessages) val.channelId: val}; | ||||
|       channels.sort((a, b) { | ||||
|         if (_lastMessages!.containsKey(a.id) && _lastMessages!.containsKey(b.id)) { | ||||
|           return _lastMessages![b.id]!.createdAt.compareTo(_lastMessages![a.id]!.createdAt); | ||||
|         } | ||||
|         if (_lastMessages!.containsKey(a.id)) return -1; | ||||
|         if (_lastMessages!.containsKey(b.id)) return 1; | ||||
|         return 0; | ||||
|       }); | ||||
|  | ||||
|       if (!mounted) return; | ||||
|       final ud = context.read<UserDirectoryProvider>(); | ||||
|       for (final channel in channels) { | ||||
|         if (channel.type == 1) { | ||||
|           await ud.listAccount( | ||||
|             channel.members | ||||
|                     ?.cast<SnChannelMember?>() | ||||
|                     .map((ele) => ele?.accountId) | ||||
|                     .where((ele) => ele != null) | ||||
|                     .toSet() ?? | ||||
|                 {}, | ||||
|           ); | ||||
|         } | ||||
|       } | ||||
|  | ||||
|       if (mounted) setState(() => _channels = channels); | ||||
|     }) | ||||
|       ..onError((err) { | ||||
|         if (!mounted) return; | ||||
|         context.showErrorDialog(err); | ||||
|         setState(() => _isBusy = false); | ||||
|       }) | ||||
|       ..onDone(() { | ||||
|         if (!mounted) return; | ||||
|         setState(() => _isBusy = false); | ||||
|       }); | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   void initState() { | ||||
|     super.initState(); | ||||
|     _refreshChannels(); | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     final ua = context.read<UserProvider>(); | ||||
|     final ud = context.read<UserDirectoryProvider>(); | ||||
|  | ||||
|     return Column( | ||||
|       children: [ | ||||
|         Row( | ||||
|           crossAxisAlignment: CrossAxisAlignment.center, | ||||
|           children: [ | ||||
|             const Icon(Symbols.chat, size: 24), | ||||
|             const Gap(16), | ||||
|             Text('shareIntentSendChannel', style: Theme.of(context).textTheme.titleLarge).tr(), | ||||
|           ], | ||||
|         ).padding(horizontal: 20, top: 16, bottom: 12), | ||||
|         LoadingIndicator(isActive: _isBusy), | ||||
|         Expanded( | ||||
|           child: MediaQuery.removePadding( | ||||
|             context: context, | ||||
|             removeTop: true, | ||||
|             child: RefreshIndicator( | ||||
|               onRefresh: () => Future.sync(() => _refreshChannels()), | ||||
|               child: ListView.builder( | ||||
|                 itemCount: _channels?.length ?? 0, | ||||
|                 itemBuilder: (context, idx) { | ||||
|                   final channel = _channels![idx]; | ||||
|                   final lastMessage = _lastMessages?[channel.id]; | ||||
|  | ||||
|                   if (channel.type == 1) { | ||||
|                     final otherMember = channel.members?.cast<SnChannelMember?>().firstWhere( | ||||
|                           (ele) => ele?.accountId != ua.user?.id, | ||||
|                           orElse: () => null, | ||||
|                         ); | ||||
|  | ||||
|                     return ListTile( | ||||
|                       title: Text(ud.getAccountFromCache(otherMember?.accountId)?.nick ?? channel.name), | ||||
|                       subtitle: lastMessage != null | ||||
|                           ? Text( | ||||
|                               '${ud.getAccountFromCache(lastMessage.sender.accountId)?.nick}: ${lastMessage.body['text'] ?? 'Unable preview'}', | ||||
|                               maxLines: 1, | ||||
|                               overflow: TextOverflow.ellipsis, | ||||
|                             ) | ||||
|                           : Text( | ||||
|                               'channelDirectMessageDescription'.tr(args: [ | ||||
|                                 '@${ud.getAccountFromCache(otherMember?.accountId)?.name}', | ||||
|                               ]), | ||||
|                               maxLines: 1, | ||||
|                               overflow: TextOverflow.ellipsis, | ||||
|                             ), | ||||
|                       contentPadding: const EdgeInsets.symmetric(horizontal: 16), | ||||
|                       leading: AccountImage( | ||||
|                         content: ud.getAccountFromCache(otherMember?.accountId)?.avatar, | ||||
|                       ), | ||||
|                       onTap: () { | ||||
|                         GoRouter.of(context).pushNamed( | ||||
|                           'chatRoom', | ||||
|                           pathParameters: { | ||||
|                             'scope': channel.realm?.alias ?? 'global', | ||||
|                             'alias': channel.alias, | ||||
|                           }, | ||||
|                         ).then((value) { | ||||
|                           if (mounted) _refreshChannels(); | ||||
|                         }); | ||||
|                       }, | ||||
|                     ); | ||||
|                   } | ||||
|  | ||||
|                   return ListTile( | ||||
|                     title: Text(channel.name), | ||||
|                     subtitle: lastMessage != null | ||||
|                         ? Text( | ||||
|                             '${ud.getAccountFromCache(lastMessage.sender.accountId)?.nick}: ${lastMessage.body['text'] ?? 'Unable preview'}', | ||||
|                             maxLines: 1, | ||||
|                             overflow: TextOverflow.ellipsis, | ||||
|                           ) | ||||
|                         : Text( | ||||
|                             channel.description, | ||||
|                             maxLines: 1, | ||||
|                             overflow: TextOverflow.ellipsis, | ||||
|                           ), | ||||
|                     contentPadding: const EdgeInsets.symmetric(horizontal: 16), | ||||
|                     leading: AccountImage( | ||||
|                       content: null, | ||||
|                       fallbackWidget: const Icon(Symbols.chat, size: 20), | ||||
|                     ), | ||||
|                     onTap: () { | ||||
|                       Navigator.pop(context, true); | ||||
|                       GoRouter.of(context) | ||||
|                           .pushNamed( | ||||
|                         'chatRoom', | ||||
|                         pathParameters: { | ||||
|                           'scope': channel.realm?.alias ?? 'global', | ||||
|                           'alias': channel.alias, | ||||
|                         }, | ||||
|                         extra: ChatRoomScreenExtra( | ||||
|                           initialText: widget.value | ||||
|                               .where((e) => [SharedMediaType.text, SharedMediaType.url].contains(e.type)) | ||||
|                               .map((e) => e.path) | ||||
|                               .join('\n'), | ||||
|                           initialAttachments: widget.value | ||||
|                               .where((e) => | ||||
|                                   [SharedMediaType.video, SharedMediaType.file, SharedMediaType.image].contains(e.type)) | ||||
|                               .map((e) => PostWriteMedia.fromFile(XFile(e.path))) | ||||
|                               .toList(), | ||||
|                         ), | ||||
|                       ) | ||||
|                           .then((value) { | ||||
|                         if (value == true) _refreshChannels(); | ||||
|                       }); | ||||
|                     }, | ||||
|                   ); | ||||
|                 }, | ||||
|               ), | ||||
|             ), | ||||
|           ), | ||||
|         ), | ||||
|       ], | ||||
|     ); | ||||
|   } | ||||
| } | ||||
|   | ||||
							
								
								
									
										279
									
								
								lib/screens/wallet.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,279 @@ | ||||
| import 'package:easy_localization/easy_localization.dart'; | ||||
| import 'package:flutter/material.dart'; | ||||
| import 'package:gap/gap.dart'; | ||||
| import 'package:material_symbols_icons/material_symbols_icons.dart'; | ||||
| import 'package:provider/provider.dart'; | ||||
| import 'package:styled_widget/styled_widget.dart'; | ||||
| import 'package:surface/providers/sn_network.dart'; | ||||
| import 'package:surface/types/wallet.dart'; | ||||
| import 'package:surface/widgets/dialog.dart'; | ||||
| import 'package:surface/widgets/loading_indicator.dart'; | ||||
| import 'package:surface/widgets/navigation/app_scaffold.dart'; | ||||
| import 'package:very_good_infinite_list/very_good_infinite_list.dart'; | ||||
|  | ||||
| class WalletScreen extends StatefulWidget { | ||||
|   const WalletScreen({super.key}); | ||||
|  | ||||
|   @override | ||||
|   State<WalletScreen> createState() => _WalletScreenState(); | ||||
| } | ||||
|  | ||||
| class _WalletScreenState extends State<WalletScreen> { | ||||
|   bool _isBusy = false; | ||||
|   SnWallet? _wallet; | ||||
|  | ||||
|   Future<void> _fetchWallet() async { | ||||
|     try { | ||||
|       setState(() => _isBusy = true); | ||||
|       final sn = context.read<SnNetworkProvider>(); | ||||
|       final resp = await sn.client.get('/cgi/wa/wallets/me'); | ||||
|       _wallet = SnWallet.fromJson(resp.data); | ||||
|     } catch (err) { | ||||
|       if (!mounted) return; | ||||
|       context.showErrorDialog(err); | ||||
|     } finally { | ||||
|       setState(() => _isBusy = false); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   void initState() { | ||||
|     super.initState(); | ||||
|     _fetchWallet(); | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     return AppScaffold( | ||||
|       appBar: AppBar( | ||||
|         leading: PageBackButton(), | ||||
|         title: Text('screenAccountWallet').tr(), | ||||
|       ), | ||||
|       body: Column( | ||||
|         children: [ | ||||
|           LoadingIndicator(isActive: _isBusy), | ||||
|           if (_wallet == null) | ||||
|             Expanded( | ||||
|               child: _CreateWalletWidget( | ||||
|                 onCreate: () { | ||||
|                   _fetchWallet(); | ||||
|                 }, | ||||
|               ), | ||||
|             ) | ||||
|           else | ||||
|             Card( | ||||
|               child: Column( | ||||
|                 mainAxisSize: MainAxisSize.min, | ||||
|                 crossAxisAlignment: CrossAxisAlignment.start, | ||||
|                 children: [ | ||||
|                   CircleAvatar( | ||||
|                     radius: 28, | ||||
|                     child: Icon(Symbols.wallet, size: 28), | ||||
|                   ), | ||||
|                   const Gap(12), | ||||
|                   SizedBox(width: double.infinity), | ||||
|                   Text( | ||||
|                     NumberFormat.compactCurrency( | ||||
|                       locale: EasyLocalization.of(context)!.currentLocale.toString(), | ||||
|                       symbol: '${'walletCurrencyShort'.tr()} ', | ||||
|                       decimalDigits: 2, | ||||
|                     ).format(double.parse(_wallet!.balance)), | ||||
|                     style: Theme.of(context).textTheme.titleLarge, | ||||
|                   ), | ||||
|                   Text('walletCurrency'.plural(double.parse(_wallet!.balance))), | ||||
|                 ], | ||||
|               ).padding(horizontal: 20, vertical: 24), | ||||
|             ).padding(horizontal: 8, top: 16, bottom: 4), | ||||
|           if (_wallet != null) Expanded(child: _WalletTransactionList(myself: _wallet!)), | ||||
|         ], | ||||
|       ), | ||||
|     ); | ||||
|   } | ||||
| } | ||||
|  | ||||
| class _WalletTransactionList extends StatefulWidget { | ||||
|   final SnWallet myself; | ||||
|  | ||||
|   const _WalletTransactionList({required this.myself}); | ||||
|  | ||||
|   @override | ||||
|   State<_WalletTransactionList> createState() => _WalletTransactionListState(); | ||||
| } | ||||
|  | ||||
| class _WalletTransactionListState extends State<_WalletTransactionList> { | ||||
|   bool _isBusy = false; | ||||
|   int? _totalCount; | ||||
|   final List<SnTransaction> _transactions = List.empty(growable: true); | ||||
|  | ||||
|   Future<void> _fetchTransactions() async { | ||||
|     try { | ||||
|       setState(() => _isBusy = true); | ||||
|       final sn = context.read<SnNetworkProvider>(); | ||||
|       final resp = await sn.client.get('/cgi/wa/transactions/me', queryParameters: { | ||||
|         'take': 10, | ||||
|         'offset': _transactions.length, | ||||
|       }); | ||||
|       _totalCount = resp.data['count']; | ||||
|       _transactions.addAll( | ||||
|         resp.data['data']?.map((e) => SnTransaction.fromJson(e)).cast<SnTransaction>() ?? [], | ||||
|       ); | ||||
|     } catch (err) { | ||||
|       if (!mounted) return; | ||||
|       context.showErrorDialog(err); | ||||
|     } finally { | ||||
|       setState(() => _isBusy = false); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   void initState() { | ||||
|     super.initState(); | ||||
|     _fetchTransactions(); | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     return MediaQuery.removePadding( | ||||
|       context: context, | ||||
|       removeTop: true, | ||||
|       child: RefreshIndicator( | ||||
|         onRefresh: _fetchTransactions, | ||||
|         child: InfiniteList( | ||||
|           itemCount: _transactions.length, | ||||
|           isLoading: _isBusy, | ||||
|           hasReachedMax: _totalCount != null && _transactions.length >= _totalCount!, | ||||
|           onFetchData: () { | ||||
|             _fetchTransactions(); | ||||
|           }, | ||||
|           itemBuilder: (context, idx) { | ||||
|             final ele = _transactions[idx]; | ||||
|             final isIncoming = ele.payeeId == widget.myself.id; | ||||
|             return ListTile( | ||||
|               leading: isIncoming ? const Icon(Symbols.call_received) : const Icon(Symbols.call_made), | ||||
|               title: Text( | ||||
|                 '${isIncoming ? '+' : '-'}${ele.amount} ${'walletCurrencyShort'.tr()}', | ||||
|                 style: TextStyle(color: isIncoming ? Colors.green : Colors.red), | ||||
|               ), | ||||
|               subtitle: Column( | ||||
|                 crossAxisAlignment: CrossAxisAlignment.start, | ||||
|                 children: [ | ||||
|                   Text(ele.remark), | ||||
|                   const Gap(2), | ||||
|                   Text( | ||||
|                     DateFormat( | ||||
|                       null, | ||||
|                       EasyLocalization.of(context)!.currentLocale.toString(), | ||||
|                     ).format(ele.createdAt), | ||||
|                     style: Theme.of(context).textTheme.labelSmall, | ||||
|                   ), | ||||
|                 ], | ||||
|               ), | ||||
|               contentPadding: const EdgeInsets.symmetric(horizontal: 24), | ||||
|             ); | ||||
|           }, | ||||
|         ), | ||||
|       ), | ||||
|     ); | ||||
|   } | ||||
| } | ||||
|  | ||||
| class _CreateWalletWidget extends StatefulWidget { | ||||
|   final Function()? onCreate; | ||||
|  | ||||
|   const _CreateWalletWidget({required this.onCreate}); | ||||
|  | ||||
|   @override | ||||
|   State<_CreateWalletWidget> createState() => _CreateWalletWidgetState(); | ||||
| } | ||||
|  | ||||
| class _CreateWalletWidgetState extends State<_CreateWalletWidget> { | ||||
|   bool _isBusy = false; | ||||
|  | ||||
|   Future<void> _createWallet() async { | ||||
|     final TextEditingController passwordController = TextEditingController(); | ||||
|     final password = await showDialog<String?>( | ||||
|       context: context, | ||||
|       builder: (ctx) => AlertDialog( | ||||
|         title: Text('walletCreate').tr(), | ||||
|         content: Column( | ||||
|           crossAxisAlignment: CrossAxisAlignment.start, | ||||
|           mainAxisSize: MainAxisSize.min, | ||||
|           children: [ | ||||
|             Text('walletCreatePassword').tr(), | ||||
|             const Gap(8), | ||||
|             TextField( | ||||
|               autofocus: true, | ||||
|               obscureText: true, | ||||
|               controller: passwordController, | ||||
|               decoration: InputDecoration( | ||||
|                 labelText: 'fieldPassword'.tr(), | ||||
|               ), | ||||
|             ), | ||||
|           ], | ||||
|         ), | ||||
|         actions: [ | ||||
|           TextButton( | ||||
|             onPressed: () => Navigator.of(ctx).pop(), | ||||
|             child: Text('cancel').tr(), | ||||
|           ), | ||||
|           TextButton( | ||||
|             onPressed: () { | ||||
|               Navigator.of(ctx).pop(passwordController.text); | ||||
|             }, | ||||
|             child: Text('next').tr(), | ||||
|           ), | ||||
|         ], | ||||
|       ), | ||||
|     ); | ||||
|     WidgetsBinding.instance.addPostFrameCallback((_) { | ||||
|       passwordController.dispose(); | ||||
|     }); | ||||
|     if (password == null || password.isEmpty) return; | ||||
|     if (!mounted) return; | ||||
|  | ||||
|     try { | ||||
|       setState(() => _isBusy = true); | ||||
|       final sn = context.read<SnNetworkProvider>(); | ||||
|       await sn.client.post('/cgi/wa/wallets/me', data: { | ||||
|         'password': password, | ||||
|       }); | ||||
|     } catch (err) { | ||||
|       if (!mounted) return; | ||||
|       context.showErrorDialog(err); | ||||
|     } finally { | ||||
|       setState(() => _isBusy = false); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     return Center( | ||||
|       child: Container( | ||||
|         constraints: const BoxConstraints(maxWidth: 380), | ||||
|         child: Card( | ||||
|           child: Column( | ||||
|             mainAxisSize: MainAxisSize.min, | ||||
|             crossAxisAlignment: CrossAxisAlignment.start, | ||||
|             children: [ | ||||
|               CircleAvatar( | ||||
|                 radius: 28, | ||||
|                 child: Icon(Symbols.add, size: 28), | ||||
|               ), | ||||
|               const Gap(12), | ||||
|               Text('walletCreate', style: Theme.of(context).textTheme.titleLarge).tr(), | ||||
|               Text('walletCreateSubtitle', style: Theme.of(context).textTheme.bodyMedium).tr(), | ||||
|               const Gap(8), | ||||
|               Align( | ||||
|                 alignment: Alignment.centerRight, | ||||
|                 child: TextButton( | ||||
|                   onPressed: _isBusy ? null : () => _createWallet(), | ||||
|                   child: Text('next').tr(), | ||||
|                 ), | ||||
|               ), | ||||
|             ], | ||||
|           ).padding(horizontal: 20, vertical: 24), | ||||
|         ), | ||||
|       ), | ||||
|     ); | ||||
|   } | ||||
| } | ||||
| @@ -1,5 +1,6 @@ | ||||
| import 'package:flutter/material.dart'; | ||||
| import 'package:shared_preferences/shared_preferences.dart'; | ||||
| import 'package:surface/providers/config.dart'; | ||||
|  | ||||
| const kMaterialYouToggleStoreKey = 'app_theme_material_you'; | ||||
|  | ||||
| @@ -10,7 +11,7 @@ class ThemeSet { | ||||
|   ThemeSet({required this.light, required this.dark}); | ||||
| } | ||||
|  | ||||
| Future<ThemeSet> createAppThemeSet({bool? useMaterial3}) async { | ||||
| Future<ThemeSet> createAppThemeSet({Color? seedColorOverride, bool? useMaterial3}) async { | ||||
|   return ThemeSet( | ||||
|     light: await createAppTheme(Brightness.light, useMaterial3: useMaterial3), | ||||
|     dark: await createAppTheme(Brightness.dark, useMaterial3: useMaterial3), | ||||
| @@ -19,19 +20,24 @@ Future<ThemeSet> createAppThemeSet({bool? useMaterial3}) async { | ||||
|  | ||||
| Future<ThemeData> createAppTheme( | ||||
|   Brightness brightness, { | ||||
|   Color? seedColorOverride, | ||||
|   bool? useMaterial3, | ||||
| }) async { | ||||
|   final prefs = await SharedPreferences.getInstance(); | ||||
|  | ||||
|   final seedColorString = prefs.getInt(kAppColorSchemeStoreKey); | ||||
|   final seedColor = seedColorString != null ? Color(seedColorString) : Colors.indigo; | ||||
|  | ||||
|   final colorScheme = ColorScheme.fromSeed( | ||||
|     seedColor: Colors.indigo, | ||||
|     seedColor: seedColorOverride ?? seedColor, | ||||
|     brightness: brightness, | ||||
|   ); | ||||
|  | ||||
|   final hasBackground = prefs.getBool('has_background_image') ?? false; | ||||
|   final hasAppBarTransparent = prefs.getBool(kAppbarTransparentStoreKey) ?? false; | ||||
|   final useM3 = useMaterial3 ?? (prefs.getBool(kMaterialYouToggleStoreKey) ?? true); | ||||
|  | ||||
|   return ThemeData( | ||||
|     useMaterial3: useMaterial3 ?? (prefs.getBool(kMaterialYouToggleStoreKey) ?? false), | ||||
|     useMaterial3: useM3, | ||||
|     colorScheme: colorScheme, | ||||
|     brightness: brightness, | ||||
|     iconTheme: IconThemeData( | ||||
| @@ -40,11 +46,24 @@ Future<ThemeData> createAppTheme( | ||||
|       opticalSize: 20, | ||||
|       color: colorScheme.onSurface, | ||||
|     ), | ||||
|     snackBarTheme: SnackBarThemeData( | ||||
|       behavior: useM3 ? SnackBarBehavior.floating : SnackBarBehavior.fixed, | ||||
|     ), | ||||
|     appBarTheme: AppBarTheme( | ||||
|       centerTitle: true, | ||||
|       backgroundColor: hasBackground ? colorScheme.primary.withOpacity(0.75) : colorScheme.primary, | ||||
|       foregroundColor: colorScheme.onPrimary, | ||||
|       elevation: hasAppBarTransparent ? 0 : null, | ||||
|       backgroundColor: hasAppBarTransparent ? Colors.transparent : colorScheme.primary, | ||||
|       foregroundColor: hasAppBarTransparent ? colorScheme.onSurface : colorScheme.onPrimary, | ||||
|     ), | ||||
|     pageTransitionsTheme: PageTransitionsTheme( | ||||
|       builders: { | ||||
|         TargetPlatform.android: PredictiveBackPageTransitionsBuilder(), | ||||
|         TargetPlatform.iOS: CupertinoPageTransitionsBuilder(), | ||||
|         TargetPlatform.macOS: ZoomPageTransitionsBuilder(), | ||||
|         TargetPlatform.fuchsia: ZoomPageTransitionsBuilder(), | ||||
|         TargetPlatform.linux: ZoomPageTransitionsBuilder(), | ||||
|         TargetPlatform.windows: ZoomPageTransitionsBuilder(), | ||||
|       }, | ||||
|     ), | ||||
|     scaffoldBackgroundColor: Colors.transparent, | ||||
|   ); | ||||
| } | ||||
|   | ||||
| @@ -15,12 +15,13 @@ class SnAccount with _$SnAccount { | ||||
|     required DateTime? deletedAt, | ||||
|     required DateTime? confirmedAt, | ||||
|     required List<SnAccountContact>? contacts, | ||||
|     required String avatar, | ||||
|     required String banner, | ||||
|     @Default("") String avatar, | ||||
|     @Default("") String banner, | ||||
|     required String description, | ||||
|     required String name, | ||||
|     required String nick, | ||||
|     required Map<String, dynamic> permNodes, | ||||
|     required String language, | ||||
|     required SnAccountProfile? profile, | ||||
|     @Default([]) List<SnAccountBadge> badges, | ||||
|     required DateTime? suspendedAt, | ||||
|   | ||||
| @@ -33,6 +33,7 @@ mixin _$SnAccount { | ||||
|   String get name => throw _privateConstructorUsedError; | ||||
|   String get nick => throw _privateConstructorUsedError; | ||||
|   Map<String, dynamic> get permNodes => throw _privateConstructorUsedError; | ||||
|   String get language => throw _privateConstructorUsedError; | ||||
|   SnAccountProfile? get profile => throw _privateConstructorUsedError; | ||||
|   List<SnAccountBadge> get badges => throw _privateConstructorUsedError; | ||||
|   DateTime? get suspendedAt => throw _privateConstructorUsedError; | ||||
| @@ -69,6 +70,7 @@ abstract class $SnAccountCopyWith<$Res> { | ||||
|       String name, | ||||
|       String nick, | ||||
|       Map<String, dynamic> permNodes, | ||||
|       String language, | ||||
|       SnAccountProfile? profile, | ||||
|       List<SnAccountBadge> badges, | ||||
|       DateTime? suspendedAt, | ||||
| @@ -107,6 +109,7 @@ class _$SnAccountCopyWithImpl<$Res, $Val extends SnAccount> | ||||
|     Object? name = null, | ||||
|     Object? nick = null, | ||||
|     Object? permNodes = null, | ||||
|     Object? language = null, | ||||
|     Object? profile = freezed, | ||||
|     Object? badges = null, | ||||
|     Object? suspendedAt = freezed, | ||||
| @@ -164,6 +167,10 @@ class _$SnAccountCopyWithImpl<$Res, $Val extends SnAccount> | ||||
|           ? _value.permNodes | ||||
|           : permNodes // ignore: cast_nullable_to_non_nullable | ||||
|               as Map<String, dynamic>, | ||||
|       language: null == language | ||||
|           ? _value.language | ||||
|           : language // ignore: cast_nullable_to_non_nullable | ||||
|               as String, | ||||
|       profile: freezed == profile | ||||
|           ? _value.profile | ||||
|           : profile // ignore: cast_nullable_to_non_nullable | ||||
| @@ -231,6 +238,7 @@ abstract class _$$SnAccountImplCopyWith<$Res> | ||||
|       String name, | ||||
|       String nick, | ||||
|       Map<String, dynamic> permNodes, | ||||
|       String language, | ||||
|       SnAccountProfile? profile, | ||||
|       List<SnAccountBadge> badges, | ||||
|       DateTime? suspendedAt, | ||||
| @@ -268,6 +276,7 @@ class __$$SnAccountImplCopyWithImpl<$Res> | ||||
|     Object? name = null, | ||||
|     Object? nick = null, | ||||
|     Object? permNodes = null, | ||||
|     Object? language = null, | ||||
|     Object? profile = freezed, | ||||
|     Object? badges = null, | ||||
|     Object? suspendedAt = freezed, | ||||
| @@ -325,6 +334,10 @@ class __$$SnAccountImplCopyWithImpl<$Res> | ||||
|           ? _value._permNodes | ||||
|           : permNodes // ignore: cast_nullable_to_non_nullable | ||||
|               as Map<String, dynamic>, | ||||
|       language: null == language | ||||
|           ? _value.language | ||||
|           : language // ignore: cast_nullable_to_non_nullable | ||||
|               as String, | ||||
|       profile: freezed == profile | ||||
|           ? _value.profile | ||||
|           : profile // ignore: cast_nullable_to_non_nullable | ||||
| @@ -367,12 +380,13 @@ class _$SnAccountImpl extends _SnAccount { | ||||
|       required this.deletedAt, | ||||
|       required this.confirmedAt, | ||||
|       required final List<SnAccountContact>? contacts, | ||||
|       required this.avatar, | ||||
|       required this.banner, | ||||
|       this.avatar = "", | ||||
|       this.banner = "", | ||||
|       required this.description, | ||||
|       required this.name, | ||||
|       required this.nick, | ||||
|       required final Map<String, dynamic> permNodes, | ||||
|       required this.language, | ||||
|       required this.profile, | ||||
|       final List<SnAccountBadge> badges = const [], | ||||
|       required this.suspendedAt, | ||||
| @@ -410,8 +424,10 @@ class _$SnAccountImpl extends _SnAccount { | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   @JsonKey() | ||||
|   final String avatar; | ||||
|   @override | ||||
|   @JsonKey() | ||||
|   final String banner; | ||||
|   @override | ||||
|   final String description; | ||||
| @@ -427,6 +443,8 @@ class _$SnAccountImpl extends _SnAccount { | ||||
|     return EqualUnmodifiableMapView(_permNodes); | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   final String language; | ||||
|   @override | ||||
|   final SnAccountProfile? profile; | ||||
|   final List<SnAccountBadge> _badges; | ||||
| @@ -451,7 +469,7 @@ class _$SnAccountImpl extends _SnAccount { | ||||
|  | ||||
|   @override | ||||
|   String toString() { | ||||
|     return 'SnAccount(id: $id, createdAt: $createdAt, updatedAt: $updatedAt, deletedAt: $deletedAt, confirmedAt: $confirmedAt, contacts: $contacts, avatar: $avatar, banner: $banner, description: $description, name: $name, nick: $nick, permNodes: $permNodes, profile: $profile, badges: $badges, suspendedAt: $suspendedAt, affiliatedId: $affiliatedId, affiliatedTo: $affiliatedTo, automatedBy: $automatedBy, automatedId: $automatedId)'; | ||||
|     return 'SnAccount(id: $id, createdAt: $createdAt, updatedAt: $updatedAt, deletedAt: $deletedAt, confirmedAt: $confirmedAt, contacts: $contacts, avatar: $avatar, banner: $banner, description: $description, name: $name, nick: $nick, permNodes: $permNodes, language: $language, profile: $profile, badges: $badges, suspendedAt: $suspendedAt, affiliatedId: $affiliatedId, affiliatedTo: $affiliatedTo, automatedBy: $automatedBy, automatedId: $automatedId)'; | ||||
|   } | ||||
|  | ||||
|   @override | ||||
| @@ -477,6 +495,8 @@ class _$SnAccountImpl extends _SnAccount { | ||||
|             (identical(other.nick, nick) || other.nick == nick) && | ||||
|             const DeepCollectionEquality() | ||||
|                 .equals(other._permNodes, _permNodes) && | ||||
|             (identical(other.language, language) || | ||||
|                 other.language == language) && | ||||
|             (identical(other.profile, profile) || other.profile == profile) && | ||||
|             const DeepCollectionEquality().equals(other._badges, _badges) && | ||||
|             (identical(other.suspendedAt, suspendedAt) || | ||||
| @@ -507,6 +527,7 @@ class _$SnAccountImpl extends _SnAccount { | ||||
|         name, | ||||
|         nick, | ||||
|         const DeepCollectionEquality().hash(_permNodes), | ||||
|         language, | ||||
|         profile, | ||||
|         const DeepCollectionEquality().hash(_badges), | ||||
|         suspendedAt, | ||||
| @@ -540,12 +561,13 @@ abstract class _SnAccount extends SnAccount { | ||||
|       required final DateTime? deletedAt, | ||||
|       required final DateTime? confirmedAt, | ||||
|       required final List<SnAccountContact>? contacts, | ||||
|       required final String avatar, | ||||
|       required final String banner, | ||||
|       final String avatar, | ||||
|       final String banner, | ||||
|       required final String description, | ||||
|       required final String name, | ||||
|       required final String nick, | ||||
|       required final Map<String, dynamic> permNodes, | ||||
|       required final String language, | ||||
|       required final SnAccountProfile? profile, | ||||
|       final List<SnAccountBadge> badges, | ||||
|       required final DateTime? suspendedAt, | ||||
| @@ -584,6 +606,8 @@ abstract class _SnAccount extends SnAccount { | ||||
|   @override | ||||
|   Map<String, dynamic> get permNodes; | ||||
|   @override | ||||
|   String get language; | ||||
|   @override | ||||
|   SnAccountProfile? get profile; | ||||
|   @override | ||||
|   List<SnAccountBadge> get badges; | ||||
|   | ||||
| @@ -20,12 +20,13 @@ _$SnAccountImpl _$$SnAccountImplFromJson(Map<String, dynamic> json) => | ||||
|       contacts: (json['contacts'] as List<dynamic>?) | ||||
|           ?.map((e) => SnAccountContact.fromJson(e as Map<String, dynamic>)) | ||||
|           .toList(), | ||||
|       avatar: json['avatar'] as String, | ||||
|       banner: json['banner'] as String, | ||||
|       avatar: json['avatar'] as String? ?? "", | ||||
|       banner: json['banner'] as String? ?? "", | ||||
|       description: json['description'] as String, | ||||
|       name: json['name'] as String, | ||||
|       nick: json['nick'] as String, | ||||
|       permNodes: json['perm_nodes'] as Map<String, dynamic>, | ||||
|       language: json['language'] as String, | ||||
|       profile: json['profile'] == null | ||||
|           ? null | ||||
|           : SnAccountProfile.fromJson(json['profile'] as Map<String, dynamic>), | ||||
| @@ -56,6 +57,7 @@ Map<String, dynamic> _$$SnAccountImplToJson(_$SnAccountImpl instance) => | ||||
|       'name': instance.name, | ||||
|       'nick': instance.nick, | ||||
|       'perm_nodes': instance.permNodes, | ||||
|       'language': instance.language, | ||||
|       'profile': instance.profile?.toJson(), | ||||
|       'badges': instance.badges.map((e) => e.toJson()).toList(), | ||||
|       'suspended_at': instance.suspendedAt?.toIso8601String(), | ||||
|   | ||||
| @@ -1,15 +1,25 @@ | ||||
| import 'package:freezed_annotation/freezed_annotation.dart'; | ||||
|  | ||||
| part 'attachment.freezed.dart'; | ||||
|  | ||||
| part 'attachment.g.dart'; | ||||
|  | ||||
| enum SnMediaType { | ||||
|   image, | ||||
|   video, | ||||
|   audio, | ||||
|   file, | ||||
| } | ||||
|  | ||||
| @freezed | ||||
| class SnAttachment with _$SnAttachment { | ||||
|   const SnAttachment._(); | ||||
|  | ||||
|   const factory SnAttachment({ | ||||
|     required int id, | ||||
|     required DateTime createdAt, | ||||
|     required DateTime updatedAt, | ||||
|     required dynamic deletedAt, | ||||
|     required DateTime? deletedAt, | ||||
|     required String rid, | ||||
|     required String uuid, | ||||
|     required int size, | ||||
| @@ -19,22 +29,70 @@ class SnAttachment with _$SnAttachment { | ||||
|     required String hash, | ||||
|     required int destination, | ||||
|     required int refCount, | ||||
|     required dynamic fileChunks, | ||||
|     required dynamic cleanedAt, | ||||
|     required bool isMature, | ||||
|     @Default(0) int contentRating, | ||||
|     @Default(0) int qualityRating, | ||||
|     required DateTime? cleanedAt, | ||||
|     required bool isAnalyzed, | ||||
|     required bool isUploaded, | ||||
|     required bool isSelfRef, | ||||
|     required dynamic ref, | ||||
|     required dynamic refId, | ||||
|     required bool isIndexable, | ||||
|     required SnAttachment? ref, | ||||
|     required int? refId, | ||||
|     required SnAttachmentPool? pool, | ||||
|     required int poolId, | ||||
|     required int? poolId, | ||||
|     required int accountId, | ||||
|     int? thumbnailId, | ||||
|     SnAttachment? thumbnail, | ||||
|     int? compressedId, | ||||
|     SnAttachment? compressed, | ||||
|     @Default([]) List<SnAttachmentBoost> boosts, | ||||
|     @Default({}) Map<String, dynamic> usermeta, | ||||
|     @Default({}) Map<String, dynamic> metadata, | ||||
|   }) = _SnAttachment; | ||||
|  | ||||
|   factory SnAttachment.fromJson(Map<String, Object?> json) => | ||||
|       _$SnAttachmentFromJson(json); | ||||
|   factory SnAttachment.fromJson(Map<String, Object?> json) => _$SnAttachmentFromJson(json); | ||||
|  | ||||
|   Map<String, dynamic> get data => { | ||||
|         ...metadata, | ||||
|         ...usermeta, | ||||
|       }; | ||||
|  | ||||
|   SnMediaType get mediaType => switch (mimetype.split('/').firstOrNull) { | ||||
|         'image' => SnMediaType.image, | ||||
|         'video' => SnMediaType.video, | ||||
|         'audio' => SnMediaType.audio, | ||||
|         _ => SnMediaType.file, | ||||
|       }; | ||||
| } | ||||
|  | ||||
| @freezed | ||||
| class SnAttachmentFragment with _$SnAttachmentFragment { | ||||
|   const SnAttachmentFragment._(); | ||||
|  | ||||
|   const factory SnAttachmentFragment({ | ||||
|     required int id, | ||||
|     required DateTime createdAt, | ||||
|     required DateTime updatedAt, | ||||
|     required DateTime? deletedAt, | ||||
|     required String rid, | ||||
|     required String uuid, | ||||
|     required int size, | ||||
|     required String name, | ||||
|     required String alt, | ||||
|     required String mimetype, | ||||
|     required String hash, | ||||
|     String? fingerprint, | ||||
|     @Default({}) Map<String, int> fileChunks, | ||||
|     @Default([]) List<String> fileChunksMissing, | ||||
|   }) = _SnAttachmentFragment; | ||||
|  | ||||
|   factory SnAttachmentFragment.fromJson(Map<String, Object?> json) => _$SnAttachmentFragmentFromJson(json); | ||||
|  | ||||
|   SnMediaType get mediaType => switch (mimetype.split('/').firstOrNull) { | ||||
|         'image' => SnMediaType.image, | ||||
|         'video' => SnMediaType.video, | ||||
|         'audio' => SnMediaType.audio, | ||||
|         _ => SnMediaType.file, | ||||
|       }; | ||||
| } | ||||
|  | ||||
| @freezed | ||||
| @@ -51,6 +109,71 @@ class SnAttachmentPool with _$SnAttachmentPool { | ||||
|     required int? accountId, | ||||
|   }) = _SnAttachmentPool; | ||||
|  | ||||
|   factory SnAttachmentPool.fromJson(Map<String, Object?> json) => | ||||
|       _$SnAttachmentPoolFromJson(json); | ||||
|   factory SnAttachmentPool.fromJson(Map<String, Object?> json) => _$SnAttachmentPoolFromJson(json); | ||||
| } | ||||
|  | ||||
| @freezed | ||||
| class SnAttachmentDestination with _$SnAttachmentDestination { | ||||
|   const factory SnAttachmentDestination({ | ||||
|     @Default(0) int id, | ||||
|     required String type, | ||||
|     required String label, | ||||
|     required String region, | ||||
|     required bool isBoost, | ||||
|   }) = _SnAttachmentDestination; | ||||
|  | ||||
|   factory SnAttachmentDestination.fromJson(Map<String, Object?> json) => _$SnAttachmentDestinationFromJson(json); | ||||
| } | ||||
|  | ||||
| @freezed | ||||
| class SnAttachmentBoost with _$SnAttachmentBoost { | ||||
|   const factory SnAttachmentBoost({ | ||||
|     required int id, | ||||
|     required DateTime createdAt, | ||||
|     required DateTime updatedAt, | ||||
|     required DateTime? deletedAt, | ||||
|     required int status, | ||||
|     required int destination, | ||||
|     required int attachmentId, | ||||
|     required SnAttachment attachment, | ||||
|     required int account, | ||||
|   }) = _SnAttachmentBoost; | ||||
|  | ||||
|   factory SnAttachmentBoost.fromJson(Map<String, Object?> json) => _$SnAttachmentBoostFromJson(json); | ||||
| } | ||||
|  | ||||
| @freezed | ||||
| class SnSticker with _$SnSticker { | ||||
|   const factory SnSticker({ | ||||
|     required int id, | ||||
|     required DateTime createdAt, | ||||
|     required DateTime updatedAt, | ||||
|     required DateTime? deletedAt, | ||||
|     required String alias, | ||||
|     required String name, | ||||
|     required int attachmentId, | ||||
|     required SnAttachment attachment, | ||||
|     required int packId, | ||||
|     required SnStickerPack pack, | ||||
|     required int accountId, | ||||
|   }) = _SnSticker; | ||||
|  | ||||
|   factory SnSticker.fromJson(Map<String, Object?> json) => _$SnStickerFromJson(json); | ||||
| } | ||||
|  | ||||
| @freezed | ||||
| class SnStickerPack with _$SnStickerPack { | ||||
|   const factory SnStickerPack({ | ||||
|     required int id, | ||||
|     required DateTime createdAt, | ||||
|     required DateTime updatedAt, | ||||
|     required DateTime? deletedAt, | ||||
|     required String prefix, | ||||
|     required String name, | ||||
|     required String description, | ||||
|     required List<SnSticker>? stickers, | ||||
|     required int accountId, | ||||
|   }) = _SnStickerPack; | ||||
|  | ||||
|   factory SnStickerPack.fromJson(Map<String, Object?> json) => _$SnStickerPackFromJson(json); | ||||
| } | ||||
|   | ||||
| @@ -11,7 +11,9 @@ _$SnAttachmentImpl _$$SnAttachmentImplFromJson(Map<String, dynamic> json) => | ||||
|       id: (json['id'] as num).toInt(), | ||||
|       createdAt: DateTime.parse(json['created_at'] as String), | ||||
|       updatedAt: DateTime.parse(json['updated_at'] as String), | ||||
|       deletedAt: json['deleted_at'], | ||||
|       deletedAt: json['deleted_at'] == null | ||||
|           ? null | ||||
|           : DateTime.parse(json['deleted_at'] as String), | ||||
|       rid: json['rid'] as String, | ||||
|       uuid: json['uuid'] as String, | ||||
|       size: (json['size'] as num).toInt(), | ||||
| @@ -21,19 +23,37 @@ _$SnAttachmentImpl _$$SnAttachmentImplFromJson(Map<String, dynamic> json) => | ||||
|       hash: json['hash'] as String, | ||||
|       destination: (json['destination'] as num).toInt(), | ||||
|       refCount: (json['ref_count'] as num).toInt(), | ||||
|       fileChunks: json['file_chunks'], | ||||
|       cleanedAt: json['cleaned_at'], | ||||
|       isMature: json['is_mature'] as bool, | ||||
|       contentRating: (json['content_rating'] as num?)?.toInt() ?? 0, | ||||
|       qualityRating: (json['quality_rating'] as num?)?.toInt() ?? 0, | ||||
|       cleanedAt: json['cleaned_at'] == null | ||||
|           ? null | ||||
|           : DateTime.parse(json['cleaned_at'] as String), | ||||
|       isAnalyzed: json['is_analyzed'] as bool, | ||||
|       isUploaded: json['is_uploaded'] as bool, | ||||
|       isSelfRef: json['is_self_ref'] as bool, | ||||
|       ref: json['ref'], | ||||
|       refId: json['ref_id'], | ||||
|       isIndexable: json['is_indexable'] as bool, | ||||
|       ref: json['ref'] == null | ||||
|           ? null | ||||
|           : SnAttachment.fromJson(json['ref'] as Map<String, dynamic>), | ||||
|       refId: (json['ref_id'] as num?)?.toInt(), | ||||
|       pool: json['pool'] == null | ||||
|           ? null | ||||
|           : SnAttachmentPool.fromJson(json['pool'] as Map<String, dynamic>), | ||||
|       poolId: (json['pool_id'] as num).toInt(), | ||||
|       poolId: (json['pool_id'] as num?)?.toInt(), | ||||
|       accountId: (json['account_id'] as num).toInt(), | ||||
|       thumbnailId: (json['thumbnail_id'] as num?)?.toInt(), | ||||
|       thumbnail: json['thumbnail'] == null | ||||
|           ? null | ||||
|           : SnAttachment.fromJson(json['thumbnail'] as Map<String, dynamic>), | ||||
|       compressedId: (json['compressed_id'] as num?)?.toInt(), | ||||
|       compressed: json['compressed'] == null | ||||
|           ? null | ||||
|           : SnAttachment.fromJson(json['compressed'] as Map<String, dynamic>), | ||||
|       boosts: (json['boosts'] as List<dynamic>?) | ||||
|               ?.map( | ||||
|                   (e) => SnAttachmentBoost.fromJson(e as Map<String, dynamic>)) | ||||
|               .toList() ?? | ||||
|           const [], | ||||
|       usermeta: json['usermeta'] as Map<String, dynamic>? ?? const {}, | ||||
|       metadata: json['metadata'] as Map<String, dynamic>? ?? const {}, | ||||
|     ); | ||||
|  | ||||
| @@ -42,7 +62,7 @@ Map<String, dynamic> _$$SnAttachmentImplToJson(_$SnAttachmentImpl instance) => | ||||
|       'id': instance.id, | ||||
|       'created_at': instance.createdAt.toIso8601String(), | ||||
|       'updated_at': instance.updatedAt.toIso8601String(), | ||||
|       'deleted_at': instance.deletedAt, | ||||
|       'deleted_at': instance.deletedAt?.toIso8601String(), | ||||
|       'rid': instance.rid, | ||||
|       'uuid': instance.uuid, | ||||
|       'size': instance.size, | ||||
| @@ -52,20 +72,72 @@ Map<String, dynamic> _$$SnAttachmentImplToJson(_$SnAttachmentImpl instance) => | ||||
|       'hash': instance.hash, | ||||
|       'destination': instance.destination, | ||||
|       'ref_count': instance.refCount, | ||||
|       'file_chunks': instance.fileChunks, | ||||
|       'cleaned_at': instance.cleanedAt, | ||||
|       'is_mature': instance.isMature, | ||||
|       'content_rating': instance.contentRating, | ||||
|       'quality_rating': instance.qualityRating, | ||||
|       'cleaned_at': instance.cleanedAt?.toIso8601String(), | ||||
|       'is_analyzed': instance.isAnalyzed, | ||||
|       'is_uploaded': instance.isUploaded, | ||||
|       'is_self_ref': instance.isSelfRef, | ||||
|       'ref': instance.ref, | ||||
|       'is_indexable': instance.isIndexable, | ||||
|       'ref': instance.ref?.toJson(), | ||||
|       'ref_id': instance.refId, | ||||
|       'pool': instance.pool?.toJson(), | ||||
|       'pool_id': instance.poolId, | ||||
|       'account_id': instance.accountId, | ||||
|       'thumbnail_id': instance.thumbnailId, | ||||
|       'thumbnail': instance.thumbnail?.toJson(), | ||||
|       'compressed_id': instance.compressedId, | ||||
|       'compressed': instance.compressed?.toJson(), | ||||
|       'boosts': instance.boosts.map((e) => e.toJson()).toList(), | ||||
|       'usermeta': instance.usermeta, | ||||
|       'metadata': instance.metadata, | ||||
|     }; | ||||
|  | ||||
| _$SnAttachmentFragmentImpl _$$SnAttachmentFragmentImplFromJson( | ||||
|         Map<String, dynamic> json) => | ||||
|     _$SnAttachmentFragmentImpl( | ||||
|       id: (json['id'] as num).toInt(), | ||||
|       createdAt: DateTime.parse(json['created_at'] as String), | ||||
|       updatedAt: DateTime.parse(json['updated_at'] as String), | ||||
|       deletedAt: json['deleted_at'] == null | ||||
|           ? null | ||||
|           : DateTime.parse(json['deleted_at'] as String), | ||||
|       rid: json['rid'] as String, | ||||
|       uuid: json['uuid'] as String, | ||||
|       size: (json['size'] as num).toInt(), | ||||
|       name: json['name'] as String, | ||||
|       alt: json['alt'] as String, | ||||
|       mimetype: json['mimetype'] as String, | ||||
|       hash: json['hash'] as String, | ||||
|       fingerprint: json['fingerprint'] as String?, | ||||
|       fileChunks: (json['file_chunks'] as Map<String, dynamic>?)?.map( | ||||
|             (k, e) => MapEntry(k, (e as num).toInt()), | ||||
|           ) ?? | ||||
|           const {}, | ||||
|       fileChunksMissing: (json['file_chunks_missing'] as List<dynamic>?) | ||||
|               ?.map((e) => e as String) | ||||
|               .toList() ?? | ||||
|           const [], | ||||
|     ); | ||||
|  | ||||
| Map<String, dynamic> _$$SnAttachmentFragmentImplToJson( | ||||
|         _$SnAttachmentFragmentImpl instance) => | ||||
|     <String, dynamic>{ | ||||
|       'id': instance.id, | ||||
|       'created_at': instance.createdAt.toIso8601String(), | ||||
|       'updated_at': instance.updatedAt.toIso8601String(), | ||||
|       'deleted_at': instance.deletedAt?.toIso8601String(), | ||||
|       'rid': instance.rid, | ||||
|       'uuid': instance.uuid, | ||||
|       'size': instance.size, | ||||
|       'name': instance.name, | ||||
|       'alt': instance.alt, | ||||
|       'mimetype': instance.mimetype, | ||||
|       'hash': instance.hash, | ||||
|       'fingerprint': instance.fingerprint, | ||||
|       'file_chunks': instance.fileChunks, | ||||
|       'file_chunks_missing': instance.fileChunksMissing, | ||||
|     }; | ||||
|  | ||||
| _$SnAttachmentPoolImpl _$$SnAttachmentPoolImplFromJson( | ||||
|         Map<String, dynamic> json) => | ||||
|     _$SnAttachmentPoolImpl( | ||||
| @@ -95,3 +167,117 @@ Map<String, dynamic> _$$SnAttachmentPoolImplToJson( | ||||
|       'config': instance.config, | ||||
|       'account_id': instance.accountId, | ||||
|     }; | ||||
|  | ||||
| _$SnAttachmentDestinationImpl _$$SnAttachmentDestinationImplFromJson( | ||||
|         Map<String, dynamic> json) => | ||||
|     _$SnAttachmentDestinationImpl( | ||||
|       id: (json['id'] as num?)?.toInt() ?? 0, | ||||
|       type: json['type'] as String, | ||||
|       label: json['label'] as String, | ||||
|       region: json['region'] as String, | ||||
|       isBoost: json['is_boost'] as bool, | ||||
|     ); | ||||
|  | ||||
| Map<String, dynamic> _$$SnAttachmentDestinationImplToJson( | ||||
|         _$SnAttachmentDestinationImpl instance) => | ||||
|     <String, dynamic>{ | ||||
|       'id': instance.id, | ||||
|       'type': instance.type, | ||||
|       'label': instance.label, | ||||
|       'region': instance.region, | ||||
|       'is_boost': instance.isBoost, | ||||
|     }; | ||||
|  | ||||
| _$SnAttachmentBoostImpl _$$SnAttachmentBoostImplFromJson( | ||||
|         Map<String, dynamic> json) => | ||||
|     _$SnAttachmentBoostImpl( | ||||
|       id: (json['id'] as num).toInt(), | ||||
|       createdAt: DateTime.parse(json['created_at'] as String), | ||||
|       updatedAt: DateTime.parse(json['updated_at'] as String), | ||||
|       deletedAt: json['deleted_at'] == null | ||||
|           ? null | ||||
|           : DateTime.parse(json['deleted_at'] as String), | ||||
|       status: (json['status'] as num).toInt(), | ||||
|       destination: (json['destination'] as num).toInt(), | ||||
|       attachmentId: (json['attachment_id'] as num).toInt(), | ||||
|       attachment: | ||||
|           SnAttachment.fromJson(json['attachment'] as Map<String, dynamic>), | ||||
|       account: (json['account'] as num).toInt(), | ||||
|     ); | ||||
|  | ||||
| Map<String, dynamic> _$$SnAttachmentBoostImplToJson( | ||||
|         _$SnAttachmentBoostImpl instance) => | ||||
|     <String, dynamic>{ | ||||
|       'id': instance.id, | ||||
|       'created_at': instance.createdAt.toIso8601String(), | ||||
|       'updated_at': instance.updatedAt.toIso8601String(), | ||||
|       'deleted_at': instance.deletedAt?.toIso8601String(), | ||||
|       'status': instance.status, | ||||
|       'destination': instance.destination, | ||||
|       'attachment_id': instance.attachmentId, | ||||
|       'attachment': instance.attachment.toJson(), | ||||
|       'account': instance.account, | ||||
|     }; | ||||
|  | ||||
| _$SnStickerImpl _$$SnStickerImplFromJson(Map<String, dynamic> json) => | ||||
|     _$SnStickerImpl( | ||||
|       id: (json['id'] as num).toInt(), | ||||
|       createdAt: DateTime.parse(json['created_at'] as String), | ||||
|       updatedAt: DateTime.parse(json['updated_at'] as String), | ||||
|       deletedAt: json['deleted_at'] == null | ||||
|           ? null | ||||
|           : DateTime.parse(json['deleted_at'] as String), | ||||
|       alias: json['alias'] as String, | ||||
|       name: json['name'] as String, | ||||
|       attachmentId: (json['attachment_id'] as num).toInt(), | ||||
|       attachment: | ||||
|           SnAttachment.fromJson(json['attachment'] as Map<String, dynamic>), | ||||
|       packId: (json['pack_id'] as num).toInt(), | ||||
|       pack: SnStickerPack.fromJson(json['pack'] as Map<String, dynamic>), | ||||
|       accountId: (json['account_id'] as num).toInt(), | ||||
|     ); | ||||
|  | ||||
| Map<String, dynamic> _$$SnStickerImplToJson(_$SnStickerImpl instance) => | ||||
|     <String, dynamic>{ | ||||
|       'id': instance.id, | ||||
|       'created_at': instance.createdAt.toIso8601String(), | ||||
|       'updated_at': instance.updatedAt.toIso8601String(), | ||||
|       'deleted_at': instance.deletedAt?.toIso8601String(), | ||||
|       'alias': instance.alias, | ||||
|       'name': instance.name, | ||||
|       'attachment_id': instance.attachmentId, | ||||
|       'attachment': instance.attachment.toJson(), | ||||
|       'pack_id': instance.packId, | ||||
|       'pack': instance.pack.toJson(), | ||||
|       'account_id': instance.accountId, | ||||
|     }; | ||||
|  | ||||
| _$SnStickerPackImpl _$$SnStickerPackImplFromJson(Map<String, dynamic> json) => | ||||
|     _$SnStickerPackImpl( | ||||
|       id: (json['id'] as num).toInt(), | ||||
|       createdAt: DateTime.parse(json['created_at'] as String), | ||||
|       updatedAt: DateTime.parse(json['updated_at'] as String), | ||||
|       deletedAt: json['deleted_at'] == null | ||||
|           ? null | ||||
|           : DateTime.parse(json['deleted_at'] as String), | ||||
|       prefix: json['prefix'] as String, | ||||
|       name: json['name'] as String, | ||||
|       description: json['description'] as String, | ||||
|       stickers: (json['stickers'] as List<dynamic>?) | ||||
|           ?.map((e) => SnSticker.fromJson(e as Map<String, dynamic>)) | ||||
|           .toList(), | ||||
|       accountId: (json['account_id'] as num).toInt(), | ||||
|     ); | ||||
|  | ||||
| Map<String, dynamic> _$$SnStickerPackImplToJson(_$SnStickerPackImpl instance) => | ||||
|     <String, dynamic>{ | ||||
|       'id': instance.id, | ||||
|       'created_at': instance.createdAt.toIso8601String(), | ||||
|       'updated_at': instance.updatedAt.toIso8601String(), | ||||
|       'deleted_at': instance.deletedAt?.toIso8601String(), | ||||
|       'prefix': instance.prefix, | ||||
|       'name': instance.name, | ||||
|       'description': instance.description, | ||||
|       'stickers': instance.stickers?.map((e) => e.toJson()).toList(), | ||||
|       'account_id': instance.accountId, | ||||
|     }; | ||||
|   | ||||
| @@ -3,6 +3,8 @@ import 'package:freezed_annotation/freezed_annotation.dart'; | ||||
| part 'check_in.freezed.dart'; | ||||
| part 'check_in.g.dart'; | ||||
|  | ||||
| const List<String> kCheckInResultTierSymbols = ['大凶', '凶', '中平', '吉', '大吉']; | ||||
|  | ||||
| @freezed | ||||
| class SnCheckInRecord with _$SnCheckInRecord { | ||||
|   const SnCheckInRecord._(); | ||||
| @@ -14,6 +16,7 @@ class SnCheckInRecord with _$SnCheckInRecord { | ||||
|     required DateTime? deletedAt, | ||||
|     required int resultTier, | ||||
|     required int resultExperience, | ||||
|     required double resultCoin, | ||||
|     required List<int> resultModifiers, | ||||
|     required int accountId, | ||||
|   }) = _SnCheckInRecord; | ||||
| @@ -21,11 +24,5 @@ class SnCheckInRecord with _$SnCheckInRecord { | ||||
|   factory SnCheckInRecord.fromJson(Map<String, dynamic> json) => | ||||
|       _$SnCheckInRecordFromJson(json); | ||||
|  | ||||
|   String get symbol => switch (resultTier) { | ||||
|         0 => '大凶', | ||||
|         1 => '凶', | ||||
|         2 => '中平', | ||||
|         3 => '吉', | ||||
|         _ => '大吉', | ||||
|       }; | ||||
|   String get symbol => kCheckInResultTierSymbols[resultTier]; | ||||
| } | ||||
|   | ||||
| @@ -26,6 +26,7 @@ mixin _$SnCheckInRecord { | ||||
|   DateTime? get deletedAt => throw _privateConstructorUsedError; | ||||
|   int get resultTier => throw _privateConstructorUsedError; | ||||
|   int get resultExperience => throw _privateConstructorUsedError; | ||||
|   double get resultCoin => throw _privateConstructorUsedError; | ||||
|   List<int> get resultModifiers => throw _privateConstructorUsedError; | ||||
|   int get accountId => throw _privateConstructorUsedError; | ||||
|  | ||||
| @@ -52,6 +53,7 @@ abstract class $SnCheckInRecordCopyWith<$Res> { | ||||
|       DateTime? deletedAt, | ||||
|       int resultTier, | ||||
|       int resultExperience, | ||||
|       double resultCoin, | ||||
|       List<int> resultModifiers, | ||||
|       int accountId}); | ||||
| } | ||||
| @@ -77,6 +79,7 @@ class _$SnCheckInRecordCopyWithImpl<$Res, $Val extends SnCheckInRecord> | ||||
|     Object? deletedAt = freezed, | ||||
|     Object? resultTier = null, | ||||
|     Object? resultExperience = null, | ||||
|     Object? resultCoin = null, | ||||
|     Object? resultModifiers = null, | ||||
|     Object? accountId = null, | ||||
|   }) { | ||||
| @@ -105,6 +108,10 @@ class _$SnCheckInRecordCopyWithImpl<$Res, $Val extends SnCheckInRecord> | ||||
|           ? _value.resultExperience | ||||
|           : resultExperience // ignore: cast_nullable_to_non_nullable | ||||
|               as int, | ||||
|       resultCoin: null == resultCoin | ||||
|           ? _value.resultCoin | ||||
|           : resultCoin // ignore: cast_nullable_to_non_nullable | ||||
|               as double, | ||||
|       resultModifiers: null == resultModifiers | ||||
|           ? _value.resultModifiers | ||||
|           : resultModifiers // ignore: cast_nullable_to_non_nullable | ||||
| @@ -132,6 +139,7 @@ abstract class _$$SnCheckInRecordImplCopyWith<$Res> | ||||
|       DateTime? deletedAt, | ||||
|       int resultTier, | ||||
|       int resultExperience, | ||||
|       double resultCoin, | ||||
|       List<int> resultModifiers, | ||||
|       int accountId}); | ||||
| } | ||||
| @@ -155,6 +163,7 @@ class __$$SnCheckInRecordImplCopyWithImpl<$Res> | ||||
|     Object? deletedAt = freezed, | ||||
|     Object? resultTier = null, | ||||
|     Object? resultExperience = null, | ||||
|     Object? resultCoin = null, | ||||
|     Object? resultModifiers = null, | ||||
|     Object? accountId = null, | ||||
|   }) { | ||||
| @@ -183,6 +192,10 @@ class __$$SnCheckInRecordImplCopyWithImpl<$Res> | ||||
|           ? _value.resultExperience | ||||
|           : resultExperience // ignore: cast_nullable_to_non_nullable | ||||
|               as int, | ||||
|       resultCoin: null == resultCoin | ||||
|           ? _value.resultCoin | ||||
|           : resultCoin // ignore: cast_nullable_to_non_nullable | ||||
|               as double, | ||||
|       resultModifiers: null == resultModifiers | ||||
|           ? _value._resultModifiers | ||||
|           : resultModifiers // ignore: cast_nullable_to_non_nullable | ||||
| @@ -205,6 +218,7 @@ class _$SnCheckInRecordImpl extends _SnCheckInRecord { | ||||
|       required this.deletedAt, | ||||
|       required this.resultTier, | ||||
|       required this.resultExperience, | ||||
|       required this.resultCoin, | ||||
|       required final List<int> resultModifiers, | ||||
|       required this.accountId}) | ||||
|       : _resultModifiers = resultModifiers, | ||||
| @@ -225,6 +239,8 @@ class _$SnCheckInRecordImpl extends _SnCheckInRecord { | ||||
|   final int resultTier; | ||||
|   @override | ||||
|   final int resultExperience; | ||||
|   @override | ||||
|   final double resultCoin; | ||||
|   final List<int> _resultModifiers; | ||||
|   @override | ||||
|   List<int> get resultModifiers { | ||||
| @@ -238,7 +254,7 @@ class _$SnCheckInRecordImpl extends _SnCheckInRecord { | ||||
|  | ||||
|   @override | ||||
|   String toString() { | ||||
|     return 'SnCheckInRecord(id: $id, createdAt: $createdAt, updatedAt: $updatedAt, deletedAt: $deletedAt, resultTier: $resultTier, resultExperience: $resultExperience, resultModifiers: $resultModifiers, accountId: $accountId)'; | ||||
|     return 'SnCheckInRecord(id: $id, createdAt: $createdAt, updatedAt: $updatedAt, deletedAt: $deletedAt, resultTier: $resultTier, resultExperience: $resultExperience, resultCoin: $resultCoin, resultModifiers: $resultModifiers, accountId: $accountId)'; | ||||
|   } | ||||
|  | ||||
|   @override | ||||
| @@ -257,6 +273,8 @@ class _$SnCheckInRecordImpl extends _SnCheckInRecord { | ||||
|                 other.resultTier == resultTier) && | ||||
|             (identical(other.resultExperience, resultExperience) || | ||||
|                 other.resultExperience == resultExperience) && | ||||
|             (identical(other.resultCoin, resultCoin) || | ||||
|                 other.resultCoin == resultCoin) && | ||||
|             const DeepCollectionEquality() | ||||
|                 .equals(other._resultModifiers, _resultModifiers) && | ||||
|             (identical(other.accountId, accountId) || | ||||
| @@ -273,6 +291,7 @@ class _$SnCheckInRecordImpl extends _SnCheckInRecord { | ||||
|       deletedAt, | ||||
|       resultTier, | ||||
|       resultExperience, | ||||
|       resultCoin, | ||||
|       const DeepCollectionEquality().hash(_resultModifiers), | ||||
|       accountId); | ||||
|  | ||||
| @@ -301,6 +320,7 @@ abstract class _SnCheckInRecord extends SnCheckInRecord { | ||||
|       required final DateTime? deletedAt, | ||||
|       required final int resultTier, | ||||
|       required final int resultExperience, | ||||
|       required final double resultCoin, | ||||
|       required final List<int> resultModifiers, | ||||
|       required final int accountId}) = _$SnCheckInRecordImpl; | ||||
|   const _SnCheckInRecord._() : super._(); | ||||
| @@ -321,6 +341,8 @@ abstract class _SnCheckInRecord extends SnCheckInRecord { | ||||
|   @override | ||||
|   int get resultExperience; | ||||
|   @override | ||||
|   double get resultCoin; | ||||
|   @override | ||||
|   List<int> get resultModifiers; | ||||
|   @override | ||||
|   int get accountId; | ||||
|   | ||||
| @@ -17,6 +17,7 @@ _$SnCheckInRecordImpl _$$SnCheckInRecordImplFromJson( | ||||
|           : DateTime.parse(json['deleted_at'] as String), | ||||
|       resultTier: (json['result_tier'] as num).toInt(), | ||||
|       resultExperience: (json['result_experience'] as num).toInt(), | ||||
|       resultCoin: (json['result_coin'] as num).toDouble(), | ||||
|       resultModifiers: (json['result_modifiers'] as List<dynamic>) | ||||
|           .map((e) => (e as num).toInt()) | ||||
|           .toList(), | ||||
| @@ -32,6 +33,7 @@ Map<String, dynamic> _$$SnCheckInRecordImplToJson( | ||||
|       'deleted_at': instance.deletedAt?.toIso8601String(), | ||||
|       'result_tier': instance.resultTier, | ||||
|       'result_experience': instance.resultExperience, | ||||
|       'result_coin': instance.resultCoin, | ||||
|       'result_modifiers': instance.resultModifiers, | ||||
|       'account_id': instance.accountId, | ||||
|     }; | ||||
|   | ||||
							
								
								
									
										38
									
								
								lib/types/news.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,38 @@ | ||||
| import 'package:freezed_annotation/freezed_annotation.dart'; | ||||
|  | ||||
| part 'news.freezed.dart'; | ||||
| part 'news.g.dart'; | ||||
|  | ||||
| @freezed | ||||
| class SnNewsSource with _$SnNewsSource { | ||||
|   const factory SnNewsSource({ | ||||
|     required String id, | ||||
|     required String label, | ||||
|     required String type, | ||||
|     required String source, | ||||
|     required int depth, | ||||
|     required bool enabled, | ||||
|   }) = _SnNewsSource; | ||||
|  | ||||
|   factory SnNewsSource.fromJson(Map<String, dynamic> json) => _$SnNewsSourceFromJson(json); | ||||
| } | ||||
|  | ||||
| @freezed | ||||
| class SnNewsArticle with _$SnNewsArticle { | ||||
|   const factory SnNewsArticle({ | ||||
|     required int id, | ||||
|     required DateTime createdAt, | ||||
|     required DateTime updatedAt, | ||||
|     required dynamic deletedAt, | ||||
|     required String thumbnail, | ||||
|     required String title, | ||||
|     required String description, | ||||
|     required String content, | ||||
|     required String url, | ||||
|     required String hash, | ||||
|     required String source, | ||||
|     required DateTime? publishedAt, | ||||
|   }) = _SnNewsArticle; | ||||
|  | ||||
|   factory SnNewsArticle.fromJson(Map<String, dynamic> json) => _$SnNewsArticleFromJson(json); | ||||
| } | ||||
							
								
								
									
										660
									
								
								lib/types/news.freezed.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,660 @@ | ||||
| // coverage:ignore-file | ||||
| // GENERATED CODE - DO NOT MODIFY BY HAND | ||||
| // ignore_for_file: type=lint | ||||
| // ignore_for_file: unused_element, deprecated_member_use, deprecated_member_use_from_same_package, use_function_type_syntax_for_parameters, unnecessary_const, avoid_init_to_null, invalid_override_different_default_values_named, prefer_expression_function_bodies, annotate_overrides, invalid_annotation_target, unnecessary_question_mark | ||||
|  | ||||
| part of 'news.dart'; | ||||
|  | ||||
| // ************************************************************************** | ||||
| // FreezedGenerator | ||||
| // ************************************************************************** | ||||
|  | ||||
| T _$identity<T>(T value) => value; | ||||
|  | ||||
| final _privateConstructorUsedError = UnsupportedError( | ||||
|     'It seems like you constructed your class using `MyClass._()`. This constructor is only meant to be used by freezed and you are not supposed to need it nor use it.\nPlease check the documentation here for more information: https://github.com/rrousselGit/freezed#adding-getters-and-methods-to-our-models'); | ||||
|  | ||||
| SnNewsSource _$SnNewsSourceFromJson(Map<String, dynamic> json) { | ||||
|   return _SnNewsSource.fromJson(json); | ||||
| } | ||||
|  | ||||
| /// @nodoc | ||||
| mixin _$SnNewsSource { | ||||
|   String get id => throw _privateConstructorUsedError; | ||||
|   String get label => throw _privateConstructorUsedError; | ||||
|   String get type => throw _privateConstructorUsedError; | ||||
|   String get source => throw _privateConstructorUsedError; | ||||
|   int get depth => throw _privateConstructorUsedError; | ||||
|   bool get enabled => throw _privateConstructorUsedError; | ||||
|  | ||||
|   /// Serializes this SnNewsSource to a JSON map. | ||||
|   Map<String, dynamic> toJson() => throw _privateConstructorUsedError; | ||||
|  | ||||
|   /// Create a copy of SnNewsSource | ||||
|   /// with the given fields replaced by the non-null parameter values. | ||||
|   @JsonKey(includeFromJson: false, includeToJson: false) | ||||
|   $SnNewsSourceCopyWith<SnNewsSource> get copyWith => | ||||
|       throw _privateConstructorUsedError; | ||||
| } | ||||
|  | ||||
| /// @nodoc | ||||
| abstract class $SnNewsSourceCopyWith<$Res> { | ||||
|   factory $SnNewsSourceCopyWith( | ||||
|           SnNewsSource value, $Res Function(SnNewsSource) then) = | ||||
|       _$SnNewsSourceCopyWithImpl<$Res, SnNewsSource>; | ||||
|   @useResult | ||||
|   $Res call( | ||||
|       {String id, | ||||
|       String label, | ||||
|       String type, | ||||
|       String source, | ||||
|       int depth, | ||||
|       bool enabled}); | ||||
| } | ||||
|  | ||||
| /// @nodoc | ||||
| class _$SnNewsSourceCopyWithImpl<$Res, $Val extends SnNewsSource> | ||||
|     implements $SnNewsSourceCopyWith<$Res> { | ||||
|   _$SnNewsSourceCopyWithImpl(this._value, this._then); | ||||
|  | ||||
|   // ignore: unused_field | ||||
|   final $Val _value; | ||||
|   // ignore: unused_field | ||||
|   final $Res Function($Val) _then; | ||||
|  | ||||
|   /// Create a copy of SnNewsSource | ||||
|   /// with the given fields replaced by the non-null parameter values. | ||||
|   @pragma('vm:prefer-inline') | ||||
|   @override | ||||
|   $Res call({ | ||||
|     Object? id = null, | ||||
|     Object? label = null, | ||||
|     Object? type = null, | ||||
|     Object? source = null, | ||||
|     Object? depth = null, | ||||
|     Object? enabled = null, | ||||
|   }) { | ||||
|     return _then(_value.copyWith( | ||||
|       id: null == id | ||||
|           ? _value.id | ||||
|           : id // ignore: cast_nullable_to_non_nullable | ||||
|               as String, | ||||
|       label: null == label | ||||
|           ? _value.label | ||||
|           : label // ignore: cast_nullable_to_non_nullable | ||||
|               as String, | ||||
|       type: null == type | ||||
|           ? _value.type | ||||
|           : type // ignore: cast_nullable_to_non_nullable | ||||
|               as String, | ||||
|       source: null == source | ||||
|           ? _value.source | ||||
|           : source // ignore: cast_nullable_to_non_nullable | ||||
|               as String, | ||||
|       depth: null == depth | ||||
|           ? _value.depth | ||||
|           : depth // ignore: cast_nullable_to_non_nullable | ||||
|               as int, | ||||
|       enabled: null == enabled | ||||
|           ? _value.enabled | ||||
|           : enabled // ignore: cast_nullable_to_non_nullable | ||||
|               as bool, | ||||
|     ) as $Val); | ||||
|   } | ||||
| } | ||||
|  | ||||
| /// @nodoc | ||||
| abstract class _$$SnNewsSourceImplCopyWith<$Res> | ||||
|     implements $SnNewsSourceCopyWith<$Res> { | ||||
|   factory _$$SnNewsSourceImplCopyWith( | ||||
|           _$SnNewsSourceImpl value, $Res Function(_$SnNewsSourceImpl) then) = | ||||
|       __$$SnNewsSourceImplCopyWithImpl<$Res>; | ||||
|   @override | ||||
|   @useResult | ||||
|   $Res call( | ||||
|       {String id, | ||||
|       String label, | ||||
|       String type, | ||||
|       String source, | ||||
|       int depth, | ||||
|       bool enabled}); | ||||
| } | ||||
|  | ||||
| /// @nodoc | ||||
| class __$$SnNewsSourceImplCopyWithImpl<$Res> | ||||
|     extends _$SnNewsSourceCopyWithImpl<$Res, _$SnNewsSourceImpl> | ||||
|     implements _$$SnNewsSourceImplCopyWith<$Res> { | ||||
|   __$$SnNewsSourceImplCopyWithImpl( | ||||
|       _$SnNewsSourceImpl _value, $Res Function(_$SnNewsSourceImpl) _then) | ||||
|       : super(_value, _then); | ||||
|  | ||||
|   /// Create a copy of SnNewsSource | ||||
|   /// with the given fields replaced by the non-null parameter values. | ||||
|   @pragma('vm:prefer-inline') | ||||
|   @override | ||||
|   $Res call({ | ||||
|     Object? id = null, | ||||
|     Object? label = null, | ||||
|     Object? type = null, | ||||
|     Object? source = null, | ||||
|     Object? depth = null, | ||||
|     Object? enabled = null, | ||||
|   }) { | ||||
|     return _then(_$SnNewsSourceImpl( | ||||
|       id: null == id | ||||
|           ? _value.id | ||||
|           : id // ignore: cast_nullable_to_non_nullable | ||||
|               as String, | ||||
|       label: null == label | ||||
|           ? _value.label | ||||
|           : label // ignore: cast_nullable_to_non_nullable | ||||
|               as String, | ||||
|       type: null == type | ||||
|           ? _value.type | ||||
|           : type // ignore: cast_nullable_to_non_nullable | ||||
|               as String, | ||||
|       source: null == source | ||||
|           ? _value.source | ||||
|           : source // ignore: cast_nullable_to_non_nullable | ||||
|               as String, | ||||
|       depth: null == depth | ||||
|           ? _value.depth | ||||
|           : depth // ignore: cast_nullable_to_non_nullable | ||||
|               as int, | ||||
|       enabled: null == enabled | ||||
|           ? _value.enabled | ||||
|           : enabled // ignore: cast_nullable_to_non_nullable | ||||
|               as bool, | ||||
|     )); | ||||
|   } | ||||
| } | ||||
|  | ||||
| /// @nodoc | ||||
| @JsonSerializable() | ||||
| class _$SnNewsSourceImpl implements _SnNewsSource { | ||||
|   const _$SnNewsSourceImpl( | ||||
|       {required this.id, | ||||
|       required this.label, | ||||
|       required this.type, | ||||
|       required this.source, | ||||
|       required this.depth, | ||||
|       required this.enabled}); | ||||
|  | ||||
|   factory _$SnNewsSourceImpl.fromJson(Map<String, dynamic> json) => | ||||
|       _$$SnNewsSourceImplFromJson(json); | ||||
|  | ||||
|   @override | ||||
|   final String id; | ||||
|   @override | ||||
|   final String label; | ||||
|   @override | ||||
|   final String type; | ||||
|   @override | ||||
|   final String source; | ||||
|   @override | ||||
|   final int depth; | ||||
|   @override | ||||
|   final bool enabled; | ||||
|  | ||||
|   @override | ||||
|   String toString() { | ||||
|     return 'SnNewsSource(id: $id, label: $label, type: $type, source: $source, depth: $depth, enabled: $enabled)'; | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   bool operator ==(Object other) { | ||||
|     return identical(this, other) || | ||||
|         (other.runtimeType == runtimeType && | ||||
|             other is _$SnNewsSourceImpl && | ||||
|             (identical(other.id, id) || other.id == id) && | ||||
|             (identical(other.label, label) || other.label == label) && | ||||
|             (identical(other.type, type) || other.type == type) && | ||||
|             (identical(other.source, source) || other.source == source) && | ||||
|             (identical(other.depth, depth) || other.depth == depth) && | ||||
|             (identical(other.enabled, enabled) || other.enabled == enabled)); | ||||
|   } | ||||
|  | ||||
|   @JsonKey(includeFromJson: false, includeToJson: false) | ||||
|   @override | ||||
|   int get hashCode => | ||||
|       Object.hash(runtimeType, id, label, type, source, depth, enabled); | ||||
|  | ||||
|   /// Create a copy of SnNewsSource | ||||
|   /// with the given fields replaced by the non-null parameter values. | ||||
|   @JsonKey(includeFromJson: false, includeToJson: false) | ||||
|   @override | ||||
|   @pragma('vm:prefer-inline') | ||||
|   _$$SnNewsSourceImplCopyWith<_$SnNewsSourceImpl> get copyWith => | ||||
|       __$$SnNewsSourceImplCopyWithImpl<_$SnNewsSourceImpl>(this, _$identity); | ||||
|  | ||||
|   @override | ||||
|   Map<String, dynamic> toJson() { | ||||
|     return _$$SnNewsSourceImplToJson( | ||||
|       this, | ||||
|     ); | ||||
|   } | ||||
| } | ||||
|  | ||||
| abstract class _SnNewsSource implements SnNewsSource { | ||||
|   const factory _SnNewsSource( | ||||
|       {required final String id, | ||||
|       required final String label, | ||||
|       required final String type, | ||||
|       required final String source, | ||||
|       required final int depth, | ||||
|       required final bool enabled}) = _$SnNewsSourceImpl; | ||||
|  | ||||
|   factory _SnNewsSource.fromJson(Map<String, dynamic> json) = | ||||
|       _$SnNewsSourceImpl.fromJson; | ||||
|  | ||||
|   @override | ||||
|   String get id; | ||||
|   @override | ||||
|   String get label; | ||||
|   @override | ||||
|   String get type; | ||||
|   @override | ||||
|   String get source; | ||||
|   @override | ||||
|   int get depth; | ||||
|   @override | ||||
|   bool get enabled; | ||||
|  | ||||
|   /// Create a copy of SnNewsSource | ||||
|   /// with the given fields replaced by the non-null parameter values. | ||||
|   @override | ||||
|   @JsonKey(includeFromJson: false, includeToJson: false) | ||||
|   _$$SnNewsSourceImplCopyWith<_$SnNewsSourceImpl> get copyWith => | ||||
|       throw _privateConstructorUsedError; | ||||
| } | ||||
|  | ||||
| SnNewsArticle _$SnNewsArticleFromJson(Map<String, dynamic> json) { | ||||
|   return _SnNewsArticle.fromJson(json); | ||||
| } | ||||
|  | ||||
| /// @nodoc | ||||
| mixin _$SnNewsArticle { | ||||
|   int get id => throw _privateConstructorUsedError; | ||||
|   DateTime get createdAt => throw _privateConstructorUsedError; | ||||
|   DateTime get updatedAt => throw _privateConstructorUsedError; | ||||
|   dynamic get deletedAt => throw _privateConstructorUsedError; | ||||
|   String get thumbnail => throw _privateConstructorUsedError; | ||||
|   String get title => throw _privateConstructorUsedError; | ||||
|   String get description => throw _privateConstructorUsedError; | ||||
|   String get content => throw _privateConstructorUsedError; | ||||
|   String get url => throw _privateConstructorUsedError; | ||||
|   String get hash => throw _privateConstructorUsedError; | ||||
|   String get source => throw _privateConstructorUsedError; | ||||
|   DateTime? get publishedAt => throw _privateConstructorUsedError; | ||||
|  | ||||
|   /// Serializes this SnNewsArticle to a JSON map. | ||||
|   Map<String, dynamic> toJson() => throw _privateConstructorUsedError; | ||||
|  | ||||
|   /// Create a copy of SnNewsArticle | ||||
|   /// with the given fields replaced by the non-null parameter values. | ||||
|   @JsonKey(includeFromJson: false, includeToJson: false) | ||||
|   $SnNewsArticleCopyWith<SnNewsArticle> get copyWith => | ||||
|       throw _privateConstructorUsedError; | ||||
| } | ||||
|  | ||||
| /// @nodoc | ||||
| abstract class $SnNewsArticleCopyWith<$Res> { | ||||
|   factory $SnNewsArticleCopyWith( | ||||
|           SnNewsArticle value, $Res Function(SnNewsArticle) then) = | ||||
|       _$SnNewsArticleCopyWithImpl<$Res, SnNewsArticle>; | ||||
|   @useResult | ||||
|   $Res call( | ||||
|       {int id, | ||||
|       DateTime createdAt, | ||||
|       DateTime updatedAt, | ||||
|       dynamic deletedAt, | ||||
|       String thumbnail, | ||||
|       String title, | ||||
|       String description, | ||||
|       String content, | ||||
|       String url, | ||||
|       String hash, | ||||
|       String source, | ||||
|       DateTime? publishedAt}); | ||||
| } | ||||
|  | ||||
| /// @nodoc | ||||
| class _$SnNewsArticleCopyWithImpl<$Res, $Val extends SnNewsArticle> | ||||
|     implements $SnNewsArticleCopyWith<$Res> { | ||||
|   _$SnNewsArticleCopyWithImpl(this._value, this._then); | ||||
|  | ||||
|   // ignore: unused_field | ||||
|   final $Val _value; | ||||
|   // ignore: unused_field | ||||
|   final $Res Function($Val) _then; | ||||
|  | ||||
|   /// Create a copy of SnNewsArticle | ||||
|   /// with the given fields replaced by the non-null parameter values. | ||||
|   @pragma('vm:prefer-inline') | ||||
|   @override | ||||
|   $Res call({ | ||||
|     Object? id = null, | ||||
|     Object? createdAt = null, | ||||
|     Object? updatedAt = null, | ||||
|     Object? deletedAt = freezed, | ||||
|     Object? thumbnail = null, | ||||
|     Object? title = null, | ||||
|     Object? description = null, | ||||
|     Object? content = null, | ||||
|     Object? url = null, | ||||
|     Object? hash = null, | ||||
|     Object? source = null, | ||||
|     Object? publishedAt = freezed, | ||||
|   }) { | ||||
|     return _then(_value.copyWith( | ||||
|       id: null == id | ||||
|           ? _value.id | ||||
|           : id // ignore: cast_nullable_to_non_nullable | ||||
|               as int, | ||||
|       createdAt: null == createdAt | ||||
|           ? _value.createdAt | ||||
|           : createdAt // ignore: cast_nullable_to_non_nullable | ||||
|               as DateTime, | ||||
|       updatedAt: null == updatedAt | ||||
|           ? _value.updatedAt | ||||
|           : updatedAt // ignore: cast_nullable_to_non_nullable | ||||
|               as DateTime, | ||||
|       deletedAt: freezed == deletedAt | ||||
|           ? _value.deletedAt | ||||
|           : deletedAt // ignore: cast_nullable_to_non_nullable | ||||
|               as dynamic, | ||||
|       thumbnail: null == thumbnail | ||||
|           ? _value.thumbnail | ||||
|           : thumbnail // ignore: cast_nullable_to_non_nullable | ||||
|               as String, | ||||
|       title: null == title | ||||
|           ? _value.title | ||||
|           : title // ignore: cast_nullable_to_non_nullable | ||||
|               as String, | ||||
|       description: null == description | ||||
|           ? _value.description | ||||
|           : description // ignore: cast_nullable_to_non_nullable | ||||
|               as String, | ||||
|       content: null == content | ||||
|           ? _value.content | ||||
|           : content // ignore: cast_nullable_to_non_nullable | ||||
|               as String, | ||||
|       url: null == url | ||||
|           ? _value.url | ||||
|           : url // ignore: cast_nullable_to_non_nullable | ||||
|               as String, | ||||
|       hash: null == hash | ||||
|           ? _value.hash | ||||
|           : hash // ignore: cast_nullable_to_non_nullable | ||||
|               as String, | ||||
|       source: null == source | ||||
|           ? _value.source | ||||
|           : source // ignore: cast_nullable_to_non_nullable | ||||
|               as String, | ||||
|       publishedAt: freezed == publishedAt | ||||
|           ? _value.publishedAt | ||||
|           : publishedAt // ignore: cast_nullable_to_non_nullable | ||||
|               as DateTime?, | ||||
|     ) as $Val); | ||||
|   } | ||||
| } | ||||
|  | ||||
| /// @nodoc | ||||
| abstract class _$$SnNewsArticleImplCopyWith<$Res> | ||||
|     implements $SnNewsArticleCopyWith<$Res> { | ||||
|   factory _$$SnNewsArticleImplCopyWith( | ||||
|           _$SnNewsArticleImpl value, $Res Function(_$SnNewsArticleImpl) then) = | ||||
|       __$$SnNewsArticleImplCopyWithImpl<$Res>; | ||||
|   @override | ||||
|   @useResult | ||||
|   $Res call( | ||||
|       {int id, | ||||
|       DateTime createdAt, | ||||
|       DateTime updatedAt, | ||||
|       dynamic deletedAt, | ||||
|       String thumbnail, | ||||
|       String title, | ||||
|       String description, | ||||
|       String content, | ||||
|       String url, | ||||
|       String hash, | ||||
|       String source, | ||||
|       DateTime? publishedAt}); | ||||
| } | ||||
|  | ||||
| /// @nodoc | ||||
| class __$$SnNewsArticleImplCopyWithImpl<$Res> | ||||
|     extends _$SnNewsArticleCopyWithImpl<$Res, _$SnNewsArticleImpl> | ||||
|     implements _$$SnNewsArticleImplCopyWith<$Res> { | ||||
|   __$$SnNewsArticleImplCopyWithImpl( | ||||
|       _$SnNewsArticleImpl _value, $Res Function(_$SnNewsArticleImpl) _then) | ||||
|       : super(_value, _then); | ||||
|  | ||||
|   /// Create a copy of SnNewsArticle | ||||
|   /// with the given fields replaced by the non-null parameter values. | ||||
|   @pragma('vm:prefer-inline') | ||||
|   @override | ||||
|   $Res call({ | ||||
|     Object? id = null, | ||||
|     Object? createdAt = null, | ||||
|     Object? updatedAt = null, | ||||
|     Object? deletedAt = freezed, | ||||
|     Object? thumbnail = null, | ||||
|     Object? title = null, | ||||
|     Object? description = null, | ||||
|     Object? content = null, | ||||
|     Object? url = null, | ||||
|     Object? hash = null, | ||||
|     Object? source = null, | ||||
|     Object? publishedAt = freezed, | ||||
|   }) { | ||||
|     return _then(_$SnNewsArticleImpl( | ||||
|       id: null == id | ||||
|           ? _value.id | ||||
|           : id // ignore: cast_nullable_to_non_nullable | ||||
|               as int, | ||||
|       createdAt: null == createdAt | ||||
|           ? _value.createdAt | ||||
|           : createdAt // ignore: cast_nullable_to_non_nullable | ||||
|               as DateTime, | ||||
|       updatedAt: null == updatedAt | ||||
|           ? _value.updatedAt | ||||
|           : updatedAt // ignore: cast_nullable_to_non_nullable | ||||
|               as DateTime, | ||||
|       deletedAt: freezed == deletedAt | ||||
|           ? _value.deletedAt | ||||
|           : deletedAt // ignore: cast_nullable_to_non_nullable | ||||
|               as dynamic, | ||||
|       thumbnail: null == thumbnail | ||||
|           ? _value.thumbnail | ||||
|           : thumbnail // ignore: cast_nullable_to_non_nullable | ||||
|               as String, | ||||
|       title: null == title | ||||
|           ? _value.title | ||||
|           : title // ignore: cast_nullable_to_non_nullable | ||||
|               as String, | ||||
|       description: null == description | ||||
|           ? _value.description | ||||
|           : description // ignore: cast_nullable_to_non_nullable | ||||
|               as String, | ||||
|       content: null == content | ||||
|           ? _value.content | ||||
|           : content // ignore: cast_nullable_to_non_nullable | ||||
|               as String, | ||||
|       url: null == url | ||||
|           ? _value.url | ||||
|           : url // ignore: cast_nullable_to_non_nullable | ||||
|               as String, | ||||
|       hash: null == hash | ||||
|           ? _value.hash | ||||
|           : hash // ignore: cast_nullable_to_non_nullable | ||||
|               as String, | ||||
|       source: null == source | ||||
|           ? _value.source | ||||
|           : source // ignore: cast_nullable_to_non_nullable | ||||
|               as String, | ||||
|       publishedAt: freezed == publishedAt | ||||
|           ? _value.publishedAt | ||||
|           : publishedAt // ignore: cast_nullable_to_non_nullable | ||||
|               as DateTime?, | ||||
|     )); | ||||
|   } | ||||
| } | ||||
|  | ||||
| /// @nodoc | ||||
| @JsonSerializable() | ||||
| class _$SnNewsArticleImpl implements _SnNewsArticle { | ||||
|   const _$SnNewsArticleImpl( | ||||
|       {required this.id, | ||||
|       required this.createdAt, | ||||
|       required this.updatedAt, | ||||
|       required this.deletedAt, | ||||
|       required this.thumbnail, | ||||
|       required this.title, | ||||
|       required this.description, | ||||
|       required this.content, | ||||
|       required this.url, | ||||
|       required this.hash, | ||||
|       required this.source, | ||||
|       required this.publishedAt}); | ||||
|  | ||||
|   factory _$SnNewsArticleImpl.fromJson(Map<String, dynamic> json) => | ||||
|       _$$SnNewsArticleImplFromJson(json); | ||||
|  | ||||
|   @override | ||||
|   final int id; | ||||
|   @override | ||||
|   final DateTime createdAt; | ||||
|   @override | ||||
|   final DateTime updatedAt; | ||||
|   @override | ||||
|   final dynamic deletedAt; | ||||
|   @override | ||||
|   final String thumbnail; | ||||
|   @override | ||||
|   final String title; | ||||
|   @override | ||||
|   final String description; | ||||
|   @override | ||||
|   final String content; | ||||
|   @override | ||||
|   final String url; | ||||
|   @override | ||||
|   final String hash; | ||||
|   @override | ||||
|   final String source; | ||||
|   @override | ||||
|   final DateTime? publishedAt; | ||||
|  | ||||
|   @override | ||||
|   String toString() { | ||||
|     return 'SnNewsArticle(id: $id, createdAt: $createdAt, updatedAt: $updatedAt, deletedAt: $deletedAt, thumbnail: $thumbnail, title: $title, description: $description, content: $content, url: $url, hash: $hash, source: $source, publishedAt: $publishedAt)'; | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   bool operator ==(Object other) { | ||||
|     return identical(this, other) || | ||||
|         (other.runtimeType == runtimeType && | ||||
|             other is _$SnNewsArticleImpl && | ||||
|             (identical(other.id, id) || other.id == id) && | ||||
|             (identical(other.createdAt, createdAt) || | ||||
|                 other.createdAt == createdAt) && | ||||
|             (identical(other.updatedAt, updatedAt) || | ||||
|                 other.updatedAt == updatedAt) && | ||||
|             const DeepCollectionEquality().equals(other.deletedAt, deletedAt) && | ||||
|             (identical(other.thumbnail, thumbnail) || | ||||
|                 other.thumbnail == thumbnail) && | ||||
|             (identical(other.title, title) || other.title == title) && | ||||
|             (identical(other.description, description) || | ||||
|                 other.description == description) && | ||||
|             (identical(other.content, content) || other.content == content) && | ||||
|             (identical(other.url, url) || other.url == url) && | ||||
|             (identical(other.hash, hash) || other.hash == hash) && | ||||
|             (identical(other.source, source) || other.source == source) && | ||||
|             (identical(other.publishedAt, publishedAt) || | ||||
|                 other.publishedAt == publishedAt)); | ||||
|   } | ||||
|  | ||||
|   @JsonKey(includeFromJson: false, includeToJson: false) | ||||
|   @override | ||||
|   int get hashCode => Object.hash( | ||||
|       runtimeType, | ||||
|       id, | ||||
|       createdAt, | ||||
|       updatedAt, | ||||
|       const DeepCollectionEquality().hash(deletedAt), | ||||
|       thumbnail, | ||||
|       title, | ||||
|       description, | ||||
|       content, | ||||
|       url, | ||||
|       hash, | ||||
|       source, | ||||
|       publishedAt); | ||||
|  | ||||
|   /// Create a copy of SnNewsArticle | ||||
|   /// with the given fields replaced by the non-null parameter values. | ||||
|   @JsonKey(includeFromJson: false, includeToJson: false) | ||||
|   @override | ||||
|   @pragma('vm:prefer-inline') | ||||
|   _$$SnNewsArticleImplCopyWith<_$SnNewsArticleImpl> get copyWith => | ||||
|       __$$SnNewsArticleImplCopyWithImpl<_$SnNewsArticleImpl>(this, _$identity); | ||||
|  | ||||
|   @override | ||||
|   Map<String, dynamic> toJson() { | ||||
|     return _$$SnNewsArticleImplToJson( | ||||
|       this, | ||||
|     ); | ||||
|   } | ||||
| } | ||||
|  | ||||
| abstract class _SnNewsArticle implements SnNewsArticle { | ||||
|   const factory _SnNewsArticle( | ||||
|       {required final int id, | ||||
|       required final DateTime createdAt, | ||||
|       required final DateTime updatedAt, | ||||
|       required final dynamic deletedAt, | ||||
|       required final String thumbnail, | ||||
|       required final String title, | ||||
|       required final String description, | ||||
|       required final String content, | ||||
|       required final String url, | ||||
|       required final String hash, | ||||
|       required final String source, | ||||
|       required final DateTime? publishedAt}) = _$SnNewsArticleImpl; | ||||
|  | ||||
|   factory _SnNewsArticle.fromJson(Map<String, dynamic> json) = | ||||
|       _$SnNewsArticleImpl.fromJson; | ||||
|  | ||||
|   @override | ||||
|   int get id; | ||||
|   @override | ||||
|   DateTime get createdAt; | ||||
|   @override | ||||
|   DateTime get updatedAt; | ||||
|   @override | ||||
|   dynamic get deletedAt; | ||||
|   @override | ||||
|   String get thumbnail; | ||||
|   @override | ||||
|   String get title; | ||||
|   @override | ||||
|   String get description; | ||||
|   @override | ||||
|   String get content; | ||||
|   @override | ||||
|   String get url; | ||||
|   @override | ||||
|   String get hash; | ||||
|   @override | ||||
|   String get source; | ||||
|   @override | ||||
|   DateTime? get publishedAt; | ||||
|  | ||||
|   /// Create a copy of SnNewsArticle | ||||
|   /// with the given fields replaced by the non-null parameter values. | ||||
|   @override | ||||
|   @JsonKey(includeFromJson: false, includeToJson: false) | ||||
|   _$$SnNewsArticleImplCopyWith<_$SnNewsArticleImpl> get copyWith => | ||||
|       throw _privateConstructorUsedError; | ||||
| } | ||||
							
								
								
									
										61
									
								
								lib/types/news.g.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,61 @@ | ||||
| // GENERATED CODE - DO NOT MODIFY BY HAND | ||||
|  | ||||
| part of 'news.dart'; | ||||
|  | ||||
| // ************************************************************************** | ||||
| // JsonSerializableGenerator | ||||
| // ************************************************************************** | ||||
|  | ||||
| _$SnNewsSourceImpl _$$SnNewsSourceImplFromJson(Map<String, dynamic> json) => | ||||
|     _$SnNewsSourceImpl( | ||||
|       id: json['id'] as String, | ||||
|       label: json['label'] as String, | ||||
|       type: json['type'] as String, | ||||
|       source: json['source'] as String, | ||||
|       depth: (json['depth'] as num).toInt(), | ||||
|       enabled: json['enabled'] as bool, | ||||
|     ); | ||||
|  | ||||
| Map<String, dynamic> _$$SnNewsSourceImplToJson(_$SnNewsSourceImpl instance) => | ||||
|     <String, dynamic>{ | ||||
|       'id': instance.id, | ||||
|       'label': instance.label, | ||||
|       'type': instance.type, | ||||
|       'source': instance.source, | ||||
|       'depth': instance.depth, | ||||
|       'enabled': instance.enabled, | ||||
|     }; | ||||
|  | ||||
| _$SnNewsArticleImpl _$$SnNewsArticleImplFromJson(Map<String, dynamic> json) => | ||||
|     _$SnNewsArticleImpl( | ||||
|       id: (json['id'] as num).toInt(), | ||||
|       createdAt: DateTime.parse(json['created_at'] as String), | ||||
|       updatedAt: DateTime.parse(json['updated_at'] as String), | ||||
|       deletedAt: json['deleted_at'], | ||||
|       thumbnail: json['thumbnail'] as String, | ||||
|       title: json['title'] as String, | ||||
|       description: json['description'] as String, | ||||
|       content: json['content'] as String, | ||||
|       url: json['url'] as String, | ||||
|       hash: json['hash'] as String, | ||||
|       source: json['source'] as String, | ||||
|       publishedAt: json['published_at'] == null | ||||
|           ? null | ||||
|           : DateTime.parse(json['published_at'] as String), | ||||
|     ); | ||||
|  | ||||
| Map<String, dynamic> _$$SnNewsArticleImplToJson(_$SnNewsArticleImpl instance) => | ||||
|     <String, dynamic>{ | ||||
|       'id': instance.id, | ||||
|       'created_at': instance.createdAt.toIso8601String(), | ||||
|       'updated_at': instance.updatedAt.toIso8601String(), | ||||
|       'deleted_at': instance.deletedAt, | ||||
|       'thumbnail': instance.thumbnail, | ||||
|       'title': instance.title, | ||||
|       'description': instance.description, | ||||
|       'content': instance.content, | ||||
|       'url': instance.url, | ||||
|       'hash': instance.hash, | ||||
|       'source': instance.source, | ||||
|       'published_at': instance.publishedAt?.toIso8601String(), | ||||
|     }; | ||||