Compare commits
	
		
			194 Commits
		
	
	
		
			2.0.0+2
			...
			d6d60e60a9
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| d6d60e60a9 | |||
| 435b730f3b | |||
| 73468c5c6d | |||
| 8db6513eef | |||
| 65a8f1e6c3 | |||
| 2671ffad4b | |||
| 8a628823e0 | |||
| 94d19a1524 | |||
| d98f6c8d18 | |||
| 6d0f62016a | |||
| 7e0faba5db | |||
| 7508a54907 | |||
| 2eb1f4b52b | |||
| 00678c0ac8 | |||
| abc21f858b | |||
| d67e33a41d | |||
| 4daff41b3e | |||
| f92418ea4b | |||
| 89c912a35b | |||
| 09ad917e5d | |||
| 5c377dc0b6 | |||
| 8bdaf05223 | |||
| e920bd954c | |||
| e395ac87c5 | |||
| 026a4dfb27 | |||
| df18370bde | |||
| 80a66136ce | |||
| 1f8d47f6c3 | |||
| b750cc3c67 | |||
| b618fcc6da | |||
| f763c7515a | |||
| c7d5cb48ac | |||
| 39470d7dbf | |||
| 4328de21ef | |||
| a3a0e8c7a2 | |||
| 210c73a831 | |||
| edaeae386e | |||
| be66ea354e | |||
| d7c1ffe3cc | |||
| 240ad7dc7e | |||
| bb5fe9c380 | |||
| 1347aacbc5 | |||
| 8880647360 | |||
| 717bccbf3f | |||
| 018441ea0b | |||
| 336bb88ca4 | |||
| 811fc40d79 | |||
| e05209ba3c | |||
| 623095473e | |||
| f47f1b175a | |||
| 3b1d291037 | |||
| 2abc9808e2 | |||
| 41dd7d0b64 | |||
| 20f4e780bc | |||
| da43c940f2 | |||
| a9ca8d36bc | |||
| 1980843ac0 | |||
| 96f6752bbe | |||
| 04b9427cdf | |||
| eab939928f | |||
| d3148ab89d | |||
| f3b7b02e77 | |||
| 687db37daf | |||
| 415446e3bb | |||
| 0afb6b9c5b | |||
| 9f4185dff6 | |||
| 772a33896d | |||
| afc49a7a2a | |||
| 3c621187a7 | |||
| 3f0a7a2227 | |||
| f1dbea190b | |||
| 893b820e24 | |||
| 830da43193 | |||
| c43cca1aae | |||
| 49d1d607ce | |||
| 67feaacf5a | |||
| 45f61533ee | |||
| add904cc41 | |||
| e6a9185d11 | |||
| 669107a99f | |||
| 4805e68fcd | |||
| a693bfdc94 | |||
| be9b3f76d2 | |||
| ed4fcf9944 | |||
| a688e33e33 | |||
| 62d4806b95 | |||
| ed02ba02a8 | |||
| efddaf50f2 | |||
| d4aaf61091 | |||
| fa346b528e | |||
| 4a9ccc7c7a | |||
| 76cf08830b | |||
| 2cbb7fb29e | |||
| c55db308a1 | |||
| 2a837227d5 | |||
| b583780cfc | |||
| 599dd4827b | |||
| 45f489dcb6 | |||
| f16053c475 | |||
| c603b3fcb0 | |||
| d0a4eeb2b2 | |||
| 5dd2e83389 | |||
| aa44a40e59 | |||
| cae4756747 | |||
| 5fc03e48a1 | |||
| 06f2c9ecc2 | |||
| ac06d35c10 | |||
| c5a40702b9 | |||
| 468b7f2c2e | |||
| 273c66f5d5 | |||
| 6d5b690450 | |||
| a70092c6f4 | |||
| 7a617a4f8c | |||
| 441df4090f | |||
| e8384338f8 | |||
| b0790ea145 | |||
| 9588fc0475 | |||
| 177ff513ee | |||
| cf1c4403c1 | |||
| 23c5a1a23e | |||
| 32739821ba | |||
| 000caf4dd2 | |||
| fc025c6bd3 | |||
| db9f4504db | |||
| bb23a12be3 | |||
| a865c4d34b | |||
| 0c2df45337 | |||
| a2a42f66a2 | |||
| 51c7b03ff8 | |||
| ddfbcc5e58 | |||
| 997562d174 | |||
| df6f2af756 | |||
| 041be961c4 | |||
| 36013a3a57 | |||
| dc1ce94145 | |||
| 2261528580 | |||
| 23301764ee | |||
| aa9724102b | |||
| 9395e081f0 | |||
| bd1d6b7be9 | |||
| dabb44635e | |||
| 420588860a | |||
| 312d68286e | |||
| bedffbfad7 | |||
| 6a3cd0a60d | |||
| 356d3d4d3e | |||
| 41e2b08bcc | |||
| 731ab97209 | |||
| a59de65130 | |||
| 9b6544df46 | |||
| 7221af75eb | |||
| 66f41179ba | |||
| ed32a31819 | |||
| 33be7182d8 | |||
| 3cd08da3b6 | |||
| dfd80021b9 | |||
| d64a24454d | |||
| 0ed8c2373d | |||
| b8a1e5b5c0 | |||
| 5d6a52494e | |||
| 85a1dd3053 | |||
| 63499df99f | |||
| e70041fefa | |||
| 1af90cd9e7 | |||
| b52811d66e | |||
| 7e63611416 | |||
| d41e358c6a | |||
| 9fd30a1994 | |||
| 471d3deec5 | |||
| c7f059b6d7 | |||
| 6af695d74e | |||
| fd272ead37 | |||
| 6c5377d9fa | |||
| ce414d92a2 | |||
| 5032cccf38 | |||
| 9f7a3082cb | |||
| 359cd94532 | |||
| 432705c570 | |||
| 2065350698 | |||
| 285bb42b09 | |||
| e9fbd0c65f | |||
| 835203706d | |||
| 0e208cc320 | |||
| ee2cb0c989 | |||
| 37c61a0406 | |||
| fa73a28324 | |||
| d945b103ca | |||
| 8bc0da5188 | |||
| 2e68d227a0 | |||
| b8245b00b6 | |||
| 462e818078 | |||
| e4582b7d25 | |||
| 00eef6e45a | |||
| 9498d428cd | 
							
								
								
									
										16
									
								
								README.md
									
									
									
									
									
								
							
							
						
						| @@ -1,16 +0,0 @@ | ||||
| # surface | ||||
|  | ||||
| A new Flutter project. | ||||
|  | ||||
| ## Getting Started | ||||
|  | ||||
| This project is a starting point for a Flutter application. | ||||
|  | ||||
| A few resources to get you started if this is your first Flutter project: | ||||
|  | ||||
| - [Lab: Write your first Flutter app](https://docs.flutter.dev/get-started/codelab) | ||||
| - [Cookbook: Useful Flutter samples](https://docs.flutter.dev/cookbook) | ||||
|  | ||||
| For help getting started with Flutter development, view the | ||||
| [online documentation](https://docs.flutter.dev/), which offers tutorials, | ||||
| samples, guidance on mobile development, and a full API reference. | ||||
| @@ -9,6 +9,13 @@ | ||||
| # packages, and plugins designed to encourage good coding practices. | ||||
| include: package:flutter_lints/flutter.yaml | ||||
|  | ||||
| analyzer: | ||||
|   exclude: | ||||
|     - "**/*.g.dart" | ||||
|     - "**/*.freezed.dart" | ||||
|   errors: | ||||
|     invalid_annotation_target: ignore # Due to freezed + json_serializable issue, ref https://github.com/rrousselGit/freezed/issues/488#issuecomment-894358980 | ||||
|  | ||||
| linter: | ||||
|   # The lint rules applied to this project can be customized in the | ||||
|   # section below to disable rules from the `package:flutter_lints/flutter.yaml` | ||||
|   | ||||
| @@ -1,40 +1,66 @@ | ||||
| plugins { | ||||
|     id "com.android.application" | ||||
|     // START: FlutterFire Configuration | ||||
|     id 'com.google.gms.google-services' | ||||
|     id 'com.google.firebase.crashlytics' | ||||
|     // END: FlutterFire Configuration | ||||
|     id "kotlin-android" | ||||
|     // The Flutter Gradle Plugin must be applied after the Android and Kotlin Gradle plugins. | ||||
|     id "dev.flutter.flutter-gradle-plugin" | ||||
| } | ||||
|  | ||||
| dependencies { | ||||
|     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' | ||||
|     implementation 'io.coil-kt.coil3:coil-compose:3.0.4' | ||||
|     implementation 'io.coil-kt.coil3:coil-network-okhttp:3.0.4' | ||||
| } | ||||
|  | ||||
| android { | ||||
|     buildFeatures { | ||||
|         compose true | ||||
|     } | ||||
|  | ||||
|     namespace = "dev.solsynth.solian" | ||||
|     compileSdk = flutter.compileSdkVersion | ||||
|     ndkVersion = flutter.ndkVersion | ||||
|     ndkVersion = "27.0.12077973" | ||||
|  | ||||
|     compileOptions { | ||||
|         sourceCompatibility = JavaVersion.VERSION_1_8 | ||||
|         targetCompatibility = JavaVersion.VERSION_1_8 | ||||
|         sourceCompatibility JavaVersion.VERSION_17 | ||||
|         targetCompatibility JavaVersion.VERSION_17 | ||||
|     } | ||||
|  | ||||
|     composeOptions { | ||||
|         kotlinCompilerExtensionVersion = "1.4.8" | ||||
|     } | ||||
|  | ||||
|     kotlinOptions { | ||||
|         jvmTarget = JavaVersion.VERSION_1_8 | ||||
|         jvmTarget = JavaVersion.VERSION_17 | ||||
|     } | ||||
|  | ||||
|     defaultConfig { | ||||
|         // TODO: Specify your own unique Application ID (https://developer.android.com/studio/build/application-id.html). | ||||
|         applicationId = "dev.solsynth.solian" | ||||
|         // You can update the following values to match your application needs. | ||||
|         // For more information, see: https://flutter.dev/to/review-gradle-config. | ||||
|         minSdk = flutter.minSdkVersion | ||||
|         minSdk = 26 | ||||
|         targetSdk = flutter.targetSdkVersion | ||||
|         versionCode = flutter.versionCode | ||||
|         versionName = flutter.versionName | ||||
|     } | ||||
|  | ||||
|     buildTypes { | ||||
|         debug { | ||||
|             debuggable true | ||||
|  | ||||
|             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 | ||||
|  | ||||
|             proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro' | ||||
|         } | ||||
|     } | ||||
| } | ||||
|   | ||||
							
								
								
									
										29
									
								
								android/app/google-services.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,29 @@ | ||||
| { | ||||
|   "project_info": { | ||||
|     "project_number": "961776991058", | ||||
|     "project_id": "solian-0x001", | ||||
|     "storage_bucket": "solian-0x001.firebasestorage.app" | ||||
|   }, | ||||
|   "client": [ | ||||
|     { | ||||
|       "client_info": { | ||||
|         "mobilesdk_app_id": "1:961776991058:android:a8d3f7995b0b8e86f4188b", | ||||
|         "android_client_info": { | ||||
|           "package_name": "dev.solsynth.solian" | ||||
|         } | ||||
|       }, | ||||
|       "oauth_client": [], | ||||
|       "api_key": [ | ||||
|         { | ||||
|           "current_key": "AIzaSyDvFNudXYs29uDtcCv6pFR8h5tXBs90FYk" | ||||
|         } | ||||
|       ], | ||||
|       "services": { | ||||
|         "appinvite_service": { | ||||
|           "other_platform_oauth_client": [] | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|   ], | ||||
|   "configuration_version": "1" | ||||
| } | ||||
| @@ -1,28 +1,91 @@ | ||||
| <manifest xmlns:android="http://schemas.android.com/apk/res/android"> | ||||
|     <uses-feature android:name="android.hardware.camera" /> | ||||
|     <uses-feature android:name="android.hardware.camera.autofocus" /> | ||||
|     <uses-permission android:name="android.permission.INTERNET" /> | ||||
|     <uses-permission android:name="android.permission.CAMERA" /> | ||||
|     <uses-permission android:name="android.permission.RECORD_AUDIO" /> | ||||
|     <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" /> | ||||
|     <uses-permission android:name="android.permission.CHANGE_NETWORK_STATE" /> | ||||
|     <uses-permission android:name="android.permission.MODIFY_AUDIO_SETTINGS" /> | ||||
|     <uses-permission android:name="android.permission.BLUETOOTH" android:maxSdkVersion="30" /> | ||||
|     <uses-permission android:name="android.permission.BLUETOOTH_ADMIN" android:maxSdkVersion="30" /> | ||||
|     <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" android:maxSdkVersion="29" /> | ||||
|     <uses-permission android:name="android.permission.POST_NOTIFICATIONS"/> | ||||
|     <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/> | ||||
|  | ||||
|     <application | ||||
|         android:label="surface" | ||||
|         android:label="Solian" | ||||
|         android:name="${applicationName}" | ||||
|         android:icon="@mipmap/ic_launcher"> | ||||
|         android:icon="@mipmap/ic_launcher" | ||||
|         android:requestLegacyExternalStorage="true"> | ||||
|         <activity | ||||
|             android:name=".MainActivity" | ||||
|             android:exported="true" | ||||
|             android:launchMode="singleTop" | ||||
|             android:launchMode="singleTask" | ||||
|             android:taskAffinity="" | ||||
|             android:theme="@style/LaunchTheme" | ||||
|             android:configChanges="orientation|keyboardHidden|keyboard|screenSize|smallestScreenSize|locale|layoutDirection|fontScale|screenLayout|density|uiMode" | ||||
|             android:hardwareAccelerated="true" | ||||
|             android:windowSoftInputMode="adjustResize"> | ||||
|             <!-- Widgets Indents --> | ||||
|             <intent-filter> | ||||
|                 <action android:name="es.antonborri.home_widget.action.LAUNCH" /> | ||||
|             </intent-filter> | ||||
|  | ||||
|             <!-- Sharing Intents --> | ||||
|             <intent-filter> | ||||
|                 <action android:name="android.intent.action.VIEW" /> | ||||
|                 <category android:name="android.intent.category.DEFAULT" /> | ||||
|                 <category android:name="android.intent.category.BROWSABLE" /> | ||||
|                 <data | ||||
|                     android:scheme="https" | ||||
|                     android:host="sn.solsynth.dev" | ||||
|                     android:pathPrefix="/invite"/> | ||||
|             </intent-filter> | ||||
|             <intent-filter> | ||||
|                 <action android:name="android.intent.action.VIEW" /> | ||||
|                 <category android:name="android.intent.category.DEFAULT" /> | ||||
|                 <data | ||||
|                     android:mimeType="*/*" | ||||
|                     android:scheme="content" /> | ||||
|             </intent-filter> | ||||
|             <intent-filter> | ||||
|                 <action android:name="android.intent.action.SEND" /> | ||||
|                 <category android:name="android.intent.category.DEFAULT" /> | ||||
|                 <data android:mimeType="text/*" /> | ||||
|             </intent-filter> | ||||
|             <intent-filter> | ||||
|                 <action android:name="android.intent.action.SEND" /> | ||||
|                 <category android:name="android.intent.category.DEFAULT" /> | ||||
|                 <data android:mimeType="image/*" /> | ||||
|             </intent-filter> | ||||
|             <intent-filter> | ||||
|                 <action android:name="android.intent.action.SEND_MULTIPLE" /> | ||||
|                 <category android:name="android.intent.category.DEFAULT" /> | ||||
|                 <data android:mimeType="image/*" /> | ||||
|             </intent-filter> | ||||
|             <intent-filter> | ||||
|                 <action android:name="android.intent.action.SEND" /> | ||||
|                 <category android:name="android.intent.category.DEFAULT" /> | ||||
|                 <data android:mimeType="video/*" /> | ||||
|             </intent-filter> | ||||
|             <intent-filter> | ||||
|                 <action android:name="android.intent.action.SEND_MULTIPLE" /> | ||||
|                 <category android:name="android.intent.category.DEFAULT" /> | ||||
|                 <data android:mimeType="video/*" /> | ||||
|             </intent-filter> | ||||
|  | ||||
|             <!-- Specifies an Android theme to apply to this Activity as soon as | ||||
|                  the Android process has started. This theme is visible to the user | ||||
|                  while the Flutter UI initializes. After that, this theme continues | ||||
|                  to determine the Window background behind the Flutter UI. --> | ||||
|             <meta-data | ||||
|               android:name="io.flutter.embedding.android.NormalTheme" | ||||
|               android:resource="@style/NormalTheme" | ||||
|               /> | ||||
|                 android:name="io.flutter.embedding.android.NormalTheme" | ||||
|                 android:resource="@style/NormalTheme" | ||||
|             /> | ||||
|             <intent-filter> | ||||
|                 <action android:name="android.intent.action.MAIN"/> | ||||
|                 <category android:name="android.intent.category.LAUNCHER"/> | ||||
|                 <action android:name="android.intent.action.MAIN" /> | ||||
|                 <category android:name="android.intent.category.LAUNCHER" /> | ||||
|             </intent-filter> | ||||
|         </activity> | ||||
|         <!-- Don't delete the meta-data below. | ||||
| @@ -30,7 +93,30 @@ | ||||
|         <meta-data | ||||
|             android:name="flutterEmbedding" | ||||
|             android:value="2" /> | ||||
|  | ||||
|         <!-- Widgets --> | ||||
|         <receiver android:name=".widgets.CheckInWidgetReceiver" | ||||
|             android:label="Check In" | ||||
|             android:exported="true"> | ||||
|             <intent-filter> | ||||
|                 <action android:name="android.appwidget.action.APPWIDGET_UPDATE" /> | ||||
|             </intent-filter> | ||||
|             <meta-data | ||||
|                 android:name="android.appwidget.provider" | ||||
|                 android:resource="@xml/check_in_widget" /> | ||||
|         </receiver> | ||||
|         <receiver android:name=".widgets.RandomPostWidgetReceiver" | ||||
|             android:label="Random Post" | ||||
|             android:exported="true"> | ||||
|             <intent-filter> | ||||
|                 <action android:name="android.appwidget.action.APPWIDGET_UPDATE" /> | ||||
|             </intent-filter> | ||||
|             <meta-data | ||||
|                 android:name="android.appwidget.provider" | ||||
|                 android:resource="@xml/random_post_widget" /> | ||||
|         </receiver> | ||||
|     </application> | ||||
|  | ||||
|     <!-- Required to query activities that can process text, see: | ||||
|          https://developer.android.com/training/package-visibility and | ||||
|          https://developer.android.com/reference/android/content/Intent#ACTION_PROCESS_TEXT. | ||||
| @@ -38,8 +124,8 @@ | ||||
|          In particular, this is used by the Flutter engine in io.flutter.plugin.text.ProcessTextPlugin. --> | ||||
|     <queries> | ||||
|         <intent> | ||||
|             <action android:name="android.intent.action.PROCESS_TEXT"/> | ||||
|             <data android:mimeType="text/plain"/> | ||||
|             <action android:name="android.intent.action.PROCESS_TEXT" /> | ||||
|             <data android:mimeType="text/plain" /> | ||||
|         </intent> | ||||
|     </queries> | ||||
| </manifest> | ||||
|   | ||||
| Before Width: | Height: | Size: 13 KiB After Width: | Height: | Size: 17 KiB | 
| @@ -0,0 +1,6 @@ | ||||
| package dev.solsynth.solian.data | ||||
|  | ||||
| import androidx.annotation.Keep | ||||
|  | ||||
| @Keep | ||||
| data class SolarPagination<T>(val count: Int, val data: List<T>) | ||||
| @@ -0,0 +1,35 @@ | ||||
| package dev.solsynth.solian.data | ||||
|  | ||||
| import androidx.annotation.Keep | ||||
| import java.time.Instant | ||||
|  | ||||
| @Keep | ||||
| data class SolarPost( | ||||
|     val id: Int, | ||||
|     val body: SolarPostBody, | ||||
|     val publisher: SolarPublisher, | ||||
|     val publisherId: Int, | ||||
|     val createdAt: Instant, | ||||
|     val updatedAt: Instant, | ||||
|     val editedAt: Instant?, | ||||
|     val publishedAt: Instant? | ||||
| ) | ||||
|  | ||||
| @Keep | ||||
| data class SolarPostBody( | ||||
|     val content: String?, | ||||
|     val title: String?, | ||||
|     val description: String?, | ||||
| ) | ||||
|  | ||||
| @Keep | ||||
| data class SolarPublisher( | ||||
|     val id: Int, | ||||
|     val name: String, | ||||
|     val nick: String, | ||||
|     val description: String?, | ||||
|     val avatar: String?, | ||||
|     val banner: String?, | ||||
|     val createdAt: Instant, | ||||
|     val updatedAt: Instant | ||||
| ) | ||||
| @@ -0,0 +1,38 @@ | ||||
| package dev.solsynth.solian.data | ||||
|  | ||||
| import androidx.annotation.Keep | ||||
| import com.google.gson.JsonDeserializationContext | ||||
| import com.google.gson.JsonDeserializer | ||||
| import com.google.gson.JsonElement | ||||
| import com.google.gson.JsonParseException | ||||
| import com.google.gson.JsonPrimitive | ||||
| import com.google.gson.JsonSerializationContext | ||||
| import com.google.gson.JsonSerializer | ||||
| import java.lang.reflect.Type | ||||
| import java.time.Instant | ||||
| import java.time.format.DateTimeFormatter | ||||
|  | ||||
| @Keep | ||||
| class InstantAdapter : JsonSerializer<Instant?>, | ||||
|     JsonDeserializer<Instant?> { | ||||
|     override fun serialize( | ||||
|         src: Instant?, | ||||
|         typeOfSrc: Type?, | ||||
|         context: JsonSerializationContext? | ||||
|     ): JsonElement { | ||||
|         return JsonPrimitive(formatter.format(src)) | ||||
|     } | ||||
|  | ||||
|     @Throws(JsonParseException::class) | ||||
|     override fun deserialize( | ||||
|         json: JsonElement, | ||||
|         typeOfT: Type?, | ||||
|         context: JsonDeserializationContext? | ||||
|     ): Instant { | ||||
|         return Instant.parse(json.asString) | ||||
|     } | ||||
|  | ||||
|     companion object { | ||||
|         private val formatter: DateTimeFormatter = DateTimeFormatter.ISO_INSTANT | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,19 @@ | ||||
| package dev.solsynth.solian.data | ||||
|  | ||||
| import androidx.annotation.Keep | ||||
| import java.time.Instant | ||||
|  | ||||
| @Keep | ||||
| data class SolarUser( | ||||
|     val id: Int, | ||||
|     val name: String, | ||||
|     val nick: String | ||||
| ) | ||||
|  | ||||
| @Keep | ||||
| data class SolarCheckInRecord( | ||||
|     val id: Int, | ||||
|     val resultTier: Int, | ||||
|     val resultExperience: Int, | ||||
|     val createdAt: Instant | ||||
| ) | ||||
| @@ -0,0 +1,105 @@ | ||||
| import android.content.Context | ||||
| import android.net.Uri | ||||
| import androidx.compose.runtime.Composable | ||||
| import androidx.compose.ui.graphics.Color | ||||
| import androidx.compose.ui.unit.dp | ||||
| import androidx.compose.ui.unit.sp | ||||
| import androidx.glance.GlanceId | ||||
| import androidx.glance.GlanceModifier | ||||
| import androidx.glance.action.clickable | ||||
| import androidx.glance.appwidget.GlanceAppWidget | ||||
| import androidx.glance.appwidget.provideContent | ||||
| import androidx.glance.background | ||||
| import androidx.glance.currentState | ||||
| import androidx.glance.layout.Alignment | ||||
| import androidx.glance.layout.Column | ||||
| import androidx.glance.layout.Row | ||||
| import androidx.glance.layout.Spacer | ||||
| import androidx.glance.layout.fillMaxHeight | ||||
| import androidx.glance.layout.fillMaxWidth | ||||
| import androidx.glance.layout.height | ||||
| import androidx.glance.layout.padding | ||||
| import androidx.glance.state.GlanceStateDefinition | ||||
| import androidx.glance.text.FontFamily | ||||
| import androidx.glance.text.Text | ||||
| import androidx.glance.text.TextStyle | ||||
| import com.google.gson.FieldNamingPolicy | ||||
| import com.google.gson.GsonBuilder | ||||
| import dev.solsynth.solian.data.InstantAdapter | ||||
| import dev.solsynth.solian.data.SolarCheckInRecord | ||||
| import java.time.Instant | ||||
| import java.time.LocalDate | ||||
| import java.time.OffsetDateTime | ||||
| import java.time.ZoneId | ||||
| import java.time.format.DateTimeFormatter | ||||
|  | ||||
| class CheckInWidget : GlanceAppWidget() { | ||||
|     override val stateDefinition: GlanceStateDefinition<*>? | ||||
|         get() = HomeWidgetGlanceStateDefinition() | ||||
|  | ||||
|     override suspend fun provideGlance(context: Context, id: GlanceId) { | ||||
|         provideContent { | ||||
|             GlanceContent(context, currentState()) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Composable | ||||
|     private fun GlanceContent(context: Context, currentState: HomeWidgetGlanceState) { | ||||
|         val gson = | ||||
|             GsonBuilder() | ||||
|                 .setFieldNamingPolicy(FieldNamingPolicy.LOWER_CASE_WITH_UNDERSCORES) | ||||
|                 .registerTypeAdapter(Instant::class.java, InstantAdapter()) | ||||
|                 .create() | ||||
|         val resultTierSymbols = listOf("大凶", "凶", "中平", "吉", "大吉") | ||||
|  | ||||
|         val prefs = currentState.preferences | ||||
|         val checkInRaw = prefs.getString("pas_check_in_record", null) | ||||
|  | ||||
|         Column( | ||||
|             modifier = GlanceModifier | ||||
|                 .fillMaxWidth() | ||||
|                 .fillMaxHeight() | ||||
|                 .background(Color.White) | ||||
|                 .padding(16.dp) | ||||
|         ) { | ||||
|             if (checkInRaw != null) { | ||||
|                 val checkIn: SolarCheckInRecord = | ||||
|                     gson.fromJson(checkInRaw, SolarCheckInRecord::class.java) | ||||
|                 val dateFormatter = DateTimeFormatter.ofPattern("EEE, MM/dd") | ||||
|  | ||||
|                 val checkDate = checkIn.createdAt.atZone(ZoneId.of("UTC")).toLocalDate() | ||||
|                 val currentDate = LocalDate.now() | ||||
|                 if (checkDate.isEqual(currentDate)) { | ||||
|                     Column { | ||||
|                         Text( | ||||
|                             text = resultTierSymbols[checkIn.resultTier], | ||||
|                             style = TextStyle(fontSize = 25.sp, fontFamily = FontFamily.Serif) | ||||
|                         ) | ||||
|                         Text( | ||||
|                             text = "+${checkIn.resultExperience} EXP", | ||||
|                             style = TextStyle(fontSize = 15.sp, fontFamily = FontFamily.Monospace) | ||||
|                         ) | ||||
|                     } | ||||
|                     Spacer(modifier = GlanceModifier.height(8.dp)) | ||||
|                     Row(horizontalAlignment = Alignment.CenterHorizontally) { | ||||
|                         Text( | ||||
|                             text = OffsetDateTime.ofInstant( | ||||
|                                 checkIn.createdAt, | ||||
|                                 ZoneId.systemDefault() | ||||
|                             ) | ||||
|                                 .format(dateFormatter), | ||||
|                             style = TextStyle(fontSize = 13.sp) | ||||
|                         ) | ||||
|                     } | ||||
|  | ||||
|                     return@Column; | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         Text( | ||||
|             text = "You haven't checked in today", | ||||
|             style = TextStyle(fontSize = 15.sp) | ||||
|         ) | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,8 @@ | ||||
| package dev.solsynth.solian.widgets | ||||
|  | ||||
| import CheckInWidget | ||||
| import HomeWidgetGlanceWidgetReceiver | ||||
|  | ||||
| class CheckInWidgetReceiver : HomeWidgetGlanceWidgetReceiver<CheckInWidget>() { | ||||
|     override val glanceAppWidget = CheckInWidget() | ||||
| } | ||||
| @@ -0,0 +1,202 @@ | ||||
| import android.content.Context | ||||
| import android.graphics.Bitmap | ||||
| import android.graphics.BitmapFactory | ||||
| import android.net.Uri | ||||
| import androidx.compose.runtime.Composable | ||||
| import androidx.compose.ui.graphics.Color | ||||
| import androidx.compose.ui.unit.dp | ||||
| import androidx.compose.ui.unit.sp | ||||
| import androidx.glance.GlanceId | ||||
| import androidx.glance.GlanceModifier | ||||
| import androidx.glance.GlanceTheme | ||||
| import androidx.glance.Image | ||||
| import androidx.glance.ImageProvider | ||||
| import androidx.glance.action.clickable | ||||
| import androidx.glance.appwidget.GlanceAppWidget | ||||
| import androidx.glance.appwidget.cornerRadius | ||||
| import androidx.glance.appwidget.provideContent | ||||
| import androidx.glance.background | ||||
| import androidx.glance.currentState | ||||
| import androidx.glance.layout.Alignment | ||||
| import androidx.glance.layout.Column | ||||
| import androidx.glance.layout.ContentScale | ||||
| import androidx.glance.layout.Row | ||||
| import androidx.glance.layout.Spacer | ||||
| import androidx.glance.layout.fillMaxHeight | ||||
| import androidx.glance.layout.fillMaxSize | ||||
| import androidx.glance.layout.fillMaxWidth | ||||
| import androidx.glance.layout.height | ||||
| import androidx.glance.layout.padding | ||||
| import androidx.glance.layout.width | ||||
| import androidx.glance.state.GlanceStateDefinition | ||||
| import androidx.glance.text.FontFamily | ||||
| import androidx.glance.text.FontWeight | ||||
| import androidx.glance.text.Text | ||||
| import androidx.glance.text.TextStyle | ||||
| import com.google.gson.FieldNamingPolicy | ||||
| import com.google.gson.GsonBuilder | ||||
| import dev.solsynth.solian.MainActivity | ||||
| import dev.solsynth.solian.data.InstantAdapter | ||||
| import dev.solsynth.solian.data.SolarPost | ||||
| import es.antonborri.home_widget.actionStartActivity | ||||
| import okhttp3.OkHttpClient | ||||
| import okhttp3.Request | ||||
| import okhttp3.Response | ||||
| import okio.IOException | ||||
| import java.time.Instant | ||||
| import java.time.LocalDateTime | ||||
| import java.time.ZoneId | ||||
| import java.time.format.DateTimeFormatter | ||||
|  | ||||
| class RandomPostWidget : GlanceAppWidget() { | ||||
|     override val stateDefinition: GlanceStateDefinition<*>? | ||||
|         get() = HomeWidgetGlanceStateDefinition() | ||||
|  | ||||
|     private val defaultUrl = "https://api.sn.solsynth.dev" | ||||
|  | ||||
|     override suspend fun provideGlance(context: Context, id: GlanceId) { | ||||
|         provideContent { | ||||
|             GlanceTheme { | ||||
|                 GlanceContent(context, currentState(), null) | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private val client = OkHttpClient() | ||||
|  | ||||
|     private fun resizeBitmap(bitmap: Bitmap, maxWidth: Int, maxHeight: Int): Bitmap { | ||||
|         val aspectRatio = bitmap.width.toFloat() / bitmap.height.toFloat() | ||||
|         val newWidth = if (bitmap.width > maxWidth) maxWidth else bitmap.width | ||||
|         val newHeight = (newWidth / aspectRatio).toInt() | ||||
|         val resizedBitmap = Bitmap.createScaledBitmap(bitmap, newWidth, newHeight, true) | ||||
|         return resizedBitmap | ||||
|     } | ||||
|  | ||||
|     private fun loadImageFromUrl(url: String): Bitmap? { | ||||
|         val request = Request.Builder().url(url).build() | ||||
|  | ||||
|         return try { | ||||
|             val response: Response = client.newCall(request).execute() | ||||
|             val inputStream = response.body?.byteStream() | ||||
|             val bitmap = BitmapFactory.decodeStream(inputStream) | ||||
|             resizeBitmap(bitmap, 120, 120) | ||||
|         } catch (e: IOException) { | ||||
|             e.printStackTrace() | ||||
|             null | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Composable | ||||
|     private fun GlanceContent( | ||||
|         context: Context, | ||||
|         currentState: HomeWidgetGlanceState, | ||||
|         avatar: Bitmap? | ||||
|     ) { | ||||
|         val prefs = currentState.preferences | ||||
|         val postRaw = prefs.getString("int_random_post", null) | ||||
|  | ||||
|         val gson = | ||||
|             GsonBuilder() | ||||
|                 .setFieldNamingPolicy(FieldNamingPolicy.LOWER_CASE_WITH_UNDERSCORES) | ||||
|                 .registerTypeAdapter(Instant::class.java, InstantAdapter()) | ||||
|                 .create() | ||||
|  | ||||
|         val data: SolarPost? = postRaw?.let { postRaw -> | ||||
|             gson.fromJson(postRaw, SolarPost::class.java) | ||||
|         } ?: null; | ||||
|  | ||||
|         Column( | ||||
|             modifier = GlanceModifier | ||||
|                 .fillMaxWidth() | ||||
|                 .fillMaxHeight() | ||||
|                 .background(Color.White) | ||||
|                 .padding(16.dp) | ||||
|                 .clickable( | ||||
|                     onClick = actionStartActivity<MainActivity>( | ||||
|                         context, | ||||
|                         Uri.parse("https://sn.solsynth.dev/posts/${data!!.id}") | ||||
|                     ) | ||||
|                 ) | ||||
|         ) { | ||||
|             if (data != null) { | ||||
|                 Row(verticalAlignment = Alignment.CenterVertically) { | ||||
|                     if (avatar != null) { | ||||
|                         Image( | ||||
|                             provider = ImageProvider(bitmap = avatar), | ||||
|                             contentDescription = null, | ||||
|                             modifier = GlanceModifier.width(36.dp).height(36.dp) | ||||
|                                 .cornerRadius(18.dp), | ||||
|                             contentScale = ContentScale.Crop | ||||
|                         ) | ||||
|                         Spacer(modifier = GlanceModifier.width(8.dp)) | ||||
|                     } | ||||
|  | ||||
|                     Text( | ||||
|                         text = data.publisher.nick, | ||||
|                         style = TextStyle(fontSize = 15.sp) | ||||
|                     ) | ||||
|                     Spacer(modifier = GlanceModifier.width(8.dp)) | ||||
|                     Text( | ||||
|                         text = "@${data.publisher.name}", | ||||
|                         style = TextStyle(fontSize = 13.sp, fontFamily = FontFamily.Monospace) | ||||
|                     ) | ||||
|                 } | ||||
|  | ||||
|                 Spacer(modifier = GlanceModifier.height(8.dp)) | ||||
|  | ||||
|                 if (data.body.title != null) { | ||||
|                     Text( | ||||
|                         text = data.body.title, | ||||
|                         style = TextStyle(fontSize = 25.sp) | ||||
|                     ) | ||||
|                 } | ||||
|                 if (data.body.description != null) { | ||||
|                     Text( | ||||
|                         text = data.body.description, | ||||
|                         style = TextStyle(fontSize = 19.sp) | ||||
|                     ) | ||||
|                 } | ||||
|  | ||||
|                 if (data.body.title != null || data.body.description != null) { | ||||
|                     Spacer(modifier = GlanceModifier.height(8.dp)) | ||||
|                 } | ||||
|  | ||||
|                 Text( | ||||
|                     text = data.body.content ?: "No content", | ||||
|                     style = TextStyle(fontSize = 15.sp), | ||||
|                 ) | ||||
|  | ||||
|                 Spacer(modifier = GlanceModifier.height(8.dp)) | ||||
|  | ||||
|  | ||||
|                 Text( | ||||
|                     LocalDateTime.ofInstant(data.createdAt, ZoneId.systemDefault()) | ||||
|                         .format(DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm")), | ||||
|                     style = TextStyle(fontSize = 13.sp), | ||||
|                 ) | ||||
|  | ||||
|                 Text( | ||||
|                     "#${data.id}", | ||||
|                     style = TextStyle(fontSize = 11.sp, fontWeight = FontWeight.Bold), | ||||
|                 ) | ||||
|  | ||||
|                 return@Column; | ||||
|             } | ||||
|  | ||||
|             Column( | ||||
|                 modifier = GlanceModifier.fillMaxSize(), | ||||
|                 verticalAlignment = Alignment.Vertical.CenterVertically, | ||||
|                 horizontalAlignment = Alignment.Horizontal.CenterHorizontally | ||||
|             ) { | ||||
|                 Text( | ||||
|                     text = "Unable to fetch post", | ||||
|                     style = TextStyle(fontSize = 17.sp, fontWeight = FontWeight.Bold) | ||||
|                 ) | ||||
|                 Text( | ||||
|                     text = "Check your internet connection", | ||||
|                     style = TextStyle(fontSize = 15.sp) | ||||
|                 ) | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,8 @@ | ||||
| package dev.solsynth.solian.widgets | ||||
|  | ||||
| import RandomPostWidget | ||||
| import HomeWidgetGlanceWidgetReceiver | ||||
|  | ||||
| class RandomPostWidgetReceiver : HomeWidgetGlanceWidgetReceiver<RandomPostWidget>() { | ||||
|     override val glanceAppWidget = RandomPostWidget() | ||||
| } | ||||
| @@ -1,6 +1,5 @@ | ||||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <adaptive-icon xmlns:android="http://schemas.android.com/apk/res/android"> | ||||
|   <background android:drawable="@color/ic_launcher_background"/> | ||||
|   <foreground android:drawable="@mipmap/ic_launcher_foreground"/> | ||||
|   <monochrome android:drawable="@mipmap/ic_launcher_monochrome"/> | ||||
| </adaptive-icon> | ||||
|     <background android:drawable="@color/ic_launcher_background"/> | ||||
|     <foreground android:drawable="@mipmap/ic_launcher_foreground"/> | ||||
| </adaptive-icon> | ||||
| @@ -0,0 +1,5 @@ | ||||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <adaptive-icon xmlns:android="http://schemas.android.com/apk/res/android"> | ||||
|     <background android:drawable="@color/ic_launcher_background"/> | ||||
|     <foreground android:drawable="@mipmap/ic_launcher_foreground"/> | ||||
| </adaptive-icon> | ||||
| @@ -1,3 +0,0 @@ | ||||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <adaptive-icon xmlns:android="http://schemas.android.com/apk/res/android"> | ||||
| </adaptive-icon> | ||||
| Before Width: | Height: | Size: 1.5 KiB | 
							
								
								
									
										
											BIN
										
									
								
								android/app/src/main/res/mipmap-hdpi/ic_launcher.webp
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 952 B | 
| Before Width: | Height: | Size: 3.7 KiB | 
							
								
								
									
										
											BIN
										
									
								
								android/app/src/main/res/mipmap-hdpi/ic_launcher_foreground.webp
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 872 B | 
							
								
								
									
										
											BIN
										
									
								
								android/app/src/main/res/mipmap-hdpi/ic_launcher_round.webp
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 2.1 KiB | 
| Before Width: | Height: | Size: 1.7 KiB | 
| Before Width: | Height: | Size: 1017 B | 
							
								
								
									
										
											BIN
										
									
								
								android/app/src/main/res/mipmap-mdpi/ic_launcher.webp
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 644 B | 
| Before Width: | Height: | Size: 2.4 KiB | 
							
								
								
									
										
											BIN
										
									
								
								android/app/src/main/res/mipmap-mdpi/ic_launcher_foreground.webp
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 594 B | 
							
								
								
									
										
											BIN
										
									
								
								android/app/src/main/res/mipmap-mdpi/ic_launcher_round.webp
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 1.3 KiB | 
| Before Width: | Height: | Size: 1.1 KiB | 
| Before Width: | Height: | Size: 2.1 KiB | 
							
								
								
									
										
											BIN
										
									
								
								android/app/src/main/res/mipmap-xhdpi/ic_launcher.webp
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 1.2 KiB | 
| Before Width: | Height: | Size: 4.9 KiB | 
| After Width: | Height: | Size: 1.1 KiB | 
							
								
								
									
										
											BIN
										
									
								
								android/app/src/main/res/mipmap-xhdpi/ic_launcher_round.webp
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 3.0 KiB | 
| Before Width: | Height: | Size: 2.3 KiB | 
| Before Width: | Height: | Size: 3.3 KiB | 
							
								
								
									
										
											BIN
										
									
								
								android/app/src/main/res/mipmap-xxhdpi/ic_launcher.webp
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 1.8 KiB | 
| Before Width: | Height: | Size: 7.7 KiB | 
| After Width: | Height: | Size: 1.7 KiB | 
							
								
								
									
										
											BIN
										
									
								
								android/app/src/main/res/mipmap-xxhdpi/ic_launcher_round.webp
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 4.8 KiB | 
| Before Width: | Height: | Size: 3.6 KiB | 
| Before Width: | Height: | Size: 4.4 KiB | 
							
								
								
									
										
											BIN
										
									
								
								android/app/src/main/res/mipmap-xxxhdpi/ic_launcher.webp
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 2.5 KiB | 
| Before Width: | Height: | Size: 11 KiB | 
| After Width: | Height: | Size: 2.4 KiB | 
							
								
								
									
										
											BIN
										
									
								
								android/app/src/main/res/mipmap-xxxhdpi/ic_launcher_round.webp
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 6.9 KiB | 
| Before Width: | Height: | Size: 4.8 KiB | 
							
								
								
									
										7
									
								
								android/app/src/main/res/xml/check_in_widget.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,7 @@ | ||||
| <appwidget-provider xmlns:android="http://schemas.android.com/apk/res/android" | ||||
|     android:initialLayout="@layout/glance_default_loading_layout" | ||||
|     android:minWidth="120dp" | ||||
|     android:minHeight="40dp" | ||||
|     android:resizeMode="horizontal|vertical" | ||||
|     android:updatePeriodMillis="10000"> | ||||
| </appwidget-provider> | ||||
							
								
								
									
										7
									
								
								android/app/src/main/res/xml/random_post_widget.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,7 @@ | ||||
| <appwidget-provider xmlns:android="http://schemas.android.com/apk/res/android" | ||||
|     android:initialLayout="@layout/glance_default_loading_layout" | ||||
|     android:minWidth="240dp" | ||||
|     android:minHeight="40dp" | ||||
|     android:resizeMode="horizontal|vertical" | ||||
|     android:updatePeriodMillis="10000"> | ||||
| </appwidget-provider> | ||||
							
								
								
									
										14
									
								
								android/app/src/proguard-rules.pro
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,14 @@ | ||||
| -keepclassmembers class kotlin.Metadata { *; } | ||||
| -keep class dev.solsynth.solian.** { *; } | ||||
| -keep public class dev.solsynth.solian.data.** { public *; } | ||||
| -keepclassmembers class dev.solsynth.solian.data.** { *; } | ||||
|  | ||||
| -keepattributes *Annotation* | ||||
| -keepattributes Signature | ||||
| -keepattributes EnclosingMethod | ||||
|  | ||||
| -keep class com.google.gson.** { *; } | ||||
|  | ||||
| -keepclassmembers class * { | ||||
|     @com.google.gson.annotations.SerializedName <fields>; | ||||
| } | ||||
| @@ -3,6 +3,15 @@ allprojects { | ||||
|         google() | ||||
|         mavenCentral() | ||||
|     } | ||||
|     configurations.all { | ||||
|         resolutionStrategy { | ||||
|             eachDependency { | ||||
|                 if ((requested.group == "androidx.work") && (requested.name.startsWith("work-runtime"))) { | ||||
|                     useVersion("2.9.1") | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| rootProject.buildDir = "../build" | ||||
|   | ||||
| @@ -2,4 +2,4 @@ distributionBase=GRADLE_USER_HOME | ||||
| distributionPath=wrapper/dists | ||||
| zipStoreBase=GRADLE_USER_HOME | ||||
| zipStorePath=wrapper/dists | ||||
| distributionUrl=https\://services.gradle.org/distributions/gradle-8.3-all.zip | ||||
| distributionUrl=https\://services.gradle.org/distributions/gradle-8.9-all.zip | ||||
|   | ||||
| @@ -18,7 +18,11 @@ pluginManagement { | ||||
|  | ||||
| plugins { | ||||
|     id "dev.flutter.flutter-plugin-loader" version "1.0.0" | ||||
|     id "com.android.application" version "8.1.0" apply false | ||||
|     id "com.android.application" version '8.7.3' apply false | ||||
|     // START: FlutterFire Configuration | ||||
|     id "com.google.gms.google-services" version "4.3.15" apply false | ||||
|     id "com.google.firebase.crashlytics" version "2.8.1" apply false | ||||
|     // END: FlutterFire Configuration | ||||
|     id "org.jetbrains.kotlin.android" version "1.8.22" apply false | ||||
| } | ||||
|  | ||||
|   | ||||
							
								
								
									
										
											BIN
										
									
								
								assets/icon/icon-w-padding.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 44 KiB | 
| @@ -2,6 +2,7 @@ | ||||
|   "nextVersionAlert": "Heavy Development Alert", | ||||
|   "nextVersionNotice": "You are using Solian 2.0 Preview, which is the first version of Solian 2.0. The current stable branch (sn.solsynth.dev) is 1.4. This version is still under heavy development, some features may not be stable, and not all features are supported. You can roll back to 1.4.X version via TestFlight, or continue to experience the new version (sn-next.solsynth.dev).", | ||||
|   "screen": "Screen", | ||||
|   "screenAbout": "About", | ||||
|   "screenHome": "Home", | ||||
|   "screenExplore": "Explore", | ||||
|   "screenAccount": "Account", | ||||
| @@ -14,9 +15,18 @@ | ||||
|   "screenAccountPublisherNew": "New Publisher", | ||||
|   "screenAccountPublisherEdit": "Edit Publisher", | ||||
|   "screenAccountProfileEdit": "Edit Profile", | ||||
|   "screenAbuseReport": "Abuse Reports", | ||||
|   "screenSettings": "Settings", | ||||
|   "screenAlbum": "Album", | ||||
|   "screenChat": "Chat", | ||||
|   "screenChatManage": "Edit Channel", | ||||
|   "screenChatNew": "New Channel", | ||||
|   "screenRealm": "Realm", | ||||
|   "screenRealmManage": "Edit Realm", | ||||
|   "screenRealmNew": "New Realm", | ||||
|   "screenNotification": "Notification", | ||||
|   "screenPostSearch": "Search Posts", | ||||
|   "screenFriend": "Friends", | ||||
|   "dialogOkay": "Okay", | ||||
|   "dialogCancel": "Cancel", | ||||
|   "dialogConfirm": "Confirm", | ||||
| @@ -28,10 +38,12 @@ | ||||
|   "errorRequestNotFound": "The resource that you looking for is not found.", | ||||
|   "errorRequestConnection": "Network connection error, please check your network or the service status.", | ||||
|   "errorRequestUnknown": "Unknown request error, maybe you want to take screenshot and report it to us.", | ||||
|   "unknown": "Unknown", | ||||
|   "prev": "Previous", | ||||
|   "next": "Next", | ||||
|   "edit": "Edit", | ||||
|   "apply": "Apply", | ||||
|   "cancel": "Cancel", | ||||
|   "create": "Create", | ||||
|   "preview": "Preview", | ||||
|   "loading": "Loading...", | ||||
| @@ -41,18 +53,44 @@ | ||||
|   "compress": "Compress", | ||||
|   "report": "Report", | ||||
|   "repost": "Repost", | ||||
|   "replyPost": "Reply", | ||||
|   "reply": "Reply", | ||||
|   "unset": "Unset", | ||||
|   "untitled": "Untitled", | ||||
|   "postDetail": "Post detail", | ||||
|   "postNoun": "Post", | ||||
|   "postReadMore": "Read more", | ||||
|   "postReadEstimate": "Est read time {}", | ||||
|   "postTotalLength": { | ||||
|     "zero": "No character", | ||||
|     "one": "{} character", | ||||
|     "other": "{} characters" | ||||
|   }, | ||||
|   "postVisibility": "Visibility", | ||||
|   "postVisibilityDescription": "Post visibility determines who can see this post.", | ||||
|   "postVisibilityAll": "Everyone", | ||||
|   "postVisibilityFriends": "Friends", | ||||
|   "postVisibilitySelected": "Selected User", | ||||
|   "postVisibilityFiltered": "Unselected User", | ||||
|   "postVisibilityNone": "Only Me", | ||||
|   "postVisibleUsers": "Visible Users", | ||||
|   "postInvisibleUsers": "Invisible Users", | ||||
|   "postSelectedUsers": { | ||||
|     "zero": "No user", | ||||
|     "one": "{} user", | ||||
|     "other": "{} users" | ||||
|   }, | ||||
|   "fieldUsername": "Username", | ||||
|   "fieldNickname": "Nickname", | ||||
|   "fieldEmail": "Email address", | ||||
|   "fieldPassword": "Password", | ||||
|   "fieldDescription": "Description", | ||||
|   "fieldUsernameAlphanumOnly": "Username can only contain alphanumeric characters.", | ||||
|   "fieldUsernameLengthLimit": "Username must be between {} and {} characters.", | ||||
|   "fieldUsernameCannotEditHint": "Username cannot be edited after created", | ||||
|   "fieldUsernameLookupHint": "You can use username, phone number or email to login", | ||||
|   "fieldNicknameLengthLimit": "Nickname must be between {} and {} characters.", | ||||
|   "fieldEmailAddressMustBeValid": "Email address must be an email address.", | ||||
|   "fieldFirstName": "First name", | ||||
|   "fieldLastName": "Last name", | ||||
|   "fieldBirthday": "Birthday", | ||||
| @@ -81,12 +119,26 @@ | ||||
|   "publishersNew": "New Publisher", | ||||
|   "publisherNewSubtitle": "Create a new publisher identity.", | ||||
|   "publisherSyncWithAccount": "Sync with account", | ||||
|   "publisherTotalUpvote": "Upvote", | ||||
|   "publisherTotalDownvote": "Downvote", | ||||
|   "publisherSocialPoint": "Social Point", | ||||
|   "publisherJoinedAt": "Joined at {}", | ||||
|   "publisherSocialPointTotal": { | ||||
|     "zero": "No social point", | ||||
|     "one": "{} social point", | ||||
|     "other": "{} social points" | ||||
|   }, | ||||
|   "publisherAffiliatedBy": "Affiliated by {}", | ||||
|   "publisherRunBy": "Run by {}", | ||||
|   "fieldPublisherBelongToRealm": "Belongs to", | ||||
|   "fieldPublisherBelongToRealmUnset": "Unset Publisher Belongs to Realm", | ||||
|   "writePostTypeStory": "Post a story", | ||||
|   "writePostTypeArticle": "Write an article", | ||||
|   "fieldPostPublisher": "Post publisher", | ||||
|   "fieldPostContent": "What happened?!", | ||||
|   "fieldPostTitle": "Title", | ||||
|   "fieldPostDescription": "Description", | ||||
|   "fieldPostTags": "Tags", | ||||
|   "postPublish": "Publish", | ||||
|   "postPosted": "Post has been posted.", | ||||
|   "postPublishedAt": "Published At", | ||||
| @@ -96,10 +148,20 @@ | ||||
|   "postRepostingNotice": "You're about to repost a post that posted {}.", | ||||
|   "postReact": "React", | ||||
|   "postReactions": "Reactions of Post", | ||||
|   "postReactionPoints": { | ||||
|     "zero": "{} pt", | ||||
|     "one": "{} pt", | ||||
|     "other": "{} pts" | ||||
|   "postReactionUpvote": { | ||||
|     "zero": "0 upvote", | ||||
|     "one": "{} upvote", | ||||
|     "other": "{} upvotes" | ||||
|   }, | ||||
|   "postReactionDownvote": { | ||||
|     "zero": "0 downvote", | ||||
|     "one": "{} downvote", | ||||
|     "other": "{} downvotes" | ||||
|   }, | ||||
|   "postReactionSocialPoint": { | ||||
|     "zero": "0 point", | ||||
|     "one": "{} point", | ||||
|     "other": "{} points" | ||||
|   }, | ||||
|   "postReactCompleted": "Reaction has been added.", | ||||
|   "postReactUncompleted": "Reaction has been removed.", | ||||
| @@ -128,8 +190,270 @@ | ||||
|   "settingsNetworkServerPreset": "Present HyperNet Server", | ||||
|   "settingsNetworkServerPresetDescription": "You can choose one of our preset HyperNet server addresses from the list on the right.", | ||||
|   "settingsNetworkServerSaved": "Server address saved.", | ||||
|   "settingsPerformance": "Performance", | ||||
|   "settingsImageQuality": "Image Quality", | ||||
|   "settingsImageQualityDescription": "Set the image quality, it will affect the decoding speed of the image.", | ||||
|   "settingsImageQualityLowest": "Lowest", | ||||
|   "settingsImageQualityLow": "Low", | ||||
|   "settingsImageQualityMedium": "Medium", | ||||
|   "settingsImageQualityHigh": "High", | ||||
|   "settingsMisc": "Misc", | ||||
|   "settingsMiscAbout": "About", | ||||
|   "settingsMiscAboutDescription": "View the version information of Solian.", | ||||
|   "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" | ||||
|   "sensitiveContentReveal": "Reveal", | ||||
|   "serverConnecting": "Connecting to server...", | ||||
|   "serverDisconnected": "Lost connection from server", | ||||
|   "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", | ||||
|   "fieldChatDescription": "Description", | ||||
|   "fieldChatBelongToRealm": "Belongs to", | ||||
|   "fieldChatBelongToRealmUnset": "Unset Channel Belongs to Realm", | ||||
|   "channelEditingNotice": "You are editing channel {}", | ||||
|   "channelDeleted": "Chat channel {} has been deleted.", | ||||
|   "channelDelete": "Delete channel {}", | ||||
|   "channelDeleteDescription": "Are you sure you want to delete this channel? This operation is irreversible, all messages in this channel will be permanently deleted.", | ||||
|   "channelDetailPersonalRegion": "Personal", | ||||
|   "channelDetailMemberRegion": "Members", | ||||
|   "channelMemberManage": "Manage Member", | ||||
|   "channelMemberManageDescription": "Manage the existing members of this channel.", | ||||
|   "channelMemberAdd": "Add Member", | ||||
|   "channelMemberAddDescription": "Add new member to this channel.", | ||||
|   "channelMemberAdded": "Channel member has been added.", | ||||
|   "fieldMemberRelatedName": "Member name / account ID", | ||||
|   "channelDetailAdminRegion": "Administration", | ||||
|   "channelEditProfile": "Edit Channel Profile", | ||||
|   "channelEdit": "Edit Channel", | ||||
|   "channelEditDescription": "Change the basic information of the channel, metadata, etc.", | ||||
|   "channelProfileEdit": "Edit Channel Profile", | ||||
|   "channelActionDelete": "Delete Channel", | ||||
|   "channelActionDeleteDescription": "Delete the entire channel, and also delete messages in the channel.", | ||||
|   "channelLeave": "Leave Channel {}", | ||||
|   "channelLeaveDescription": "Leave this channel, but the messages in the channel will not be removed.", | ||||
|   "channelActionLeave": "Leave Channel", | ||||
|   "channelActionLeaveDescription": "Delete your profile in this channel.", | ||||
|   "channelNotifyLevel": "Notify Level", | ||||
|   "channelNotifyLevelDescription": "Decide to receive how much notifications from this channel.", | ||||
|   "channelNotifyLevelAll": "All", | ||||
|   "channelNotifyLevelMentioned": "Only Mentioned", | ||||
|   "channelNotifyLevelNone": "Muted", | ||||
|   "channelNotifyLevelApplie": "Channel notify level has been applied.", | ||||
|   "fieldChannelProfileNick": "In-Channel Display Name", | ||||
|   "fieldChannelProfileNickHint": "The nickname to display in the channel, leave blank to use the account display name.", | ||||
|   "fieldRealmAlias": "Realm Alias", | ||||
|   "fieldRealmAliasHint": "The unique realm alias within the site, used to represent the realm in URL, leave blank to auto generate. Should be URL-Safe.", | ||||
|   "fieldRealmName": "Name", | ||||
|   "fieldRealmDescription": "Description", | ||||
|   "realmEditingNotice": "You are editing realm {}", | ||||
|   "realmDeleted": "Realm {} has been deleted.", | ||||
|   "realmDelete": "Delete realm {}", | ||||
|   "realmDeleteDescription": "Are you sure you want to delete this realm? This operation is irreversible, all resources (posts, chat channels, publishers, etc) belonging to this realm will be permanently deleted. Be careful and think twice!", | ||||
|   "realmActionDelete": "Delete Realm", | ||||
|   "realmActionDeleteDescription": "Delete the realm and all its resources.", | ||||
|   "realmEdit": "Edit Realm", | ||||
|   "realmEditDescription": "Edit the basic information of the realm, metadata, etc.", | ||||
|   "realmMemberAdd": "Add Member", | ||||
|   "realmMemberAddDescription": "Add new member to this realm.", | ||||
|   "realmMemberAdded": "Realm member has been added.", | ||||
|   "fieldChatMessage": "Message in {}", | ||||
|   "fieldChatMessageDirect": "Message with {}", | ||||
|   "eventResourceTag": "Event {}", | ||||
|   "messageDelete": "Delete message {}", | ||||
|   "messageDeleteDescription": "Are you sure you want to delete this message? This operation is irreversible. You will leave a record of the deleted message.", | ||||
|   "messageDeleted": "Message {} has been deleted", | ||||
|   "messageEdited": "Message {} has been edited", | ||||
|   "messageEditedHint": "Edited", | ||||
|   "messageUnsupported": "Unsupported message {}", | ||||
|   "messageFileHint": { | ||||
|     "zero": "No attachments", | ||||
|     "one": "{} attachment", | ||||
|     "other": "{} attachments" | ||||
|   }, | ||||
|   "addAttachmentFromAlbum": "Add from album", | ||||
|   "addAttachmentFromClipboard": "Paste file", | ||||
|   "addAttachmentFromCameraPhoto": "Take photo", | ||||
|   "addAttachmentFromCameraVideo": "Take video", | ||||
|   "attachmentPastedImage": "Pasted Image", | ||||
|   "attachmentInsertLink": "Insert Link", | ||||
|   "attachmentSetAsPostThumbnail": "Set as post thumbnail", | ||||
|   "attachmentUnsetAsPostThumbnail": "Unset as post thumbnail", | ||||
|   "attachmentSetThumbnail": "Set thumbnail", | ||||
|   "attachmentUpload": "Upload", | ||||
|   "notification": "Notification", | ||||
|   "notificationUnreadCount": { | ||||
|     "zero": "All notifications read", | ||||
|     "one": "{} unread notification", | ||||
|     "other": "{} unread notifications" | ||||
|   }, | ||||
|   "notificationUnread": "Unread", | ||||
|   "notificationRead": "Read", | ||||
|   "notificationMarkAllRead": "Mark all notifications as read", | ||||
|   "notificationMarkAllReadDescription": "Are you sure you want to mark all notifications as read? This operation is irreversible.", | ||||
|   "notificationMarkAllReadPrompt": { | ||||
|     "zero": "Marked 0 notification as read.", | ||||
|     "one": "Marked {} notification as read.", | ||||
|     "other": "Marked {} notifications as read." | ||||
|   }, | ||||
|   "notificationMarkOneReadPrompt": "Marked notification {} as read.", | ||||
|   "search": "Search", | ||||
|   "postSearchResult": { | ||||
|     "zero": "No results", | ||||
|     "one": "{} result", | ||||
|     "other": "{} results" | ||||
|   }, | ||||
|   "postSearchTook": "Took {}", | ||||
|   "postDelete": "Delete post {}", | ||||
|   "postDeleteDescription": "Are you sure you want to delete this post? This operation is irreversible.", | ||||
|   "postDeleted": "Post {} has been deleted.", | ||||
|   "call": "Call", | ||||
|   "callOngoingNotice": "A call is ongoing", | ||||
|   "callJoin": "Join", | ||||
|   "callResume": "Resume", | ||||
|   "callMicrophone": "Microphone", | ||||
|   "callCamera": "Camera", | ||||
|   "callMicrophoneDisabled": "Microphone is disabled", | ||||
|   "callMicrophoneSelect": "Select a microphone", | ||||
|   "callCameraDisabled": "Camera is disabled", | ||||
|   "callCameraSelect": "Select a camera", | ||||
|   "callDisconnected": "Call has been disconnected", | ||||
|   "callEnded": "Call has been ended", | ||||
|   "callStatusConnected": "Connected", | ||||
|   "callStatusDisconnected": "Disconnected", | ||||
|   "callStatusConnecting": "Connecting", | ||||
|   "callStatusReconnecting": "Reconnecting", | ||||
|   "callDisconnect": "Disconnect", | ||||
|   "callDisconnectDescription": "Are you sure you want to disconnect from the call?", | ||||
|   "callMicrophoneOff": "Turn off microphone", | ||||
|   "callMicrophoneOn": "Turn on microphone", | ||||
|   "callCameraOff": "Turn off camera", | ||||
|   "callCameraOn": "Turn on camera", | ||||
|   "callVideoFlip": "Mirror video", | ||||
|   "callSpeakerphoneToggle": "Toggle speakerphone", | ||||
|   "callScreenOff": "Turn off screen share", | ||||
|   "callScreenOn": "Turn on screen share", | ||||
|   "callMessageEnded": "Call lasted {}", | ||||
|   "callMessageStarted": "Call started", | ||||
|   "dailyCheckIn": "Check In", | ||||
|   "dailyCheckInNone": "You haven't checked in today", | ||||
|   "dailyCheckAction": "Check in right now!", | ||||
|   "dailyCheckDetail": "Can't understand the symbol? Master, help me understand it!", | ||||
|   "dailyCheckDetailTitle": "{}'s fortune details", | ||||
|   "dailyCheckPositiveHint": "Good for {}", | ||||
|   "dailyCheckNegativeHint": "Bad for {}", | ||||
|   "dailyCheckEverythingIsPositive": "Everything going to be awesome!", | ||||
|   "dailyCheckEverythingIsNegative": "Everything may be wrong...", | ||||
|   "dailyCheckPositiveHint1": "Making friends", | ||||
|   "dailyCheckPositiveHint1Description": "Friendship lasts forever", | ||||
|   "dailyCheckPositiveHint2": "Drinking", | ||||
|   "dailyCheckPositiveHint2Description": "Drinking under the moonlight with an imaginary companion", | ||||
|   "dailyCheckPositiveHint3": "Traveling", | ||||
|   "dailyCheckPositiveHint3Description": "A journey of a thousand miles begins with a single step", | ||||
|   "dailyCheckPositiveHint4": "Exercising", | ||||
|   "dailyCheckPositiveHint4Description": "Life lies in movement", | ||||
|   "dailyCheckPositiveHint5": "Learning", | ||||
|   "dailyCheckPositiveHint5Description": "Knowledge knows no bounds; progress every day", | ||||
|   "dailyCheckPositiveHint6": "Planting", | ||||
|   "dailyCheckPositiveHint6Description": "Sow hope, reap the future", | ||||
|   "dailyCheckNegativeHint1": "Eating", | ||||
|   "dailyCheckNegativeHint1Description": "Biting your tongue while eating", | ||||
|   "dailyCheckNegativeHint2": "Taking exams", | ||||
|   "dailyCheckNegativeHint2Description": "The exam covered what you didn't review", | ||||
|   "dailyCheckNegativeHint3": "Catching a bus", | ||||
|   "dailyCheckNegativeHint3Description": "Just missed the bus", | ||||
|   "dailyCheckNegativeHint4": "Shopping", | ||||
|   "dailyCheckNegativeHint4Description": "Bought clothes that don't fit", | ||||
|   "dailyCheckNegativeHint5": "Gaming", | ||||
|   "dailyCheckNegativeHint5Description": "Lost connection at a crucial moment", | ||||
|   "dailyCheckNegativeHint6": "Going out", | ||||
|   "dailyCheckNegativeHint6Description": "Forgot your umbrella and got caught in the rain", | ||||
|   "happyBirthday": "Happy birthday, {}!", | ||||
|   "friendNew": "Add Friend", | ||||
|   "friendRequests": "Friend Requests", | ||||
|   "friendRequestsDescription": { | ||||
|     "zero": "You have no friend request", | ||||
|     "one": "You have {} friend request", | ||||
|     "other": "You have {} friend requests" | ||||
|   }, | ||||
|   "friendBlocklist": "Blocklist", | ||||
|   "friendBlocklistDescription": { | ||||
|     "zero": "You blocked no one", | ||||
|     "one": "You blocked {} user", | ||||
|     "other": "You blocked {} users" | ||||
|   }, | ||||
|   "friendStatusPending": "Pending", | ||||
|   "friendStatusWaiting": "Waiting", | ||||
|   "friendStatusActive": "Friend", | ||||
|   "friendStatusBlocked": "Blocked", | ||||
|   "friendRequestSent": "Friend request has been sent.", | ||||
|   "fieldFriendRelatedName": "Friend name / account ID", | ||||
|   "friendBlock": "Block", | ||||
|   "friendUnblock": "Unblock", | ||||
|   "friendDeleteAction": "Delete", | ||||
|   "friendDelete": "Delete relation with {}", | ||||
|   "friendDeleteDescription": "Are you sure you want to delete the relation with {}? This operation is irreversible.", | ||||
|   "friendRequestAccept": "Accept", | ||||
|   "friendRequestDecline": "Decline", | ||||
|   "subscribe": "Subscribe", | ||||
|   "unsubscribe": "Unsubscribe", | ||||
|   "attachmentUploadBy": "Upload by", | ||||
|   "attachmentShotOn": "Shot on {}", | ||||
|   "accountJoinedAt": "Joined at {}", | ||||
|   "accountBirthday": "Born on {}", | ||||
|   "accountBadge": "Badge", | ||||
|   "badgeCompanyStaff": "Solsynth Staff", | ||||
|   "badgeSiteMigration": "Solar Network Native", | ||||
|   "accountStatus": "Status", | ||||
|   "accountStatusOnline": "Online", | ||||
|   "accountStatusOffline": "Offline", | ||||
|   "accountStatusLastSeen": "Last seen at {}", | ||||
|   "postArticle": "Article on the Solar Network", | ||||
|   "postStory": "Story on the Solar Network", | ||||
|   "articleWrittenAt": "Written at {}", | ||||
|   "articleEditedAt": "Edited at {}", | ||||
|   "attachmentSaved": "Saved to album", | ||||
|   "attachmentSavedDesktop": "Saved to Downloads folder", | ||||
|   "openInAlbum": "Open in album", | ||||
|   "postAbuseReport": "Report Post", | ||||
|   "postAbuseReportDescription": "Report posts that violate our user agreement and community guidelines to help us improve the content on Solar Network. Please describe how this post violates the relevant rules. Do not include any sensitive information. We will process your report within 24 hours.", | ||||
|   "abuseReport": "Abuse Report", | ||||
|   "abuseReportDescription": "Report any resources that violate our user agreement and community guidelines to help us improve the content on Solar Network. Please describe the location of the resource (provide resource ID as best as possible) and how this violates the relevant rules. Do not include any sensitive information. We will process your report within 24 hours.", | ||||
|   "abuseReportAction": "Submit Abuse Report", | ||||
|   "abuseReportActionDescription": "Report abuse usage behavior.", | ||||
|   "abuseReportResource": "Resource Location / ID", | ||||
|   "abuseReportReason": "Reason", | ||||
|   "abuseReportSubmitted": "Report submitted, thank you for your contribution.", | ||||
|   "submit": "Submit", | ||||
|   "accountDeletion": "Delete Account", | ||||
|   "accountDeletionDescription": "Are you sure you want to delete this account? This operation is irreversible, all resources (posts, chat channels, publishers, etc) belonging to this account will be permanently deleted. Be careful and think twice!", | ||||
|   "accountDeletionActionDescription": "Delete your Solarpass account.", | ||||
|   "accountDeletionSubmitted": "Account deletion request has been sent, you can check your inbox and follow the instructions in the email to complete the deletion operation.", | ||||
|   "channelNewChannel": "New Channel", | ||||
|   "channelNewDirectMessage": "New Direct Message", | ||||
|   "channelDirectMessageDescription": "Direct Message with {}", | ||||
|   "fieldCannotBeEmpty": "This field cannot be empty.", | ||||
|   "termAcceptLink": "View terms", | ||||
|   "termAcceptNextWithAgree": "By clicking the \"Next\", it means you agree to our terms and its updates.", | ||||
|   "unauthorized": "Unauthorized", | ||||
|   "unauthorizedDescription": "Login to explore the entire Solar Network.", | ||||
|   "serviceStatus": "Service Status", | ||||
|   "termRelated": "Related Terms", | ||||
|   "appDetails": "App Details", | ||||
|   "postRecommendation": "Highlight Posts", | ||||
|   "publisherBlockHint": "Block {}", | ||||
|   "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...", | ||||
|   "postImageShareReadMore": "Scan the QR code to read full post", | ||||
|   "postImageShareAds": "Explore posts on the Solar Network", | ||||
|   "postShare": "Share", | ||||
|   "postShareImage": "Share via Image", | ||||
|   "appInitializing": "Initializing", | ||||
|   "poweredBy": "Powered by {}", | ||||
|   "shareIntent": "Share", | ||||
|   "shareIntentDescription":  "What do you want to do with the content you are sharing?", | ||||
|   "shareIntentPostStory": "Post a Story" | ||||
| } | ||||
|   | ||||
| @@ -1,7 +1,6 @@ | ||||
| { | ||||
|   "nextVersionAlert": "高强度开发提示", | ||||
|   "nextVersionNotice": "您正在使用的是 Solian 2.0 的抢先体验版本,目前稳定分支(sn.solsynth.dev)版本为 1.4。该版本还在持续的开发中,部分功能可能不稳定,也并非所有功能都支持了。您可以通过 TestFlight 回滚到 1.4.X 或者继续体验新版本(sn-next.solsynth.dev)。", | ||||
|   "screen": "页面", | ||||
|   "screenAbout": "关于", | ||||
|   "screenHome": "首页", | ||||
|   "screenExplore": "探索", | ||||
|   "screenAccount": "您", | ||||
| @@ -14,9 +13,18 @@ | ||||
|   "screenAccountPublisherNew": "新建发布者", | ||||
|   "screenAccountPublisherEdit": "编辑发布者", | ||||
|   "screenAccountProfileEdit": "编辑资料", | ||||
|   "screenAbuseReport": "滥用检举", | ||||
|   "screenSettings": "设置", | ||||
|   "screenAlbum": "相册", | ||||
|   "screenChat": "聊天", | ||||
|   "screenChatManage": "编辑聊天频道", | ||||
|   "screenChatNew": "新建聊天频道", | ||||
|   "screenRealm": "领域", | ||||
|   "screenRealmManage": "编辑领域", | ||||
|   "screenRealmNew": "新建领域", | ||||
|   "screenNotification": "通知", | ||||
|   "screenPostSearch": "搜索帖子", | ||||
|   "screenFriend": "好友", | ||||
|   "dialogOkay": "好的", | ||||
|   "dialogCancel": "取消", | ||||
|   "dialogConfirm": "确认", | ||||
| @@ -27,12 +35,14 @@ | ||||
|   "errorRequestForbidden": "被禁止的请求,您没有足够的权限去做那件事。", | ||||
|   "errorRequestNotFound": "您正查找的资源无法被找到。", | ||||
|   "errorRequestConnection": "网络连接错误,请检查您的网络状态或者检查我们的服务状态。", | ||||
|   "errorRequestUnknown": "位置请求错误,您可能想将此对话框截图并发送给我们。", | ||||
|   "errorRequestUnknown": "未知请求错误,您可能想将此对话框截图并发送给我们。", | ||||
|   "unknown": "未知", | ||||
|   "loading": "加载中…", | ||||
|   "prev": "上一步", | ||||
|   "next": "下一步", | ||||
|   "edit": "编辑", | ||||
|   "apply": "应用", | ||||
|   "cancel": "取消", | ||||
|   "create": "创建", | ||||
|   "preview": "预览", | ||||
|   "delete": "删除", | ||||
| @@ -41,17 +51,29 @@ | ||||
|   "compress": "压缩", | ||||
|   "report": "检举", | ||||
|   "repost": "转帖", | ||||
|   "reply": "回贴", | ||||
|   "replyPost": "回贴", | ||||
|   "reply": "回复", | ||||
|   "unset": "未设置", | ||||
|   "untitled": "无题", | ||||
|   "postDetail": "帖子详情", | ||||
|   "postNoun": "帖子", | ||||
|   "postReadMore": "阅读更多", | ||||
|   "postReadEstimate": "预计花费 {} 阅读", | ||||
|   "postTotalLength": { | ||||
|     "zero": "没有内容", | ||||
|     "one": "总计 {} 字", | ||||
|     "other": "总计 {} 字" | ||||
|   }, | ||||
|   "fieldUsername": "用户名", | ||||
|   "fieldNickname": "显示名", | ||||
|   "fieldEmail": "电子邮箱地址", | ||||
|   "fieldPassword": "密码", | ||||
|   "fieldUsernameAlphanumOnly": "用户名只能包含英文大小写字母和数字。", | ||||
|   "fieldUsernameLengthLimit": "用户名必须在 {} 和 {} 之间。", | ||||
|   "fieldUsernameCannotEditHint": "用户名在创建后无法修改", | ||||
|   "fieldUsernameLookupHint": "支持用户名、电话号码或邮箱地址", | ||||
|   "fieldNicknameLengthLimit": "昵称必须在 {} 和 {} 之间。", | ||||
|   "fieldEmailAddressMustBeValid": "电子邮箱地址必须是一个电子邮箱地址。", | ||||
|   "fieldFirstName": "名", | ||||
|   "fieldLastName": "姓", | ||||
|   "fieldBirthday": "生日", | ||||
| @@ -81,25 +103,63 @@ | ||||
|   "publishersNew": "新发布者", | ||||
|   "publisherNewSubtitle": "创建一个新的公共身份。", | ||||
|   "publisherSyncWithAccount": "同步账户信息", | ||||
|   "publisherTotalUpvote": "总顶数", | ||||
|   "publisherTotalDownvote": "总踩数", | ||||
|   "publisherSocialPoint": "社会信用点", | ||||
|   "publisherJoinedAt": "加入于 {}", | ||||
|   "publisherSocialPointTotal": { | ||||
|     "zero": "无社会信用点", | ||||
|     "one": "{} 点社会信用点", | ||||
|     "other": "{} 点社会信用点" | ||||
|   }, | ||||
|   "publisherAffiliatedBy": "隶属于 {}", | ||||
|   "publisherRunBy": "由 {} 管理", | ||||
|   "fieldPublisherBelongToRealm": "所属领域", | ||||
|   "fieldPublisherBelongToRealmUnset": "未设置发布者所属领域", | ||||
|   "writePostTypeStory": "发动态", | ||||
|   "writePostTypeArticle": "写文章", | ||||
|   "fieldPostPublisher": "帖子发布者", | ||||
|   "fieldPostContent": "发生什么事了?!", | ||||
|   "fieldPostTitle": "标题", | ||||
|   "fieldPostDescription": "描述", | ||||
|   "fieldPostTags": "标签", | ||||
|   "postPublish": "发布", | ||||
|   "postPublishedAt": "发布于", | ||||
|   "postPublishedUntil": "取消发布于", | ||||
|   "postVisibility": "可见性", | ||||
|   "postVisibilityDescription": "帖子可见性决定了谁能查看该篇帖子。", | ||||
|   "postVisibilityAll": "所有人可见", | ||||
|   "postVisibilityFriends": "仅限好友可见", | ||||
|   "postVisibilitySelected": "选定的用户可见", | ||||
|   "postVisibilityFiltered": "选定用户不可见", | ||||
|   "postVisibilityNone": "仅自己可见", | ||||
|   "postVisibleUsers": "可见的用户", | ||||
|   "postInvisibleUsers": "不可见的用户", | ||||
|   "postSelectedUsers": { | ||||
|     "zero": "未选择用户", | ||||
|     "one": "选择了 {} 个用户", | ||||
|     "other": "选择了 {} 个用户" | ||||
|   }, | ||||
|   "postEditingNotice": "你正在修改由 {} 发布的帖子。", | ||||
|   "postReplyingNotice": "你正在回复由 {} 发布的帖子。", | ||||
|   "postRepostingNotice": "你正在转发由 {} 发布的帖子。", | ||||
|   "postReact": "反应", | ||||
|   "postPosted": "帖子已经发表。", | ||||
|   "postReactions": "帖子的反应", | ||||
|   "postReactionPoints": { | ||||
|     "zero": "{} 点", | ||||
|     "one": "{} 点", | ||||
|     "other": "{} 点" | ||||
|   "postReactionUpvote": { | ||||
|     "zero": "0 个顶", | ||||
|     "one": "{} 个顶", | ||||
|     "other": "{} 个顶" | ||||
|   }, | ||||
|   "postReactionDownvote": { | ||||
|     "zero": "0 个踩", | ||||
|     "one": "{} 个踩", | ||||
|     "other": "{} 个踩" | ||||
|   }, | ||||
|   "postReactionSocialPoint": { | ||||
|     "zero": "无社会信用点变更", | ||||
|     "one": "{} 点社会信用点变更", | ||||
|     "other": "{} 点社会信用点变更" | ||||
|   }, | ||||
|   "postReactCompleted": "反应已被添加。", | ||||
|   "postReactUncompleted": "反应已被移除。", | ||||
| @@ -128,8 +188,270 @@ | ||||
|   "settingsNetworkServerPreset": "预设的 HyperNet 服务器", | ||||
|   "settingsNetworkServerPresetDescription": "你可以在旁边的列表中选择我们提供的预设 HyperNet 服务器地址。", | ||||
|   "settingsNetworkServerSaved": "服务器地址已保存。", | ||||
|   "settingsPerformance": "性能", | ||||
|   "settingsImageQuality": "图片预览质量", | ||||
|   "settingsImageQualityDescription": "设置图片预览质量,会影响图片解码速度。", | ||||
|   "settingsImageQualityLowest": "极低", | ||||
|   "settingsImageQualityLow": "低", | ||||
|   "settingsImageQualityMedium": "中", | ||||
|   "settingsImageQualityHigh": "高", | ||||
|   "settingsMisc": "杂项", | ||||
|   "settingsMiscAbout": "关于", | ||||
|   "settingsMiscAboutDescription": "查看 Solian 的版本信息。", | ||||
|   "sensitiveContent": "敏感内容", | ||||
|   "sensitiveContentCollapsed": "敏感内容已折叠。", | ||||
|   "sensitiveContentDescription": "此内容已被标记,可能不适合所有人查看。", | ||||
|   "sensitiveContentReveal": "显示内容" | ||||
|   "sensitiveContentReveal": "显示内容", | ||||
|   "serverConnecting": "正在连接服务器…", | ||||
|   "serverDisconnected": "已与服务器断开连接", | ||||
|   "fieldChatAlias": "频道别名", | ||||
|   "fieldChatAliasHint": "全站范围内唯一的频道别名,用于在 URL 中表示该频道,留空则自动生成。应遵循 URL-Safe 的原则。", | ||||
|   "fieldChatName": "名称", | ||||
|   "fieldChatDescription": "描述", | ||||
|   "fieldChatBelongToRealm": "所属领域", | ||||
|   "fieldChatBelongToRealmUnset": "未设置频道所属领域", | ||||
|   "channelEditingNotice": "您正在编辑频道 {}", | ||||
|   "channelDeleted": "聊天频道 {} 已被删除", | ||||
|   "channelDelete": "删除聊天频道 {}", | ||||
|   "channelDeleteDescription": "你确定要删除这个聊天频道吗?该操作不可撤销,其频道内的所有消息将被永久删除。", | ||||
|   "channelDetailPersonalRegion": "个人区域", | ||||
|   "channelDetailMemberRegion": "成员管理", | ||||
|   "channelMemberManage": "管理成员", | ||||
|   "channelMemberManageDescription": "管理频道内现有成员。", | ||||
|   "channelMemberAdd": "添加成员", | ||||
|   "channelMemberAddDescription": "给当前频道添加新成员。", | ||||
|   "channelMemberAdded": "频道成员已添加。", | ||||
|   "fieldMemberRelatedName": "成员名 / 账户 ID", | ||||
|   "channelDetailAdminRegion": "管理区域", | ||||
|   "channelEditProfile": "更改频道身份", | ||||
|   "channelEdit": "编辑频道", | ||||
|   "channelEditDescription": "更改频道基本信息,元数据等。", | ||||
|   "channelProfileEdit": "编辑频道身份", | ||||
|   "channelActionDelete": "删除频道", | ||||
|   "channelActionDeleteDescription": "删除整个频道,并且删除频道里的所有信息。", | ||||
|   "channelLeave": "退出频道 {}", | ||||
|   "channelLeaveDescription": "退出该频道,但是你频道内的信息不会被移除。", | ||||
|   "channelActionLeave": "退出频道", | ||||
|   "channelActionLeaveDescription": "删除你在这个频道的身份。", | ||||
|   "channelNotifyLevel": "通知级别", | ||||
|   "channelNotifyLevelDescription": "有您决定要接受多少来自这个频道的消息。", | ||||
|   "channelNotifyLevelAll": "全部通知", | ||||
|   "channelNotifyLevelMentioned": "仅提及", | ||||
|   "channelNotifyLevelNone": "全部静音", | ||||
|   "channelNotifyLevelApplied": "已经保存并应用频道通知级别配置。", | ||||
|   "fieldChannelProfileNick": "频道内显示名", | ||||
|   "fieldChannelProfileNickHint": "在频道内显示的昵称,留空则使用账号显示名。", | ||||
|   "fieldRealmAlias": "领域别名", | ||||
|   "fieldRealmAliasHint": "全站范围内唯一的领域别名,用于在 URL 中表示该领域,留空则自动生成。应遵循 URL-Safe 的原则。", | ||||
|   "fieldRealmName": "名称", | ||||
|   "fieldRealmDescription": "描述", | ||||
|   "realmEditingNotice": "您正在编辑领域 {}", | ||||
|   "realmDeleted": "领域 {} 已被删除", | ||||
|   "realmDelete": "删除领域 {}", | ||||
|   "realmDeleteDescription": "你确定要删除这个领域吗?该操作不可撤销,其隶属于该领域的所有资源(帖子、聊天频道、发布者、制品等)都将被永久删除。三思而后行!", | ||||
|   "realmActionDelete": "删除领域", | ||||
|   "realmActionDeleteDescription": "删除整个领域及其附属的资源。", | ||||
|   "realmEdit": "编辑领域", | ||||
|   "realmEditDescription": "更改领域基本信息,元数据等。", | ||||
|   "realmMemberAdd": "添加成员", | ||||
|   "realmMemberAddDescription": "给当前领域添加新成员。", | ||||
|   "realmMemberAdded": "领域成员已添加。", | ||||
|   "fieldChatMessage": "在 {} 中发消息", | ||||
|   "fieldChatMessageDirect": "给 {} 发消息", | ||||
|   "eventResourceTag": "消息 {}", | ||||
|   "messageDelete": "删除消息 {}", | ||||
|   "messageDeleteDescription": "你确定要删除这个消息吗?该操作不可撤销。同时您将留下一条删除消息的记录。", | ||||
|   "messageDeleted": "消息 {} 已被删除", | ||||
|   "messageEdited": "消息 {} 已被编辑", | ||||
|   "messageEditedHint": "已编辑", | ||||
|   "messageUnsupported": "不支持的消息 {}", | ||||
|   "messageFileHint": { | ||||
|     "zero": "没有附件", | ||||
|     "one": "{} 个附件", | ||||
|     "other": "{} 个附件" | ||||
|   }, | ||||
|   "addAttachmentFromAlbum": "从相册中添加附件", | ||||
|   "addAttachmentFromClipboard": "粘贴附件", | ||||
|   "addAttachmentFromCameraPhoto": "拍摄照片", | ||||
|   "addAttachmentFromCameraVideo": "拍摄视频", | ||||
|   "attachmentPastedImage": "粘贴的图片", | ||||
|   "attachmentInsertLink": "插入连接", | ||||
|   "attachmentSetAsPostThumbnail": "设置为帖子缩略图", | ||||
|   "attachmentUnsetAsPostThumbnail": "取消设置为帖子缩略图", | ||||
|   "attachmentSetThumbnail": "设置缩略图", | ||||
|   "attachmentUpload": "上传", | ||||
|   "notification": "通知", | ||||
|   "notificationUnreadCount": { | ||||
|     "zero": "无未读通知", | ||||
|     "one": "有 {} 个未读通知", | ||||
|     "other": "有 {} 个未读通知" | ||||
|   }, | ||||
|   "notificationUnread": "未读", | ||||
|   "notificationRead": "已读", | ||||
|   "notificationMarkAllRead": "已读所有通知", | ||||
|   "notificationMarkAllReadDescription": "您确定要将所有通知设置为已读吗?该操作不可撤销。", | ||||
|   "notificationMarkAllReadPrompt": { | ||||
|     "zero": "已将 0 个通知标记为已读。", | ||||
|     "one": "已将 {} 个通知标记为已读。", | ||||
|     "other": "已将 {} 个通知标记为已读。" | ||||
|   }, | ||||
|   "notificationMarkOneReadPrompt": "已将通知 {} 标记为已读。", | ||||
|   "search": "搜索", | ||||
|   "postSearchResult": { | ||||
|     "zero": "没有搜索到结果", | ||||
|     "one": "搜索到 {} 个结果", | ||||
|     "other": "搜索到 {} 个结果" | ||||
|   }, | ||||
|   "postSearchTook": "耗时 {}", | ||||
|   "postDelete": "删除帖子 {}", | ||||
|   "postDeleteDescription": "你确定要删除这个帖子吗?该操作不可撤销。", | ||||
|   "postDeleted": "帖子 {} 已被删除。", | ||||
|   "call": "通话", | ||||
|   "callOngoingNotice": "一则通话进行中", | ||||
|   "callJoin": "加入", | ||||
|   "callResume": "恢复", | ||||
|   "callMicrophone": "麦克风", | ||||
|   "callCamera": "摄像头", | ||||
|   "callMicrophoneDisabled": "麦克风已禁用", | ||||
|   "callMicrophoneSelect": "选择麦克风", | ||||
|   "callCameraDisabled": "摄像头已禁用", | ||||
|   "callCameraSelect": "选择摄像头", | ||||
|   "callDisconnected": "通话已断开", | ||||
|   "callEnded": "通话已结束", | ||||
|   "callStatusConnected": "已连接", | ||||
|   "callStatusDisconnected": "未连接", | ||||
|   "callStatusConnecting": "正在连接", | ||||
|   "callStatusReconnecting": "正在重连", | ||||
|   "callDisconnect": "断开连接", | ||||
|   "callDisconnectDescription": "您确定要与通话断开连接吗?", | ||||
|   "callMicrophoneOff": "关闭麦克风", | ||||
|   "callMicrophoneOn": "打开麦克风", | ||||
|   "callCameraOff": "关闭摄像头", | ||||
|   "callCameraOn": "打开摄像头", | ||||
|   "callVideoFlip": "镜像画面", | ||||
|   "callSpeakerphoneToggle": "切换扬声器", | ||||
|   "callScreenOff": "关闭屏幕共享", | ||||
|   "callScreenOn": "开启屏幕共享", | ||||
|   "callMessageEnded": "通话持续了 {}", | ||||
|   "callMessageStarted": "通话开始了", | ||||
|   "dailyCheckIn": "每日签到", | ||||
|   "dailyCheckInNone": "今日尚未签到", | ||||
|   "dailyCheckAction": "现在签到", | ||||
|   "dailyCheckDetail": "看不懂符?大师帮我解惑!", | ||||
|   "dailyCheckDetailTitle": "{} 的运势详情", | ||||
|   "dailyCheckPositiveHint": "宜 {}", | ||||
|   "dailyCheckNegativeHint": "忌 {}", | ||||
|   "dailyCheckEverythingIsPositive": "诸事皆宜", | ||||
|   "dailyCheckEverythingIsNegative": "诸事不宜", | ||||
|   "dailyCheckPositiveHint1": "交友", | ||||
|   "dailyCheckPositiveHint1Description": "友谊地久天长", | ||||
|   "dailyCheckPositiveHint2": "饮酒", | ||||
|   "dailyCheckPositiveHint2Description": "对影成三人", | ||||
|   "dailyCheckPositiveHint3": "旅行", | ||||
|   "dailyCheckPositiveHint3Description": "千里之行,始于足下", | ||||
|   "dailyCheckPositiveHint4": "运动", | ||||
|   "dailyCheckPositiveHint4Description": "生命在于运动", | ||||
|   "dailyCheckPositiveHint5": "学习", | ||||
|   "dailyCheckPositiveHint5Description": "学无止境,日有所进", | ||||
|   "dailyCheckPositiveHint6": "种植", | ||||
|   "dailyCheckPositiveHint6Description": "种下希望,收获未来", | ||||
|   "dailyCheckNegativeHint1": "吃饭", | ||||
|   "dailyCheckNegativeHint1Description": "吃饭咬到舌头", | ||||
|   "dailyCheckNegativeHint2": "考试", | ||||
|   "dailyCheckNegativeHint2Description": "考的东西刚好没复习", | ||||
|   "dailyCheckNegativeHint3": "坐公交", | ||||
|   "dailyCheckNegativeHint3Description": "赶车刚好错过一班", | ||||
|   "dailyCheckNegativeHint4": "购物", | ||||
|   "dailyCheckNegativeHint4Description": "买回来的衣服发现不合适", | ||||
|   "dailyCheckNegativeHint5": "打游戏", | ||||
|   "dailyCheckNegativeHint5Description": "关键时刻断网", | ||||
|   "dailyCheckNegativeHint6": "出门", | ||||
|   "dailyCheckNegativeHint6Description": "忘带伞遇上大雨", | ||||
|   "happyBirthday": "生日快乐,{}!", | ||||
|   "friendNew": "添加好友", | ||||
|   "friendRequests": "好友请求", | ||||
|   "friendRequestsDescription": { | ||||
|     "zero": "你没有好友请求", | ||||
|     "one": "你有 {} 个好友请求", | ||||
|     "other": "你有 {} 个好友请求" | ||||
|   }, | ||||
|   "friendBlocklist": "屏蔽列表", | ||||
|   "friendBlocklistDescription": { | ||||
|     "zero": "你没有屏蔽任何人", | ||||
|     "one": "你屏蔽了 {} 个用户", | ||||
|     "other": "你屏蔽了 {} 个用户" | ||||
|   }, | ||||
|   "friendStatusPending": "待处理", | ||||
|   "friendStatusWaiting": "等待中", | ||||
|   "friendStatusActive": "正活跃", | ||||
|   "friendStatusBlocked": "已屏蔽", | ||||
|   "friendRequestSent": "好友请求已发送。", | ||||
|   "fieldFriendRelatedName": "好友名 / 账户 ID", | ||||
|   "friendBlock": "屏蔽", | ||||
|   "friendUnblock": "解除屏蔽", | ||||
|   "friendDeleteAction": "遗忘", | ||||
|   "friendDelete": "遗忘跟 {} 的关系", | ||||
|   "friendDeleteDescription": "你确定要遗忘跟 {} 的关系吗?这个操作无法撤销。", | ||||
|   "friendRequestAccept": "接受", | ||||
|   "friendRequestDecline": "拒绝", | ||||
|   "subscribe": "订阅", | ||||
|   "unsubscribe": "取消订阅", | ||||
|   "attachmentUploadBy": "上传者", | ||||
|   "attachmentShotOn": "由 {} 拍摄", | ||||
|   "accountJoinedAt": "加入于 {}", | ||||
|   "accountBirthday": "出生于 {}", | ||||
|   "accountBadge": "徽章", | ||||
|   "badgeCompanyStaff": "索尔辛茨士大夫 · 员工", | ||||
|   "badgeSiteMigration": "Solar Network 原住民", | ||||
|   "accountStatus": "状态", | ||||
|   "accountStatusOnline": "在线", | ||||
|   "accountStatusOffline": "离线", | ||||
|   "accountStatusLastSeen": "最后一次在 {} 上线", | ||||
|   "postArticle": "Solar Network 上的文章", | ||||
|   "postStory": "Solar Network 上的故事", | ||||
|   "articleWrittenAt": "发表于 {}", | ||||
|   "articleEditedAt": "编辑于 {}", | ||||
|   "attachmentSaved": "已保存到相册", | ||||
|   "attachmentSavedDesktop": "已保存到下载目录", | ||||
|   "openInAlbum": "在相册中打开", | ||||
|   "postAbuseReport": "检举帖子", | ||||
|   "postAbuseReportDescription": "检举不符合我们用户协议以及社区准则的帖子,来帮助我们更好的维护 Solar Network 上的内容。请在下面描述该帖子如何违反我么的相关规定。请勿填写任何敏感信息。我们将会在 24 小时内处理您的检举。", | ||||
|   "abuseReport": "检举", | ||||
|   "abuseReportDescription": "检举不符合我们用户协议以及社区准则的任何资源,来帮助我们更好的维护 Solar Network 上的内容。请在下面描述资源的位置(提供资源 ID 为佳)以及如何违反我么的相关规定。请勿填写任何敏感信息。我们将会在 24 小时内处理您的检举。", | ||||
|   "abuseReportAction": "提交检举", | ||||
|   "abuseReportActionDescription": "检举不合规行为。", | ||||
|   "abuseReportResource": "资源位置 / ID", | ||||
|   "abuseReportReason": "检举原因", | ||||
|   "abuseReportSubmitted": "检举已提交,感谢你的贡献。", | ||||
|   "submit": "提交", | ||||
|   "accountDeletion": "删除帐户", | ||||
|   "accountDeletionDescription": "你确定要删除这个帐户吗?该操作不可撤销,其隶属于该帐户的所有资源(帖子、聊天频道、发布者、制品等)都将被永久删除。三思而后行!", | ||||
|   "accountDeletionActionDescription": "删除你的 Solarpass 帐户。", | ||||
|   "accountDeletionSubmitted": "帐户删除申请已发出,你可以检查你的收件箱并根据邮件内的指示完成删除操作。", | ||||
|   "channelNewChannel": "新建频道", | ||||
|   "channelNewDirectMessage": "发起私信", | ||||
|   "channelDirectMessageDescription": "与 {} 的私聊", | ||||
|   "fieldCannotBeEmpty": "此字段不能为空。", | ||||
|   "termAcceptLink": "浏览条款", | ||||
|   "termAcceptNextWithAgree": "点击 “下一步”,即表示你同意我们的各项条款,包括其之后的更新。", | ||||
|   "unauthorized": "未登陆", | ||||
|   "unauthorizedDescription": "登陆以探索整个 Solar Network。", | ||||
|   "serviceStatus": "服务状态", | ||||
|   "termRelated": "相关条款", | ||||
|   "appDetails": "应用程序详情", | ||||
|   "postRecommendation": "推荐帖子", | ||||
|   "publisherBlockHint": "屏蔽 {}", | ||||
|   "publisherBlockHintDescription": "你正要屏蔽此发布者的运营者,该操作也将屏蔽由同一用户运营的发布者。", | ||||
|   "userUnblocked": "已解除屏蔽用户 {}", | ||||
|   "userBlocked": "已屏蔽用户 {}", | ||||
|   "postSharingViaPicture": "正在生成帖子截图,请稍等片刻……", | ||||
|   "postImageShareReadMore": "扫描右侧 QRCode 查看全文", | ||||
|   "postImageShareAds": "来 Solar Network 探索更多有趣帖子", | ||||
|   "postShare": "分享", | ||||
|   "postShareImage": "分享帖图", | ||||
|   "appInitializing": "正在初始化", | ||||
|   "poweredBy": "由 {} 提供支持", | ||||
|   "shareIntent": "分享", | ||||
|   "shareIntentDescription": "您想对您分享的内容做些什么?", | ||||
|   "shareIntentPostStory": "发布动态" | ||||
| } | ||||
|   | ||||
							
								
								
									
										457
									
								
								assets/translations/zh-HK.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,457 @@ | ||||
| { | ||||
|   "screen": "頁面", | ||||
|   "screenAbout": "關於", | ||||
|   "screenHome": "首頁", | ||||
|   "screenExplore": "探索", | ||||
|   "screenAccount": "您", | ||||
|   "screenAuthLogin": "登陸", | ||||
|   "screenAuthLoginSubtitle": "使用 Solarpass 登陸 Solar Network", | ||||
|   "screenAuthLoginGreeting": "歡迎回來", | ||||
|   "screenAuthRegister": "創建賬號", | ||||
|   "screenAuthRegisterSubtitle": "創建一個 Solarpass 賬號", | ||||
|   "screenAccountPublishers": "發佈者", | ||||
|   "screenAccountPublisherNew": "新建發佈者", | ||||
|   "screenAccountPublisherEdit": "編輯發佈者", | ||||
|   "screenAccountProfileEdit": "編輯資料", | ||||
|   "screenAbuseReport": "濫用檢舉", | ||||
|   "screenSettings": "設置", | ||||
|   "screenAlbum": "相冊", | ||||
|   "screenChat": "聊天", | ||||
|   "screenChatManage": "編輯聊天頻道", | ||||
|   "screenChatNew": "新建聊天頻道", | ||||
|   "screenRealm": "領域", | ||||
|   "screenRealmManage": "編輯領域", | ||||
|   "screenRealmNew": "新建領域", | ||||
|   "screenNotification": "通知", | ||||
|   "screenPostSearch": "搜索帖子", | ||||
|   "screenFriend": "好友", | ||||
|   "dialogOkay": "好的", | ||||
|   "dialogCancel": "取消", | ||||
|   "dialogConfirm": "確認", | ||||
|   "dialogDismiss": "忽略", | ||||
|   "dialogError": "出了點問題", | ||||
|   "errorRequestBad": "服務器拒絕了您的請求,請檢查您的輸入。", | ||||
|   "errorRequestUnauthorized": "未授權的請求,請登錄或者嘗試重新登陸。", | ||||
|   "errorRequestForbidden": "被禁止的請求,您沒有足夠的權限去做那件事。", | ||||
|   "errorRequestNotFound": "您正查找的資源無法被找到。", | ||||
|   "errorRequestConnection": "網絡連接錯誤,請檢查您的網絡狀態或者檢查我們的服務狀態。", | ||||
|   "errorRequestUnknown": "未知請求錯誤,您可能想將此對話框截圖併發送給我們。", | ||||
|   "unknown": "未知", | ||||
|   "loading": "加載中…", | ||||
|   "prev": "上一步", | ||||
|   "next": "下一步", | ||||
|   "edit": "編輯", | ||||
|   "apply": "應用", | ||||
|   "cancel": "取消", | ||||
|   "create": "創建", | ||||
|   "preview": "預覽", | ||||
|   "delete": "刪除", | ||||
|   "unlink": "解除鏈接", | ||||
|   "crop": "裁剪", | ||||
|   "compress": "壓縮", | ||||
|   "report": "檢舉", | ||||
|   "repost": "轉帖", | ||||
|   "replyPost": "回貼", | ||||
|   "reply": "回覆", | ||||
|   "unset": "未設置", | ||||
|   "untitled": "無題", | ||||
|   "postDetail": "帖子詳情", | ||||
|   "postNoun": "帖子", | ||||
|   "postReadMore": "閲讀更多", | ||||
|   "postReadEstimate": "預計花費 {} 閲讀", | ||||
|   "postTotalLength": { | ||||
|     "zero": "沒有內容", | ||||
|     "one": "總計 {} 字", | ||||
|     "other": "總計 {} 字" | ||||
|   }, | ||||
|   "fieldUsername": "用户名", | ||||
|   "fieldNickname": "顯示名", | ||||
|   "fieldEmail": "電子郵箱地址", | ||||
|   "fieldPassword": "密碼", | ||||
|   "fieldUsernameAlphanumOnly": "用户名只能包含英文大小寫字母和數字。", | ||||
|   "fieldUsernameLengthLimit": "用户名必須在 {} 和 {} 之間。", | ||||
|   "fieldUsernameCannotEditHint": "用户名在創建後無法修改", | ||||
|   "fieldUsernameLookupHint": "支持用户名、電話號碼或郵箱地址", | ||||
|   "fieldNicknameLengthLimit": "暱稱必須在 {} 和 {} 之間。", | ||||
|   "fieldEmailAddressMustBeValid": "電子郵箱地址必須是一個電子郵箱地址。", | ||||
|   "fieldFirstName": "名", | ||||
|   "fieldLastName": "姓", | ||||
|   "fieldBirthday": "生日", | ||||
|   "fieldImageHint": "你可以點擊這些個人頭像來編輯它們。", | ||||
|   "fieldDescription": "簡介", | ||||
|   "forgotPassword": "忘記密碼", | ||||
|   "loginPickFactor": "選擇方式驗證", | ||||
|   "loginMultiFactor": { | ||||
|     "one": "{} 步驗證", | ||||
|     "other": "{} 步驗證" | ||||
|   }, | ||||
|   "loginEnterPassword": "驗證代碼", | ||||
|   "loginSuccess": "登錄為 {}", | ||||
|   "authFactorPassword": "密碼", | ||||
|   "authFactorEmail": "電郵一次性驗證碼", | ||||
|   "accountIntroTitle": "喜歡您來!", | ||||
|   "accountIntroSubtitle": "登陸以探索更廣大的世界。", | ||||
|   "accountLogout": "退出登錄", | ||||
|   "accountLogoutSubtitle": "註銷當前賬户的登陸狀態。", | ||||
|   "accountLogoutConfirmTitle": "您確定要退出登錄嗎?", | ||||
|   "accountLogoutConfirm": "您需要重新輸入賬號密碼,甚至可能需要多步驗證來再次登陸。", | ||||
|   "accountPublishers": "你的發佈者", | ||||
|   "accountPublishersSubtitle": "管理你的公共形象。", | ||||
|   "accountProfileEdit": "編輯資料", | ||||
|   "accountProfileEditSubtitle": "使你的 Solarpass 賬户更像你。", | ||||
|   "accountProfileEditApplied": "個人資料修改已被應用。", | ||||
|   "publishersNew": "新發布者", | ||||
|   "publisherNewSubtitle": "創建一個新的公共身份。", | ||||
|   "publisherSyncWithAccount": "同步賬户信息", | ||||
|   "publisherTotalUpvote": "總頂數", | ||||
|   "publisherTotalDownvote": "總踩數", | ||||
|   "publisherSocialPoint": "社會信用點", | ||||
|   "publisherJoinedAt": "加入於 {}", | ||||
|   "publisherSocialPointTotal": { | ||||
|     "zero": "無社會信用點", | ||||
|     "one": "{} 點社會信用點", | ||||
|     "other": "{} 點社會信用點" | ||||
|   }, | ||||
|   "publisherAffiliatedBy": "隸屬於 {}", | ||||
|   "publisherRunBy": "由 {} 管理", | ||||
|   "fieldPublisherBelongToRealm": "所屬領域", | ||||
|   "fieldPublisherBelongToRealmUnset": "未設置發佈者所屬領域", | ||||
|   "writePostTypeStory": "發動態", | ||||
|   "writePostTypeArticle": "寫文章", | ||||
|   "fieldPostPublisher": "帖子發佈者", | ||||
|   "fieldPostContent": "發生什麼事了?!", | ||||
|   "fieldPostTitle": "標題", | ||||
|   "fieldPostDescription": "描述", | ||||
|   "fieldPostTags": "標籤", | ||||
|   "postPublish": "發佈", | ||||
|   "postPublishedAt": "發佈於", | ||||
|   "postPublishedUntil": "取消發佈於", | ||||
|   "postVisibility": "可見性", | ||||
|   "postVisibilityDescription": "帖子可見性決定了誰能查看該篇帖子。", | ||||
|   "postVisibilityAll": "所有人可見", | ||||
|   "postVisibilityFriends": "僅限好友可見", | ||||
|   "postVisibilitySelected": "選定的用户可見", | ||||
|   "postVisibilityFiltered": "選定用户不可見", | ||||
|   "postVisibilityNone": "僅自己可見", | ||||
|   "postVisibleUsers": "可見的用户", | ||||
|   "postInvisibleUsers": "不可見的用户", | ||||
|   "postSelectedUsers": { | ||||
|     "zero": "未選擇用户", | ||||
|     "one": "選擇了 {} 個用户", | ||||
|     "other": "選擇了 {} 個用户" | ||||
|   }, | ||||
|   "postEditingNotice": "你正在修改由 {} 發佈的帖子。", | ||||
|   "postReplyingNotice": "你正在回覆由 {} 發佈的帖子。", | ||||
|   "postRepostingNotice": "你正在轉發由 {} 發佈的帖子。", | ||||
|   "postReact": "反應", | ||||
|   "postPosted": "帖子已經發表。", | ||||
|   "postReactions": "帖子的反應", | ||||
|   "postReactionUpvote": { | ||||
|     "zero": "0 個頂", | ||||
|     "one": "{} 個頂", | ||||
|     "other": "{} 個頂" | ||||
|   }, | ||||
|   "postReactionDownvote": { | ||||
|     "zero": "0 個踩", | ||||
|     "one": "{} 個踩", | ||||
|     "other": "{} 個踩" | ||||
|   }, | ||||
|   "postReactionSocialPoint": { | ||||
|     "zero": "無社會信用點變更", | ||||
|     "one": "{} 點社會信用點變更", | ||||
|     "other": "{} 點社會信用點變更" | ||||
|   }, | ||||
|   "postReactCompleted": "反應已被添加。", | ||||
|   "postReactUncompleted": "反應已被移除。", | ||||
|   "postComments": { | ||||
|     "zero": "評論", | ||||
|     "one": "{} 條評論", | ||||
|     "other": "{} 條評論" | ||||
|   }, | ||||
|   "postCommentsDetailed": { | ||||
|     "zero": "沒有評論", | ||||
|     "one": "{} 條評論", | ||||
|     "other": "{} 條評論" | ||||
|   }, | ||||
|   "settingsAppearance": "外觀", | ||||
|   "settingsBackgroundImage": "背景圖片", | ||||
|   "settingsBackgroundImageDescription": "設置應用全局生效的的背景圖片。", | ||||
|   "settingsBackgroundImageClear": "清除現存背景圖", | ||||
|   "settingsBackgroundImageClearDescription": "將應用背景圖重置為空白。", | ||||
|   "settingsThemeMaterial3": "使用 Material You 設計範式", | ||||
|   "settingsThemeMaterial3Description": "將應用主題設置為 Material 3 設計範式的主題。", | ||||
|   "settingsNetwork": "網絡", | ||||
|   "settingsNetworkServer": "HyperNet 服務器", | ||||
|   "settingsNetworkServerDescription": "設置 HyperNet 服務器地址,選擇我們提供的,或者自己搭建。", | ||||
|   "settingsNetworkServerReset": "重設為官方服務器", | ||||
|   "settingsNetworkServerResetDescription": "重設為 Solar Network 的服務器地址。", | ||||
|   "settingsNetworkServerPreset": "預設的 HyperNet 服務器", | ||||
|   "settingsNetworkServerPresetDescription": "你可以在旁邊的列表中選擇我們提供的預設 HyperNet 服務器地址。", | ||||
|   "settingsNetworkServerSaved": "服務器地址已保存。", | ||||
|   "settingsPerformance": "性能", | ||||
|   "settingsImageQuality": "圖片預覽質量", | ||||
|   "settingsImageQualityDescription": "設置圖片預覽質量,會影響圖片解碼速度。", | ||||
|   "settingsImageQualityLowest": "極低", | ||||
|   "settingsImageQualityLow": "低", | ||||
|   "settingsImageQualityMedium": "中", | ||||
|   "settingsImageQualityHigh": "高", | ||||
|   "settingsMisc": "雜項", | ||||
|   "settingsMiscAbout": "關於", | ||||
|   "settingsMiscAboutDescription": "查看 Solian 的版本信息。", | ||||
|   "sensitiveContent": "敏感內容", | ||||
|   "sensitiveContentCollapsed": "敏感內容已摺疊。", | ||||
|   "sensitiveContentDescription": "此內容已被標記,可能不適合所有人查看。", | ||||
|   "sensitiveContentReveal": "顯示內容", | ||||
|   "serverConnecting": "正在連接服務器…", | ||||
|   "serverDisconnected": "已與服務器斷開連接", | ||||
|   "fieldChatAlias": "頻道別名", | ||||
|   "fieldChatAliasHint": "全站範圍內唯一的頻道別名,用於在 URL 中表示該頻道,留空則自動生成。應遵循 URL-Safe 的原則。", | ||||
|   "fieldChatName": "名稱", | ||||
|   "fieldChatDescription": "描述", | ||||
|   "fieldChatBelongToRealm": "所屬領域", | ||||
|   "fieldChatBelongToRealmUnset": "未設置頻道所屬領域", | ||||
|   "channelEditingNotice": "您正在編輯頻道 {}", | ||||
|   "channelDeleted": "聊天頻道 {} 已被刪除", | ||||
|   "channelDelete": "刪除聊天頻道 {}", | ||||
|   "channelDeleteDescription": "你確定要刪除這個聊天頻道嗎?該操作不可撤銷,其頻道內的所有消息將被永久刪除。", | ||||
|   "channelDetailPersonalRegion": "個人區域", | ||||
|   "channelDetailMemberRegion": "成員管理", | ||||
|   "channelMemberManage": "管理成員", | ||||
|   "channelMemberManageDescription": "管理頻道內現有成員。", | ||||
|   "channelMemberAdd": "添加成員", | ||||
|   "channelMemberAddDescription": "給當前頻道添加新成員。", | ||||
|   "channelMemberAdded": "頻道成員已添加。", | ||||
|   "fieldMemberRelatedName": "成員名 / 賬户 ID", | ||||
|   "channelDetailAdminRegion": "管理區域", | ||||
|   "channelEditProfile": "更改頻道身份", | ||||
|   "channelEdit": "編輯頻道", | ||||
|   "channelEditDescription": "更改頻道基本信息,元數據等。", | ||||
|   "channelProfileEdit": "編輯頻道身份", | ||||
|   "channelActionDelete": "刪除頻道", | ||||
|   "channelActionDeleteDescription": "刪除整個頻道,並且刪除頻道里的所有信息。", | ||||
|   "channelLeave": "退出頻道 {}", | ||||
|   "channelLeaveDescription": "退出該頻道,但是你頻道內的信息不會被移除。", | ||||
|   "channelActionLeave": "退出頻道", | ||||
|   "channelActionLeaveDescription": "刪除你在這個頻道的身份。", | ||||
|   "channelNotifyLevel": "通知級別", | ||||
|   "channelNotifyLevelDescription": "有您決定要接受多少來自這個頻道的消息。", | ||||
|   "channelNotifyLevelAll": "全部通知", | ||||
|   "channelNotifyLevelMentioned": "僅提及", | ||||
|   "channelNotifyLevelNone": "全部靜音", | ||||
|   "channelNotifyLevelApplied": "已經保存並應用頻道通知級別配置。", | ||||
|   "fieldChannelProfileNick": "頻道內顯示名", | ||||
|   "fieldChannelProfileNickHint": "在頻道內顯示的暱稱,留空則使用賬號顯示名。", | ||||
|   "fieldRealmAlias": "領域別名", | ||||
|   "fieldRealmAliasHint": "全站範圍內唯一的領域別名,用於在 URL 中表示該領域,留空則自動生成。應遵循 URL-Safe 的原則。", | ||||
|   "fieldRealmName": "名稱", | ||||
|   "fieldRealmDescription": "描述", | ||||
|   "realmEditingNotice": "您正在編輯領域 {}", | ||||
|   "realmDeleted": "領域 {} 已被刪除", | ||||
|   "realmDelete": "刪除領域 {}", | ||||
|   "realmDeleteDescription": "你確定要刪除這個領域嗎?該操作不可撤銷,其隸屬於該領域的所有資源(帖子、聊天頻道、發佈者、製品等)都將被永久刪除。三思而後行!", | ||||
|   "realmActionDelete": "刪除領域", | ||||
|   "realmActionDeleteDescription": "刪除整個領域及其附屬的資源。", | ||||
|   "realmEdit": "編輯領域", | ||||
|   "realmEditDescription": "更改領域基本信息,元數據等。", | ||||
|   "realmMemberAdd": "添加成員", | ||||
|   "realmMemberAddDescription": "給當前領域添加新成員。", | ||||
|   "realmMemberAdded": "領域成員已添加。", | ||||
|   "fieldChatMessage": "在 {} 中發消息", | ||||
|   "fieldChatMessageDirect": "給 {} 發消息", | ||||
|   "eventResourceTag": "消息 {}", | ||||
|   "messageDelete": "刪除消息 {}", | ||||
|   "messageDeleteDescription": "你確定要刪除這個消息嗎?該操作不可撤銷。同時您將留下一條刪除消息的記錄。", | ||||
|   "messageDeleted": "消息 {} 已被刪除", | ||||
|   "messageEdited": "消息 {} 已被編輯", | ||||
|   "messageEditedHint": "已編輯", | ||||
|   "messageUnsupported": "不支持的消息 {}", | ||||
|   "messageFileHint": { | ||||
|     "zero": "沒有附件", | ||||
|     "one": "{} 個附件", | ||||
|     "other": "{} 個附件" | ||||
|   }, | ||||
|   "addAttachmentFromAlbum": "從相冊中添加附件", | ||||
|   "addAttachmentFromClipboard": "粘貼附件", | ||||
|   "addAttachmentFromCameraPhoto": "拍攝照片", | ||||
|   "addAttachmentFromCameraVideo": "拍攝視頻", | ||||
|   "attachmentPastedImage": "粘貼的圖片", | ||||
|   "attachmentInsertLink": "插入連接", | ||||
|   "attachmentSetAsPostThumbnail": "設置為帖子縮略圖", | ||||
|   "attachmentUnsetAsPostThumbnail": "取消設置為帖子縮略圖", | ||||
|   "attachmentSetThumbnail": "設置縮略圖", | ||||
|   "attachmentUpload": "上傳", | ||||
|   "notification": "通知", | ||||
|   "notificationUnreadCount": { | ||||
|     "zero": "無未讀通知", | ||||
|     "one": "有 {} 個未讀通知", | ||||
|     "other": "有 {} 個未讀通知" | ||||
|   }, | ||||
|   "notificationUnread": "未讀", | ||||
|   "notificationRead": "已讀", | ||||
|   "notificationMarkAllRead": "已讀所有通知", | ||||
|   "notificationMarkAllReadDescription": "您確定要將所有通知設置為已讀嗎?該操作不可撤銷。", | ||||
|   "notificationMarkAllReadPrompt": { | ||||
|     "zero": "已將 0 個通知標記為已讀。", | ||||
|     "one": "已將 {} 個通知標記為已讀。", | ||||
|     "other": "已將 {} 個通知標記為已讀。" | ||||
|   }, | ||||
|   "notificationMarkOneReadPrompt": "已將通知 {} 標記為已讀。", | ||||
|   "search": "搜索", | ||||
|   "postSearchResult": { | ||||
|     "zero": "沒有搜索到結果", | ||||
|     "one": "搜索到 {} 個結果", | ||||
|     "other": "搜索到 {} 個結果" | ||||
|   }, | ||||
|   "postSearchTook": "耗時 {}", | ||||
|   "postDelete": "刪除帖子 {}", | ||||
|   "postDeleteDescription": "你確定要刪除這個帖子嗎?該操作不可撤銷。", | ||||
|   "postDeleted": "帖子 {} 已被刪除。", | ||||
|   "call": "通話", | ||||
|   "callOngoingNotice": "一則通話進行中", | ||||
|   "callJoin": "加入", | ||||
|   "callResume": "恢復", | ||||
|   "callMicrophone": "麥克風", | ||||
|   "callCamera": "攝像頭", | ||||
|   "callMicrophoneDisabled": "麥克風已禁用", | ||||
|   "callMicrophoneSelect": "選擇麥克風", | ||||
|   "callCameraDisabled": "攝像頭已禁用", | ||||
|   "callCameraSelect": "選擇攝像頭", | ||||
|   "callDisconnected": "通話已斷開", | ||||
|   "callEnded": "通話已結束", | ||||
|   "callStatusConnected": "已連接", | ||||
|   "callStatusDisconnected": "未連接", | ||||
|   "callStatusConnecting": "正在連接", | ||||
|   "callStatusReconnecting": "正在重連", | ||||
|   "callDisconnect": "斷開連接", | ||||
|   "callDisconnectDescription": "您確定要與通話斷開連接嗎?", | ||||
|   "callMicrophoneOff": "關閉麥克風", | ||||
|   "callMicrophoneOn": "打開麥克風", | ||||
|   "callCameraOff": "關閉攝像頭", | ||||
|   "callCameraOn": "打開攝像頭", | ||||
|   "callVideoFlip": "鏡像畫面", | ||||
|   "callSpeakerphoneToggle": "切換揚聲器", | ||||
|   "callScreenOff": "關閉屏幕共享", | ||||
|   "callScreenOn": "開啓屏幕共享", | ||||
|   "callMessageEnded": "通話持續了 {}", | ||||
|   "callMessageStarted": "通話開始了", | ||||
|   "dailyCheckIn": "每日簽到", | ||||
|   "dailyCheckInNone": "今日尚未簽到", | ||||
|   "dailyCheckAction": "現在簽到", | ||||
|   "dailyCheckDetail": "看不懂符?大師幫我解惑!", | ||||
|   "dailyCheckDetailTitle": "{} 的運勢詳情", | ||||
|   "dailyCheckPositiveHint": "宜 {}", | ||||
|   "dailyCheckNegativeHint": "忌 {}", | ||||
|   "dailyCheckEverythingIsPositive": "諸事皆宜", | ||||
|   "dailyCheckEverythingIsNegative": "諸事不宜", | ||||
|   "dailyCheckPositiveHint1": "交友", | ||||
|   "dailyCheckPositiveHint1Description": "友誼地久天長", | ||||
|   "dailyCheckPositiveHint2": "飲酒", | ||||
|   "dailyCheckPositiveHint2Description": "對影成三人", | ||||
|   "dailyCheckPositiveHint3": "旅行", | ||||
|   "dailyCheckPositiveHint3Description": "千里之行,始於足下", | ||||
|   "dailyCheckPositiveHint4": "運動", | ||||
|   "dailyCheckPositiveHint4Description": "生命在於運動", | ||||
|   "dailyCheckPositiveHint5": "學習", | ||||
|   "dailyCheckPositiveHint5Description": "學無止境,日有所進", | ||||
|   "dailyCheckPositiveHint6": "種植", | ||||
|   "dailyCheckPositiveHint6Description": "種下希望,收穫未來", | ||||
|   "dailyCheckNegativeHint1": "吃飯", | ||||
|   "dailyCheckNegativeHint1Description": "吃飯咬到舌頭", | ||||
|   "dailyCheckNegativeHint2": "考試", | ||||
|   "dailyCheckNegativeHint2Description": "考的東西剛好沒複習", | ||||
|   "dailyCheckNegativeHint3": "坐公交", | ||||
|   "dailyCheckNegativeHint3Description": "趕車剛好錯過一班", | ||||
|   "dailyCheckNegativeHint4": "購物", | ||||
|   "dailyCheckNegativeHint4Description": "買回來的衣服發現不合適", | ||||
|   "dailyCheckNegativeHint5": "打遊戲", | ||||
|   "dailyCheckNegativeHint5Description": "關鍵時刻斷網", | ||||
|   "dailyCheckNegativeHint6": "出門", | ||||
|   "dailyCheckNegativeHint6Description": "忘帶傘遇上大雨", | ||||
|   "happyBirthday": "生日快樂,{}!", | ||||
|   "friendNew": "添加好友", | ||||
|   "friendRequests": "好友請求", | ||||
|   "friendRequestsDescription": { | ||||
|     "zero": "你沒有好友請求", | ||||
|     "one": "你有 {} 個好友請求", | ||||
|     "other": "你有 {} 個好友請求" | ||||
|   }, | ||||
|   "friendBlocklist": "屏蔽列表", | ||||
|   "friendBlocklistDescription": { | ||||
|     "zero": "你沒有屏蔽任何人", | ||||
|     "one": "你屏蔽了 {} 個用户", | ||||
|     "other": "你屏蔽了 {} 個用户" | ||||
|   }, | ||||
|   "friendStatusPending": "待處理", | ||||
|   "friendStatusWaiting": "等待中", | ||||
|   "friendStatusActive": "正活躍", | ||||
|   "friendStatusBlocked": "已屏蔽", | ||||
|   "friendRequestSent": "好友請求已發送。", | ||||
|   "fieldFriendRelatedName": "好友名 / 賬户 ID", | ||||
|   "friendBlock": "屏蔽", | ||||
|   "friendUnblock": "解除屏蔽", | ||||
|   "friendDeleteAction": "遺忘", | ||||
|   "friendDelete": "遺忘跟 {} 的關係", | ||||
|   "friendDeleteDescription": "你確定要遺忘跟 {} 的關係嗎?這個操作無法撤銷。", | ||||
|   "friendRequestAccept": "接受", | ||||
|   "friendRequestDecline": "拒絕", | ||||
|   "subscribe": "訂閲", | ||||
|   "unsubscribe": "取消訂閲", | ||||
|   "attachmentUploadBy": "上傳者", | ||||
|   "attachmentShotOn": "由 {} 拍攝", | ||||
|   "accountJoinedAt": "加入於 {}", | ||||
|   "accountBirthday": "出生於 {}", | ||||
|   "accountBadge": "徽章", | ||||
|   "badgeCompanyStaff": "索爾辛茨士大夫 · 員工", | ||||
|   "badgeSiteMigration": "Solar Network 原住民", | ||||
|   "accountStatus": "狀態", | ||||
|   "accountStatusOnline": "在線", | ||||
|   "accountStatusOffline": "離線", | ||||
|   "accountStatusLastSeen": "最後一次在 {} 上線", | ||||
|   "postArticle": "Solar Network 上的文章", | ||||
|   "postStory": "Solar Network 上的故事", | ||||
|   "articleWrittenAt": "發表於 {}", | ||||
|   "articleEditedAt": "編輯於 {}", | ||||
|   "attachmentSaved": "已保存到相冊", | ||||
|   "attachmentSavedDesktop": "已保存到下載目錄", | ||||
|   "openInAlbum": "在相冊中打開", | ||||
|   "postAbuseReport": "檢舉帖子", | ||||
|   "postAbuseReportDescription": "檢舉不符合我們用户協議以及社區準則的帖子,來幫助我們更好的維護 Solar Network 上的內容。請在下面描述該帖子如何違反我麼的相關規定。請勿填寫任何敏感信息。我們將會在 24 小時內處理您的檢舉。", | ||||
|   "abuseReport": "檢舉", | ||||
|   "abuseReportDescription": "檢舉不符合我們用户協議以及社區準則的任何資源,來幫助我們更好的維護 Solar Network 上的內容。請在下面描述資源的位置(提供資源 ID 為佳)以及如何違反我麼的相關規定。請勿填寫任何敏感信息。我們將會在 24 小時內處理您的檢舉。", | ||||
|   "abuseReportAction": "提交檢舉", | ||||
|   "abuseReportActionDescription": "檢舉不合規行為。", | ||||
|   "abuseReportResource": "資源位置 / ID", | ||||
|   "abuseReportReason": "檢舉原因", | ||||
|   "abuseReportSubmitted": "檢舉已提交,感謝你的貢獻。", | ||||
|   "submit": "提交", | ||||
|   "accountDeletion": "刪除帳户", | ||||
|   "accountDeletionDescription": "你確定要刪除這個帳户嗎?該操作不可撤銷,其隸屬於該帳户的所有資源(帖子、聊天頻道、發佈者、製品等)都將被永久刪除。三思而後行!", | ||||
|   "accountDeletionActionDescription": "刪除你的 Solarpass 帳户。", | ||||
|   "accountDeletionSubmitted": "帳户刪除申請已發出,你可以檢查你的收件箱並根據郵件內的指示完成刪除操作。", | ||||
|   "channelNewChannel": "新建頻道", | ||||
|   "channelNewDirectMessage": "發起私信", | ||||
|   "channelDirectMessageDescription": "與 {} 的私聊", | ||||
|   "fieldCannotBeEmpty": "此字段不能為空。", | ||||
|   "termAcceptLink": "瀏覽條款", | ||||
|   "termAcceptNextWithAgree": "點擊 “下一步”,即表示你同意我們的各項條款,包括其之後的更新。", | ||||
|   "unauthorized": "未登陸", | ||||
|   "unauthorizedDescription": "登陸以探索整個 Solar Network。", | ||||
|   "serviceStatus": "服務狀態", | ||||
|   "termRelated": "相關條款", | ||||
|   "appDetails": "應用程序詳情", | ||||
|   "postRecommendation": "推薦帖子", | ||||
|   "publisherBlockHint": "屏蔽 {}", | ||||
|   "publisherBlockHintDescription": "你正要屏蔽此發佈者的運營者,該操作也將屏蔽由同一用户運營的發佈者。", | ||||
|   "userUnblocked": "已解除屏蔽用户 {}", | ||||
|   "userBlocked": "已屏蔽用户 {}", | ||||
|   "postSharingViaPicture": "正在生成帖子截圖,請稍等片刻……", | ||||
|   "postImageShareReadMore": "掃描右側 QRCode 查看全文", | ||||
|   "postImageShareAds": "來 Solar Network 探索更多有趣帖子", | ||||
|   "postShare": "分享", | ||||
|   "postShareImage": "分享帖圖", | ||||
|   "appInitializing": "正在初始化", | ||||
|   "poweredBy": "由 {} 提供支持", | ||||
|   "shareIntent": "分享", | ||||
|   "shareIntentDescription": "您想對您分享的內容做些什麼?", | ||||
|   "shareIntentPostStory": "發佈動態" | ||||
| } | ||||
							
								
								
									
										457
									
								
								assets/translations/zh-TW.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,457 @@ | ||||
| { | ||||
|   "screen": "頁面", | ||||
|   "screenAbout": "關於", | ||||
|   "screenHome": "首頁", | ||||
|   "screenExplore": "探索", | ||||
|   "screenAccount": "您", | ||||
|   "screenAuthLogin": "登陸", | ||||
|   "screenAuthLoginSubtitle": "使用 Solarpass 登陸 Solar Network", | ||||
|   "screenAuthLoginGreeting": "歡迎回來", | ||||
|   "screenAuthRegister": "建立賬號", | ||||
|   "screenAuthRegisterSubtitle": "建立一個 Solarpass 賬號", | ||||
|   "screenAccountPublishers": "釋出者", | ||||
|   "screenAccountPublisherNew": "新建釋出者", | ||||
|   "screenAccountPublisherEdit": "編輯釋出者", | ||||
|   "screenAccountProfileEdit": "編輯資料", | ||||
|   "screenAbuseReport": "濫用檢舉", | ||||
|   "screenSettings": "設定", | ||||
|   "screenAlbum": "相簿", | ||||
|   "screenChat": "聊天", | ||||
|   "screenChatManage": "編輯聊天頻道", | ||||
|   "screenChatNew": "新建聊天頻道", | ||||
|   "screenRealm": "領域", | ||||
|   "screenRealmManage": "編輯領域", | ||||
|   "screenRealmNew": "新建領域", | ||||
|   "screenNotification": "通知", | ||||
|   "screenPostSearch": "搜尋帖子", | ||||
|   "screenFriend": "好友", | ||||
|   "dialogOkay": "好的", | ||||
|   "dialogCancel": "取消", | ||||
|   "dialogConfirm": "確認", | ||||
|   "dialogDismiss": "忽略", | ||||
|   "dialogError": "出了點問題", | ||||
|   "errorRequestBad": "伺服器拒絕了您的請求,請檢查您的輸入。", | ||||
|   "errorRequestUnauthorized": "未授權的請求,請登入或者嘗試重新登陸。", | ||||
|   "errorRequestForbidden": "被禁止的請求,您沒有足夠的許可權去做那件事。", | ||||
|   "errorRequestNotFound": "您正查詢的資源無法被找到。", | ||||
|   "errorRequestConnection": "網路連線錯誤,請檢查您的網路狀態或者檢查我們的服務狀態。", | ||||
|   "errorRequestUnknown": "未知請求錯誤,您可能想將此對話方塊截圖併發送給我們。", | ||||
|   "unknown": "未知", | ||||
|   "loading": "載入中…", | ||||
|   "prev": "上一步", | ||||
|   "next": "下一步", | ||||
|   "edit": "編輯", | ||||
|   "apply": "應用", | ||||
|   "cancel": "取消", | ||||
|   "create": "建立", | ||||
|   "preview": "預覽", | ||||
|   "delete": "刪除", | ||||
|   "unlink": "解除連結", | ||||
|   "crop": "裁剪", | ||||
|   "compress": "壓縮", | ||||
|   "report": "檢舉", | ||||
|   "repost": "轉帖", | ||||
|   "replyPost": "回貼", | ||||
|   "reply": "回覆", | ||||
|   "unset": "未設定", | ||||
|   "untitled": "無題", | ||||
|   "postDetail": "帖子詳情", | ||||
|   "postNoun": "帖子", | ||||
|   "postReadMore": "閱讀更多", | ||||
|   "postReadEstimate": "預計花費 {} 閱讀", | ||||
|   "postTotalLength": { | ||||
|     "zero": "沒有內容", | ||||
|     "one": "總計 {} 字", | ||||
|     "other": "總計 {} 字" | ||||
|   }, | ||||
|   "fieldUsername": "使用者名稱", | ||||
|   "fieldNickname": "顯示名", | ||||
|   "fieldEmail": "電子郵箱地址", | ||||
|   "fieldPassword": "密碼", | ||||
|   "fieldUsernameAlphanumOnly": "使用者名稱只能包含英文大小寫字母和數字。", | ||||
|   "fieldUsernameLengthLimit": "使用者名稱必須在 {} 和 {} 之間。", | ||||
|   "fieldUsernameCannotEditHint": "使用者名稱在建立後無法修改", | ||||
|   "fieldUsernameLookupHint": "支援使用者名稱、電話號碼或郵箱地址", | ||||
|   "fieldNicknameLengthLimit": "暱稱必須在 {} 和 {} 之間。", | ||||
|   "fieldEmailAddressMustBeValid": "電子郵箱地址必須是一個電子郵箱地址。", | ||||
|   "fieldFirstName": "名", | ||||
|   "fieldLastName": "姓", | ||||
|   "fieldBirthday": "生日", | ||||
|   "fieldImageHint": "你可以點選這些個人頭像來編輯它們。", | ||||
|   "fieldDescription": "簡介", | ||||
|   "forgotPassword": "忘記密碼", | ||||
|   "loginPickFactor": "選擇方式驗證", | ||||
|   "loginMultiFactor": { | ||||
|     "one": "{} 步驗證", | ||||
|     "other": "{} 步驗證" | ||||
|   }, | ||||
|   "loginEnterPassword": "驗證程式碼", | ||||
|   "loginSuccess": "登入為 {}", | ||||
|   "authFactorPassword": "密碼", | ||||
|   "authFactorEmail": "電郵一次性驗證碼", | ||||
|   "accountIntroTitle": "喜歡您來!", | ||||
|   "accountIntroSubtitle": "登陸以探索更廣大的世界。", | ||||
|   "accountLogout": "退出登入", | ||||
|   "accountLogoutSubtitle": "登出當前賬戶的登陸狀態。", | ||||
|   "accountLogoutConfirmTitle": "您確定要退出登入嗎?", | ||||
|   "accountLogoutConfirm": "您需要重新輸入賬號密碼,甚至可能需要多步驗證來再次登陸。", | ||||
|   "accountPublishers": "你的釋出者", | ||||
|   "accountPublishersSubtitle": "管理你的公共形象。", | ||||
|   "accountProfileEdit": "編輯資料", | ||||
|   "accountProfileEditSubtitle": "使你的 Solarpass 賬戶更像你。", | ||||
|   "accountProfileEditApplied": "個人資料修改已被應用。", | ||||
|   "publishersNew": "新發布者", | ||||
|   "publisherNewSubtitle": "建立一個新的公共身份。", | ||||
|   "publisherSyncWithAccount": "同步賬戶資訊", | ||||
|   "publisherTotalUpvote": "總頂數", | ||||
|   "publisherTotalDownvote": "總踩數", | ||||
|   "publisherSocialPoint": "社會信用點", | ||||
|   "publisherJoinedAt": "加入於 {}", | ||||
|   "publisherSocialPointTotal": { | ||||
|     "zero": "無社會信用點", | ||||
|     "one": "{} 點社會信用點", | ||||
|     "other": "{} 點社會信用點" | ||||
|   }, | ||||
|   "publisherAffiliatedBy": "隸屬於 {}", | ||||
|   "publisherRunBy": "由 {} 管理", | ||||
|   "fieldPublisherBelongToRealm": "所屬領域", | ||||
|   "fieldPublisherBelongToRealmUnset": "未設定釋出者所屬領域", | ||||
|   "writePostTypeStory": "發動態", | ||||
|   "writePostTypeArticle": "寫文章", | ||||
|   "fieldPostPublisher": "帖子釋出者", | ||||
|   "fieldPostContent": "發生什麼事了?!", | ||||
|   "fieldPostTitle": "標題", | ||||
|   "fieldPostDescription": "描述", | ||||
|   "fieldPostTags": "標籤", | ||||
|   "postPublish": "釋出", | ||||
|   "postPublishedAt": "釋出於", | ||||
|   "postPublishedUntil": "取消釋出於", | ||||
|   "postVisibility": "可見性", | ||||
|   "postVisibilityDescription": "帖子可見性決定了誰能檢視該篇帖子。", | ||||
|   "postVisibilityAll": "所有人可見", | ||||
|   "postVisibilityFriends": "僅限好友可見", | ||||
|   "postVisibilitySelected": "選定的使用者可見", | ||||
|   "postVisibilityFiltered": "選定使用者不可見", | ||||
|   "postVisibilityNone": "僅自己可見", | ||||
|   "postVisibleUsers": "可見的使用者", | ||||
|   "postInvisibleUsers": "不可見的使用者", | ||||
|   "postSelectedUsers": { | ||||
|     "zero": "未選擇使用者", | ||||
|     "one": "選擇了 {} 個使用者", | ||||
|     "other": "選擇了 {} 個使用者" | ||||
|   }, | ||||
|   "postEditingNotice": "你正在修改由 {} 釋出的帖子。", | ||||
|   "postReplyingNotice": "你正在回覆由 {} 釋出的帖子。", | ||||
|   "postRepostingNotice": "你正在轉發由 {} 釋出的帖子。", | ||||
|   "postReact": "反應", | ||||
|   "postPosted": "帖子已經發表。", | ||||
|   "postReactions": "帖子的反應", | ||||
|   "postReactionUpvote": { | ||||
|     "zero": "0 個頂", | ||||
|     "one": "{} 個頂", | ||||
|     "other": "{} 個頂" | ||||
|   }, | ||||
|   "postReactionDownvote": { | ||||
|     "zero": "0 個踩", | ||||
|     "one": "{} 個踩", | ||||
|     "other": "{} 個踩" | ||||
|   }, | ||||
|   "postReactionSocialPoint": { | ||||
|     "zero": "無社會信用點變更", | ||||
|     "one": "{} 點社會信用點變更", | ||||
|     "other": "{} 點社會信用點變更" | ||||
|   }, | ||||
|   "postReactCompleted": "反應已被新增。", | ||||
|   "postReactUncompleted": "反應已被移除。", | ||||
|   "postComments": { | ||||
|     "zero": "評論", | ||||
|     "one": "{} 條評論", | ||||
|     "other": "{} 條評論" | ||||
|   }, | ||||
|   "postCommentsDetailed": { | ||||
|     "zero": "沒有評論", | ||||
|     "one": "{} 條評論", | ||||
|     "other": "{} 條評論" | ||||
|   }, | ||||
|   "settingsAppearance": "外觀", | ||||
|   "settingsBackgroundImage": "背景圖片", | ||||
|   "settingsBackgroundImageDescription": "設定應用全域性生效的的背景圖片。", | ||||
|   "settingsBackgroundImageClear": "清除現存背景圖", | ||||
|   "settingsBackgroundImageClearDescription": "將應用背景圖重置為空白。", | ||||
|   "settingsThemeMaterial3": "使用 Material You 設計正規化", | ||||
|   "settingsThemeMaterial3Description": "將應用主題設定為 Material 3 設計正規化的主題。", | ||||
|   "settingsNetwork": "網路", | ||||
|   "settingsNetworkServer": "HyperNet 伺服器", | ||||
|   "settingsNetworkServerDescription": "設定 HyperNet 伺服器地址,選擇我們提供的,或者自己搭建。", | ||||
|   "settingsNetworkServerReset": "重設為官方伺服器", | ||||
|   "settingsNetworkServerResetDescription": "重設為 Solar Network 的伺服器地址。", | ||||
|   "settingsNetworkServerPreset": "預設的 HyperNet 伺服器", | ||||
|   "settingsNetworkServerPresetDescription": "你可以在旁邊的列表中選擇我們提供的預設 HyperNet 伺服器地址。", | ||||
|   "settingsNetworkServerSaved": "伺服器地址已儲存。", | ||||
|   "settingsPerformance": "效能", | ||||
|   "settingsImageQuality": "圖片預覽質量", | ||||
|   "settingsImageQualityDescription": "設定圖片預覽質量,會影響圖片解碼速度。", | ||||
|   "settingsImageQualityLowest": "極低", | ||||
|   "settingsImageQualityLow": "低", | ||||
|   "settingsImageQualityMedium": "中", | ||||
|   "settingsImageQualityHigh": "高", | ||||
|   "settingsMisc": "雜項", | ||||
|   "settingsMiscAbout": "關於", | ||||
|   "settingsMiscAboutDescription": "檢視 Solian 的版本資訊。", | ||||
|   "sensitiveContent": "敏感內容", | ||||
|   "sensitiveContentCollapsed": "敏感內容已摺疊。", | ||||
|   "sensitiveContentDescription": "此內容已被標記,可能不適合所有人檢視。", | ||||
|   "sensitiveContentReveal": "顯示內容", | ||||
|   "serverConnecting": "正在連線伺服器…", | ||||
|   "serverDisconnected": "已與伺服器斷開連線", | ||||
|   "fieldChatAlias": "頻道別名", | ||||
|   "fieldChatAliasHint": "全站範圍內唯一的頻道別名,用於在 URL 中表示該頻道,留空則自動生成。應遵循 URL-Safe 的原則。", | ||||
|   "fieldChatName": "名稱", | ||||
|   "fieldChatDescription": "描述", | ||||
|   "fieldChatBelongToRealm": "所屬領域", | ||||
|   "fieldChatBelongToRealmUnset": "未設定頻道所屬領域", | ||||
|   "channelEditingNotice": "您正在編輯頻道 {}", | ||||
|   "channelDeleted": "聊天頻道 {} 已被刪除", | ||||
|   "channelDelete": "刪除聊天頻道 {}", | ||||
|   "channelDeleteDescription": "你確定要刪除這個聊天頻道嗎?該操作不可撤銷,其頻道內的所有訊息將被永久刪除。", | ||||
|   "channelDetailPersonalRegion": "個人區域", | ||||
|   "channelDetailMemberRegion": "成員管理", | ||||
|   "channelMemberManage": "管理成員", | ||||
|   "channelMemberManageDescription": "管理頻道內現有成員。", | ||||
|   "channelMemberAdd": "新增成員", | ||||
|   "channelMemberAddDescription": "給當前頻道新增新成員。", | ||||
|   "channelMemberAdded": "頻道成員已新增。", | ||||
|   "fieldMemberRelatedName": "成員名 / 賬戶 ID", | ||||
|   "channelDetailAdminRegion": "管理區域", | ||||
|   "channelEditProfile": "更改頻道身份", | ||||
|   "channelEdit": "編輯頻道", | ||||
|   "channelEditDescription": "更改頻道基本資訊,元資料等。", | ||||
|   "channelProfileEdit": "編輯頻道身份", | ||||
|   "channelActionDelete": "刪除頻道", | ||||
|   "channelActionDeleteDescription": "刪除整個頻道,並且刪除頻道里的所有資訊。", | ||||
|   "channelLeave": "退出頻道 {}", | ||||
|   "channelLeaveDescription": "退出該頻道,但是你頻道內的資訊不會被移除。", | ||||
|   "channelActionLeave": "退出頻道", | ||||
|   "channelActionLeaveDescription": "刪除你在這個頻道的身份。", | ||||
|   "channelNotifyLevel": "通知級別", | ||||
|   "channelNotifyLevelDescription": "有您決定要接受多少來自這個頻道的訊息。", | ||||
|   "channelNotifyLevelAll": "全部通知", | ||||
|   "channelNotifyLevelMentioned": "僅提及", | ||||
|   "channelNotifyLevelNone": "全部靜音", | ||||
|   "channelNotifyLevelApplied": "已經儲存並應用頻道通知級別配置。", | ||||
|   "fieldChannelProfileNick": "頻道內顯示名", | ||||
|   "fieldChannelProfileNickHint": "在頻道內顯示的暱稱,留空則使用賬號顯示名。", | ||||
|   "fieldRealmAlias": "領域別名", | ||||
|   "fieldRealmAliasHint": "全站範圍內唯一的領域別名,用於在 URL 中表示該領域,留空則自動生成。應遵循 URL-Safe 的原則。", | ||||
|   "fieldRealmName": "名稱", | ||||
|   "fieldRealmDescription": "描述", | ||||
|   "realmEditingNotice": "您正在編輯領域 {}", | ||||
|   "realmDeleted": "領域 {} 已被刪除", | ||||
|   "realmDelete": "刪除領域 {}", | ||||
|   "realmDeleteDescription": "你確定要刪除這個領域嗎?該操作不可撤銷,其隸屬於該領域的所有資源(帖子、聊天頻道、釋出者、製品等)都將被永久刪除。三思而後行!", | ||||
|   "realmActionDelete": "刪除領域", | ||||
|   "realmActionDeleteDescription": "刪除整個領域及其附屬的資源。", | ||||
|   "realmEdit": "編輯領域", | ||||
|   "realmEditDescription": "更改領域基本資訊,元資料等。", | ||||
|   "realmMemberAdd": "新增成員", | ||||
|   "realmMemberAddDescription": "給當前領域新增新成員。", | ||||
|   "realmMemberAdded": "領域成員已新增。", | ||||
|   "fieldChatMessage": "在 {} 中發訊息", | ||||
|   "fieldChatMessageDirect": "給 {} 發訊息", | ||||
|   "eventResourceTag": "訊息 {}", | ||||
|   "messageDelete": "刪除訊息 {}", | ||||
|   "messageDeleteDescription": "你確定要刪除這個訊息嗎?該操作不可撤銷。同時您將留下一條刪除訊息的記錄。", | ||||
|   "messageDeleted": "訊息 {} 已被刪除", | ||||
|   "messageEdited": "訊息 {} 已被編輯", | ||||
|   "messageEditedHint": "已編輯", | ||||
|   "messageUnsupported": "不支援的訊息 {}", | ||||
|   "messageFileHint": { | ||||
|     "zero": "沒有附件", | ||||
|     "one": "{} 個附件", | ||||
|     "other": "{} 個附件" | ||||
|   }, | ||||
|   "addAttachmentFromAlbum": "從相簿中新增附件", | ||||
|   "addAttachmentFromClipboard": "貼上附件", | ||||
|   "addAttachmentFromCameraPhoto": "拍攝照片", | ||||
|   "addAttachmentFromCameraVideo": "拍攝影片", | ||||
|   "attachmentPastedImage": "貼上的圖片", | ||||
|   "attachmentInsertLink": "插入連線", | ||||
|   "attachmentSetAsPostThumbnail": "設定為帖子縮圖", | ||||
|   "attachmentUnsetAsPostThumbnail": "取消設定為帖子縮圖", | ||||
|   "attachmentSetThumbnail": "設定縮圖", | ||||
|   "attachmentUpload": "上傳", | ||||
|   "notification": "通知", | ||||
|   "notificationUnreadCount": { | ||||
|     "zero": "無未讀通知", | ||||
|     "one": "有 {} 個未讀通知", | ||||
|     "other": "有 {} 個未讀通知" | ||||
|   }, | ||||
|   "notificationUnread": "未讀", | ||||
|   "notificationRead": "已讀", | ||||
|   "notificationMarkAllRead": "已讀所有通知", | ||||
|   "notificationMarkAllReadDescription": "您確定要將所有通知設定為已讀嗎?該操作不可撤銷。", | ||||
|   "notificationMarkAllReadPrompt": { | ||||
|     "zero": "已將 0 個通知標記為已讀。", | ||||
|     "one": "已將 {} 個通知標記為已讀。", | ||||
|     "other": "已將 {} 個通知標記為已讀。" | ||||
|   }, | ||||
|   "notificationMarkOneReadPrompt": "已將通知 {} 標記為已讀。", | ||||
|   "search": "搜尋", | ||||
|   "postSearchResult": { | ||||
|     "zero": "沒有搜尋到結果", | ||||
|     "one": "搜尋到 {} 個結果", | ||||
|     "other": "搜尋到 {} 個結果" | ||||
|   }, | ||||
|   "postSearchTook": "耗時 {}", | ||||
|   "postDelete": "刪除帖子 {}", | ||||
|   "postDeleteDescription": "你確定要刪除這個帖子嗎?該操作不可撤銷。", | ||||
|   "postDeleted": "帖子 {} 已被刪除。", | ||||
|   "call": "通話", | ||||
|   "callOngoingNotice": "一則通話進行中", | ||||
|   "callJoin": "加入", | ||||
|   "callResume": "恢復", | ||||
|   "callMicrophone": "麥克風", | ||||
|   "callCamera": "攝像頭", | ||||
|   "callMicrophoneDisabled": "麥克風已停用", | ||||
|   "callMicrophoneSelect": "選擇麥克風", | ||||
|   "callCameraDisabled": "攝像頭已停用", | ||||
|   "callCameraSelect": "選擇攝像頭", | ||||
|   "callDisconnected": "通話已斷開", | ||||
|   "callEnded": "通話已結束", | ||||
|   "callStatusConnected": "已連線", | ||||
|   "callStatusDisconnected": "未連線", | ||||
|   "callStatusConnecting": "正在連線", | ||||
|   "callStatusReconnecting": "正在重連", | ||||
|   "callDisconnect": "斷開連線", | ||||
|   "callDisconnectDescription": "您確定要與通話斷開連線嗎?", | ||||
|   "callMicrophoneOff": "關閉麥克風", | ||||
|   "callMicrophoneOn": "開啟麥克風", | ||||
|   "callCameraOff": "關閉攝像頭", | ||||
|   "callCameraOn": "開啟攝像頭", | ||||
|   "callVideoFlip": "映象畫面", | ||||
|   "callSpeakerphoneToggle": "切換揚聲器", | ||||
|   "callScreenOff": "關閉螢幕共享", | ||||
|   "callScreenOn": "開啟螢幕共享", | ||||
|   "callMessageEnded": "通話持續了 {}", | ||||
|   "callMessageStarted": "通話開始了", | ||||
|   "dailyCheckIn": "每日簽到", | ||||
|   "dailyCheckInNone": "今日尚未簽到", | ||||
|   "dailyCheckAction": "現在簽到", | ||||
|   "dailyCheckDetail": "看不懂符?大師幫我解惑!", | ||||
|   "dailyCheckDetailTitle": "{} 的運勢詳情", | ||||
|   "dailyCheckPositiveHint": "宜 {}", | ||||
|   "dailyCheckNegativeHint": "忌 {}", | ||||
|   "dailyCheckEverythingIsPositive": "諸事皆宜", | ||||
|   "dailyCheckEverythingIsNegative": "諸事不宜", | ||||
|   "dailyCheckPositiveHint1": "交友", | ||||
|   "dailyCheckPositiveHint1Description": "友誼地久天長", | ||||
|   "dailyCheckPositiveHint2": "飲酒", | ||||
|   "dailyCheckPositiveHint2Description": "對影成三人", | ||||
|   "dailyCheckPositiveHint3": "旅行", | ||||
|   "dailyCheckPositiveHint3Description": "千里之行,始於足下", | ||||
|   "dailyCheckPositiveHint4": "運動", | ||||
|   "dailyCheckPositiveHint4Description": "生命在於運動", | ||||
|   "dailyCheckPositiveHint5": "學習", | ||||
|   "dailyCheckPositiveHint5Description": "學無止境,日有所進", | ||||
|   "dailyCheckPositiveHint6": "種植", | ||||
|   "dailyCheckPositiveHint6Description": "種下希望,收穫未來", | ||||
|   "dailyCheckNegativeHint1": "吃飯", | ||||
|   "dailyCheckNegativeHint1Description": "吃飯咬到舌頭", | ||||
|   "dailyCheckNegativeHint2": "考試", | ||||
|   "dailyCheckNegativeHint2Description": "考的東西剛好沒複習", | ||||
|   "dailyCheckNegativeHint3": "坐公交", | ||||
|   "dailyCheckNegativeHint3Description": "趕車剛好錯過一班", | ||||
|   "dailyCheckNegativeHint4": "購物", | ||||
|   "dailyCheckNegativeHint4Description": "買回來的衣服發現不合適", | ||||
|   "dailyCheckNegativeHint5": "打遊戲", | ||||
|   "dailyCheckNegativeHint5Description": "關鍵時刻斷網", | ||||
|   "dailyCheckNegativeHint6": "出門", | ||||
|   "dailyCheckNegativeHint6Description": "忘帶傘遇上大雨", | ||||
|   "happyBirthday": "生日快樂,{}!", | ||||
|   "friendNew": "新增好友", | ||||
|   "friendRequests": "好友請求", | ||||
|   "friendRequestsDescription": { | ||||
|     "zero": "你沒有好友請求", | ||||
|     "one": "你有 {} 個好友請求", | ||||
|     "other": "你有 {} 個好友請求" | ||||
|   }, | ||||
|   "friendBlocklist": "遮蔽列表", | ||||
|   "friendBlocklistDescription": { | ||||
|     "zero": "你沒有遮蔽任何人", | ||||
|     "one": "你遮蔽了 {} 個使用者", | ||||
|     "other": "你遮蔽了 {} 個使用者" | ||||
|   }, | ||||
|   "friendStatusPending": "待處理", | ||||
|   "friendStatusWaiting": "等待中", | ||||
|   "friendStatusActive": "正活躍", | ||||
|   "friendStatusBlocked": "已遮蔽", | ||||
|   "friendRequestSent": "好友請求已傳送。", | ||||
|   "fieldFriendRelatedName": "好友名 / 賬戶 ID", | ||||
|   "friendBlock": "遮蔽", | ||||
|   "friendUnblock": "解除遮蔽", | ||||
|   "friendDeleteAction": "遺忘", | ||||
|   "friendDelete": "遺忘跟 {} 的關係", | ||||
|   "friendDeleteDescription": "你確定要遺忘跟 {} 的關係嗎?這個操作無法撤銷。", | ||||
|   "friendRequestAccept": "接受", | ||||
|   "friendRequestDecline": "拒絕", | ||||
|   "subscribe": "訂閱", | ||||
|   "unsubscribe": "取消訂閱", | ||||
|   "attachmentUploadBy": "上傳者", | ||||
|   "attachmentShotOn": "由 {} 拍攝", | ||||
|   "accountJoinedAt": "加入於 {}", | ||||
|   "accountBirthday": "出生於 {}", | ||||
|   "accountBadge": "徽章", | ||||
|   "badgeCompanyStaff": "索爾辛茨士大夫 · 員工", | ||||
|   "badgeSiteMigration": "Solar Network 原住民", | ||||
|   "accountStatus": "狀態", | ||||
|   "accountStatusOnline": "線上", | ||||
|   "accountStatusOffline": "離線", | ||||
|   "accountStatusLastSeen": "最後一次在 {} 上線", | ||||
|   "postArticle": "Solar Network 上的文章", | ||||
|   "postStory": "Solar Network 上的故事", | ||||
|   "articleWrittenAt": "發表於 {}", | ||||
|   "articleEditedAt": "編輯於 {}", | ||||
|   "attachmentSaved": "已儲存到相簿", | ||||
|   "attachmentSavedDesktop": "已儲存到下載目錄", | ||||
|   "openInAlbum": "在相簿中開啟", | ||||
|   "postAbuseReport": "檢舉帖子", | ||||
|   "postAbuseReportDescription": "檢舉不符合我們使用者協議以及社群準則的帖子,來幫助我們更好的維護 Solar Network 上的內容。請在下面描述該帖子如何違反我麼的相關規定。請勿填寫任何敏感資訊。我們將會在 24 小時內處理您的檢舉。", | ||||
|   "abuseReport": "檢舉", | ||||
|   "abuseReportDescription": "檢舉不符合我們使用者協議以及社群準則的任何資源,來幫助我們更好的維護 Solar Network 上的內容。請在下面描述資源的位置(提供資源 ID 為佳)以及如何違反我麼的相關規定。請勿填寫任何敏感資訊。我們將會在 24 小時內處理您的檢舉。", | ||||
|   "abuseReportAction": "提交檢舉", | ||||
|   "abuseReportActionDescription": "檢舉不合規行為。", | ||||
|   "abuseReportResource": "資源位置 / ID", | ||||
|   "abuseReportReason": "檢舉原因", | ||||
|   "abuseReportSubmitted": "檢舉已提交,感謝你的貢獻。", | ||||
|   "submit": "提交", | ||||
|   "accountDeletion": "刪除帳戶", | ||||
|   "accountDeletionDescription": "你確定要刪除這個帳戶嗎?該操作不可撤銷,其隸屬於該帳戶的所有資源(帖子、聊天頻道、釋出者、製品等)都將被永久刪除。三思而後行!", | ||||
|   "accountDeletionActionDescription": "刪除你的 Solarpass 帳戶。", | ||||
|   "accountDeletionSubmitted": "帳戶刪除申請已發出,你可以檢查你的收件箱並根據郵件內的指示完成刪除操作。", | ||||
|   "channelNewChannel": "新建頻道", | ||||
|   "channelNewDirectMessage": "發起私信", | ||||
|   "channelDirectMessageDescription": "與 {} 的私聊", | ||||
|   "fieldCannotBeEmpty": "此欄位不能為空。", | ||||
|   "termAcceptLink": "瀏覽條款", | ||||
|   "termAcceptNextWithAgree": "點選 “下一步”,即表示你同意我們的各項條款,包括其之後的更新。", | ||||
|   "unauthorized": "未登陸", | ||||
|   "unauthorizedDescription": "登陸以探索整個 Solar Network。", | ||||
|   "serviceStatus": "服務狀態", | ||||
|   "termRelated": "相關條款", | ||||
|   "appDetails": "應用程式詳情", | ||||
|   "postRecommendation": "推薦帖子", | ||||
|   "publisherBlockHint": "遮蔽 {}", | ||||
|   "publisherBlockHintDescription": "你正要遮蔽此釋出者的運營者,該操作也將遮蔽由同一使用者運營的釋出者。", | ||||
|   "userUnblocked": "已解除遮蔽使用者 {}", | ||||
|   "userBlocked": "已遮蔽使用者 {}", | ||||
|   "postSharingViaPicture": "正在生成帖子截圖,請稍等片刻……", | ||||
|   "postImageShareReadMore": "掃描右側 QRCode 檢視全文", | ||||
|   "postImageShareAds": "來 Solar Network 探索更多有趣帖子", | ||||
|   "postShare": "分享", | ||||
|   "postShareImage": "分享帖圖", | ||||
|   "appInitializing": "正在初始化", | ||||
|   "poweredBy": "由 {} 提供支援", | ||||
|   "shareIntent": "分享", | ||||
|   "shareIntentDescription": "您想對您分享的內容做些什麼?", | ||||
|   "shareIntentPostStory": "釋出動態" | ||||
| } | ||||
							
								
								
									
										1
									
								
								firebase.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1 @@ | ||||
| {"flutter":{"platforms":{"android":{"default":{"projectId":"solian-0x001","appId":"1:961776991058:android:a8d3f7995b0b8e86f4188b","fileOutput":"android/app/google-services.json"}},"ios":{"default":{"projectId":"solian-0x001","appId":"1:961776991058:ios:727229d368cc47e1f4188b","uploadDebugSymbols":false,"fileOutput":"ios/Runner/GoogleService-Info.plist"}},"macos":{"default":{"projectId":"solian-0x001","appId":"1:961776991058:ios:727229d368cc47e1f4188b","uploadDebugSymbols":false,"fileOutput":"macos/Runner/GoogleService-Info.plist"}},"dart":{"lib/firebase_options.dart":{"projectId":"solian-0x001","configurations":{"android":"1:961776991058:android:a8d3f7995b0b8e86f4188b","ios":"1:961776991058:ios:727229d368cc47e1f4188b","macos":"1:961776991058:ios:727229d368cc47e1f4188b","web":"1:961776991058:web:b91d12f2892a5609f4188b","windows":"1:961776991058:web:f152fd119699e13ef4188b"}}}}}} | ||||
							
								
								
									
										15
									
								
								ios/Podfile
									
									
									
									
									
								
							
							
						
						| @@ -1,5 +1,5 @@ | ||||
| # Uncomment this line to define a global platform for your project | ||||
| # platform :ios, '12.0' | ||||
| platform :ios, '13.0' | ||||
|  | ||||
| # CocoaPods analytics sends network stats synchronously affecting flutter build latency. | ||||
| ENV['COCOAPODS_DISABLE_STATS'] = 'true' | ||||
| @@ -35,10 +35,23 @@ target 'Runner' do | ||||
|   target 'RunnerTests' do | ||||
|     inherit! :search_paths | ||||
|   end | ||||
|  | ||||
|   target 'SolarWidgetExtension' do | ||||
|     inherit! :search_paths | ||||
|     pod 'Kingfisher', '~> 8.0' | ||||
|   end | ||||
|  | ||||
|   target 'SolarShare' do | ||||
|     inherit! :search_paths | ||||
|   end | ||||
| end | ||||
|  | ||||
| post_install do |installer| | ||||
|   installer.pods_project.targets.each do |target| | ||||
|     flutter_additional_ios_build_settings(target) | ||||
|     target.build_configurations.each do |config| | ||||
|       # Workaround for https://github.com/flutter/flutter/issues/64502 | ||||
|       config.build_settings['ONLY_ACTIVE_ARCH'] = 'YES' | ||||
|      end | ||||
|   end | ||||
| end | ||||
|   | ||||
							
								
								
									
										306
									
								
								ios/Podfile.lock
									
									
									
									
									
								
							
							
						
						| @@ -4,7 +4,7 @@ PODS: | ||||
|     - FlutterMacOS | ||||
|   - croppy (0.0.1): | ||||
|     - Flutter | ||||
|   - cupertino_http (0.0.1): | ||||
|   - device_info_plus (0.0.1): | ||||
|     - Flutter | ||||
|   - DKImagePickerController/Core (4.3.9): | ||||
|     - DKImagePickerController/ImageDataManager | ||||
| @@ -40,19 +40,169 @@ PODS: | ||||
|   - file_picker (0.0.1): | ||||
|     - DKImagePickerController/PhotoGallery | ||||
|     - Flutter | ||||
|   - Flutter (1.0.0) | ||||
|   - flutter_native_splash (0.0.1): | ||||
|   - file_saver (0.0.1): | ||||
|     - Flutter | ||||
|   - flutter_secure_storage (3.3.1): | ||||
|   - Firebase/Analytics (11.4.0): | ||||
|     - Firebase/Core | ||||
|   - Firebase/Core (11.4.0): | ||||
|     - Firebase/CoreOnly | ||||
|     - FirebaseAnalytics (~> 11.4.0) | ||||
|   - Firebase/CoreOnly (11.4.0): | ||||
|     - FirebaseCore (= 11.4.0) | ||||
|   - Firebase/Messaging (11.4.0): | ||||
|     - Firebase/CoreOnly | ||||
|     - FirebaseMessaging (~> 11.4.0) | ||||
|   - firebase_analytics (11.3.6): | ||||
|     - Firebase/Analytics (= 11.4.0) | ||||
|     - firebase_core | ||||
|     - Flutter | ||||
|   - firebase_core (3.9.0): | ||||
|     - Firebase/CoreOnly (= 11.4.0) | ||||
|     - Flutter | ||||
|   - firebase_messaging (15.1.6): | ||||
|     - Firebase/Messaging (= 11.4.0) | ||||
|     - firebase_core | ||||
|     - Flutter | ||||
|   - FirebaseAnalytics (11.4.0): | ||||
|     - FirebaseAnalytics/AdIdSupport (= 11.4.0) | ||||
|     - FirebaseCore (~> 11.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) | ||||
|     - FirebaseInstallations (~> 11.0) | ||||
|     - GoogleAppMeasurement (= 11.4.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) | ||||
|     - GoogleUtilities/Environment (~> 8.0) | ||||
|     - GoogleUtilities/Logger (~> 8.0) | ||||
|   - FirebaseCoreInternal (11.6.0): | ||||
|     - "GoogleUtilities/NSData+zlib (~> 8.0)" | ||||
|   - FirebaseInstallations (11.4.0): | ||||
|     - FirebaseCore (~> 11.0) | ||||
|     - GoogleUtilities/Environment (~> 8.0) | ||||
|     - GoogleUtilities/UserDefaults (~> 8.0) | ||||
|     - PromisesObjC (~> 2.4) | ||||
|   - FirebaseMessaging (11.4.0): | ||||
|     - FirebaseCore (~> 11.0) | ||||
|     - FirebaseInstallations (~> 11.0) | ||||
|     - GoogleDataTransport (~> 10.0) | ||||
|     - GoogleUtilities/AppDelegateSwizzler (~> 8.0) | ||||
|     - GoogleUtilities/Environment (~> 8.0) | ||||
|     - GoogleUtilities/Reachability (~> 8.0) | ||||
|     - GoogleUtilities/UserDefaults (~> 8.0) | ||||
|     - nanopb (~> 3.30910.0) | ||||
|   - Flutter (1.0.0) | ||||
|   - flutter_native_splash (2.4.3): | ||||
|     - Flutter | ||||
|   - flutter_udid (0.0.1): | ||||
|     - Flutter | ||||
|     - SAMKeychain | ||||
|   - flutter_webrtc (0.12.2): | ||||
|     - Flutter | ||||
|     - WebRTC-SDK (= 125.6422.06) | ||||
|   - gal (1.0.0): | ||||
|     - Flutter | ||||
|     - FlutterMacOS | ||||
|   - GoogleAppMeasurement (11.4.0): | ||||
|     - GoogleAppMeasurement/AdIdSupport (= 11.4.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) | ||||
|     - 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): | ||||
|     - GoogleUtilities/AppDelegateSwizzler (~> 8.0) | ||||
|     - GoogleUtilities/MethodSwizzler (~> 8.0) | ||||
|     - GoogleUtilities/Network (~> 8.0) | ||||
|     - "GoogleUtilities/NSData+zlib (~> 8.0)" | ||||
|     - nanopb (~> 3.30910.0) | ||||
|   - GoogleDataTransport (10.1.0): | ||||
|     - nanopb (~> 3.30910.0) | ||||
|     - PromisesObjC (~> 2.4) | ||||
|   - GoogleUtilities/AppDelegateSwizzler (8.0.2): | ||||
|     - GoogleUtilities/Environment | ||||
|     - GoogleUtilities/Logger | ||||
|     - GoogleUtilities/Network | ||||
|     - GoogleUtilities/Privacy | ||||
|   - GoogleUtilities/Environment (8.0.2): | ||||
|     - GoogleUtilities/Privacy | ||||
|   - GoogleUtilities/Logger (8.0.2): | ||||
|     - GoogleUtilities/Environment | ||||
|     - GoogleUtilities/Privacy | ||||
|   - GoogleUtilities/MethodSwizzler (8.0.2): | ||||
|     - GoogleUtilities/Logger | ||||
|     - GoogleUtilities/Privacy | ||||
|   - GoogleUtilities/Network (8.0.2): | ||||
|     - GoogleUtilities/Logger | ||||
|     - "GoogleUtilities/NSData+zlib" | ||||
|     - GoogleUtilities/Privacy | ||||
|     - GoogleUtilities/Reachability | ||||
|   - "GoogleUtilities/NSData+zlib (8.0.2)": | ||||
|     - GoogleUtilities/Privacy | ||||
|   - GoogleUtilities/Privacy (8.0.2) | ||||
|   - GoogleUtilities/Reachability (8.0.2): | ||||
|     - GoogleUtilities/Logger | ||||
|     - GoogleUtilities/Privacy | ||||
|   - GoogleUtilities/UserDefaults (8.0.2): | ||||
|     - GoogleUtilities/Logger | ||||
|     - GoogleUtilities/Privacy | ||||
|   - home_widget (0.0.1): | ||||
|     - Flutter | ||||
|   - image_picker_ios (0.0.1): | ||||
|     - Flutter | ||||
|   - Kingfisher (8.1.3) | ||||
|   - livekit_client (2.3.2): | ||||
|     - Flutter | ||||
|     - flutter_webrtc | ||||
|     - WebRTC-SDK (= 125.6422.06) | ||||
|   - media_kit_libs_ios_video (1.0.4): | ||||
|     - Flutter | ||||
|   - media_kit_native_event_loop (1.0.0): | ||||
|     - Flutter | ||||
|   - media_kit_video (0.0.1): | ||||
|     - Flutter | ||||
|   - nanopb (3.30910.0): | ||||
|     - nanopb/decode (= 3.30910.0) | ||||
|     - nanopb/encode (= 3.30910.0) | ||||
|   - nanopb/decode (3.30910.0) | ||||
|   - nanopb/encode (3.30910.0) | ||||
|   - package_info_plus (0.4.5): | ||||
|     - Flutter | ||||
|   - pasteboard (0.0.1): | ||||
|     - Flutter | ||||
|   - path_provider_foundation (0.0.1): | ||||
|     - Flutter | ||||
|     - FlutterMacOS | ||||
|   - SDWebImage (5.19.7): | ||||
|     - SDWebImage/Core (= 5.19.7) | ||||
|   - SDWebImage/Core (5.19.7) | ||||
|   - permission_handler_apple (9.3.0): | ||||
|     - Flutter | ||||
|   - PromisesObjC (2.4.0) | ||||
|   - receive_sharing_intent (1.8.1): | ||||
|     - Flutter | ||||
|   - SAMKeychain (1.5.3) | ||||
|   - screen_brightness_ios (0.1.0): | ||||
|     - Flutter | ||||
|   - SDWebImage (5.20.0): | ||||
|     - SDWebImage/Core (= 5.20.0) | ||||
|   - SDWebImage/Core (5.20.0) | ||||
|   - share_plus (0.0.1): | ||||
|     - Flutter | ||||
|   - shared_preferences_foundation (0.0.1): | ||||
|     - Flutter | ||||
|     - FlutterMacOS | ||||
| @@ -62,72 +212,188 @@ PODS: | ||||
|   - SwiftyGif (5.4.5) | ||||
|   - url_launcher_ios (0.0.1): | ||||
|     - Flutter | ||||
|   - volume_controller (0.0.1): | ||||
|     - Flutter | ||||
|   - wakelock_plus (0.0.1): | ||||
|     - Flutter | ||||
|   - WebRTC-SDK (125.6422.06) | ||||
|   - workmanager (0.0.1): | ||||
|     - Flutter | ||||
|  | ||||
| DEPENDENCIES: | ||||
|   - connectivity_plus (from `.symlinks/plugins/connectivity_plus/darwin`) | ||||
|   - croppy (from `.symlinks/plugins/croppy/ios`) | ||||
|   - cupertino_http (from `.symlinks/plugins/cupertino_http/ios`) | ||||
|   - device_info_plus (from `.symlinks/plugins/device_info_plus/ios`) | ||||
|   - file_picker (from `.symlinks/plugins/file_picker/ios`) | ||||
|   - file_saver (from `.symlinks/plugins/file_saver/ios`) | ||||
|   - firebase_analytics (from `.symlinks/plugins/firebase_analytics/ios`) | ||||
|   - firebase_core (from `.symlinks/plugins/firebase_core/ios`) | ||||
|   - firebase_messaging (from `.symlinks/plugins/firebase_messaging/ios`) | ||||
|   - Flutter (from `Flutter`) | ||||
|   - flutter_native_splash (from `.symlinks/plugins/flutter_native_splash/ios`) | ||||
|   - flutter_secure_storage (from `.symlinks/plugins/flutter_secure_storage/ios`) | ||||
|   - flutter_udid (from `.symlinks/plugins/flutter_udid/ios`) | ||||
|   - flutter_webrtc (from `.symlinks/plugins/flutter_webrtc/ios`) | ||||
|   - gal (from `.symlinks/plugins/gal/darwin`) | ||||
|   - home_widget (from `.symlinks/plugins/home_widget/ios`) | ||||
|   - image_picker_ios (from `.symlinks/plugins/image_picker_ios/ios`) | ||||
|   - Kingfisher (~> 8.0) | ||||
|   - livekit_client (from `.symlinks/plugins/livekit_client/ios`) | ||||
|   - media_kit_libs_ios_video (from `.symlinks/plugins/media_kit_libs_ios_video/ios`) | ||||
|   - media_kit_native_event_loop (from `.symlinks/plugins/media_kit_native_event_loop/ios`) | ||||
|   - media_kit_video (from `.symlinks/plugins/media_kit_video/ios`) | ||||
|   - package_info_plus (from `.symlinks/plugins/package_info_plus/ios`) | ||||
|   - pasteboard (from `.symlinks/plugins/pasteboard/ios`) | ||||
|   - path_provider_foundation (from `.symlinks/plugins/path_provider_foundation/darwin`) | ||||
|   - permission_handler_apple (from `.symlinks/plugins/permission_handler_apple/ios`) | ||||
|   - receive_sharing_intent (from `.symlinks/plugins/receive_sharing_intent/ios`) | ||||
|   - screen_brightness_ios (from `.symlinks/plugins/screen_brightness_ios/ios`) | ||||
|   - share_plus (from `.symlinks/plugins/share_plus/ios`) | ||||
|   - 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`) | ||||
|   - volume_controller (from `.symlinks/plugins/volume_controller/ios`) | ||||
|   - wakelock_plus (from `.symlinks/plugins/wakelock_plus/ios`) | ||||
|   - workmanager (from `.symlinks/plugins/workmanager/ios`) | ||||
|  | ||||
| SPEC REPOS: | ||||
|   trunk: | ||||
|     - DKImagePickerController | ||||
|     - DKPhotoGallery | ||||
|     - Firebase | ||||
|     - FirebaseAnalytics | ||||
|     - FirebaseCore | ||||
|     - FirebaseCoreInternal | ||||
|     - FirebaseInstallations | ||||
|     - FirebaseMessaging | ||||
|     - GoogleAppMeasurement | ||||
|     - GoogleDataTransport | ||||
|     - GoogleUtilities | ||||
|     - Kingfisher | ||||
|     - nanopb | ||||
|     - PromisesObjC | ||||
|     - SAMKeychain | ||||
|     - SDWebImage | ||||
|     - SwiftyGif | ||||
|     - WebRTC-SDK | ||||
|  | ||||
| EXTERNAL SOURCES: | ||||
|   connectivity_plus: | ||||
|     :path: ".symlinks/plugins/connectivity_plus/darwin" | ||||
|   croppy: | ||||
|     :path: ".symlinks/plugins/croppy/ios" | ||||
|   cupertino_http: | ||||
|     :path: ".symlinks/plugins/cupertino_http/ios" | ||||
|   device_info_plus: | ||||
|     :path: ".symlinks/plugins/device_info_plus/ios" | ||||
|   file_picker: | ||||
|     :path: ".symlinks/plugins/file_picker/ios" | ||||
|   file_saver: | ||||
|     :path: ".symlinks/plugins/file_saver/ios" | ||||
|   firebase_analytics: | ||||
|     :path: ".symlinks/plugins/firebase_analytics/ios" | ||||
|   firebase_core: | ||||
|     :path: ".symlinks/plugins/firebase_core/ios" | ||||
|   firebase_messaging: | ||||
|     :path: ".symlinks/plugins/firebase_messaging/ios" | ||||
|   Flutter: | ||||
|     :path: Flutter | ||||
|   flutter_native_splash: | ||||
|     :path: ".symlinks/plugins/flutter_native_splash/ios" | ||||
|   flutter_secure_storage: | ||||
|     :path: ".symlinks/plugins/flutter_secure_storage/ios" | ||||
|   flutter_udid: | ||||
|     :path: ".symlinks/plugins/flutter_udid/ios" | ||||
|   flutter_webrtc: | ||||
|     :path: ".symlinks/plugins/flutter_webrtc/ios" | ||||
|   gal: | ||||
|     :path: ".symlinks/plugins/gal/darwin" | ||||
|   home_widget: | ||||
|     :path: ".symlinks/plugins/home_widget/ios" | ||||
|   image_picker_ios: | ||||
|     :path: ".symlinks/plugins/image_picker_ios/ios" | ||||
|   livekit_client: | ||||
|     :path: ".symlinks/plugins/livekit_client/ios" | ||||
|   media_kit_libs_ios_video: | ||||
|     :path: ".symlinks/plugins/media_kit_libs_ios_video/ios" | ||||
|   media_kit_native_event_loop: | ||||
|     :path: ".symlinks/plugins/media_kit_native_event_loop/ios" | ||||
|   media_kit_video: | ||||
|     :path: ".symlinks/plugins/media_kit_video/ios" | ||||
|   package_info_plus: | ||||
|     :path: ".symlinks/plugins/package_info_plus/ios" | ||||
|   pasteboard: | ||||
|     :path: ".symlinks/plugins/pasteboard/ios" | ||||
|   path_provider_foundation: | ||||
|     :path: ".symlinks/plugins/path_provider_foundation/darwin" | ||||
|   permission_handler_apple: | ||||
|     :path: ".symlinks/plugins/permission_handler_apple/ios" | ||||
|   receive_sharing_intent: | ||||
|     :path: ".symlinks/plugins/receive_sharing_intent/ios" | ||||
|   screen_brightness_ios: | ||||
|     :path: ".symlinks/plugins/screen_brightness_ios/ios" | ||||
|   share_plus: | ||||
|     :path: ".symlinks/plugins/share_plus/ios" | ||||
|   shared_preferences_foundation: | ||||
|     :path: ".symlinks/plugins/shared_preferences_foundation/darwin" | ||||
|   sqflite_darwin: | ||||
|     :path: ".symlinks/plugins/sqflite_darwin/darwin" | ||||
|   url_launcher_ios: | ||||
|     :path: ".symlinks/plugins/url_launcher_ios/ios" | ||||
|   volume_controller: | ||||
|     :path: ".symlinks/plugins/volume_controller/ios" | ||||
|   wakelock_plus: | ||||
|     :path: ".symlinks/plugins/wakelock_plus/ios" | ||||
|   workmanager: | ||||
|     :path: ".symlinks/plugins/workmanager/ios" | ||||
|  | ||||
| SPEC CHECKSUMS: | ||||
|   connectivity_plus: 4c41c08fc6d7c91f63bc7aec70ffe3730b04f563 | ||||
|   connectivity_plus: 18382e7311ba19efcaee94442b23b32507b20695 | ||||
|   croppy: b6199bc8d56bd2e03cc11609d1c47ad9875c1321 | ||||
|   cupertino_http: 1a3a0f163c1b26e7f1a293b33d476e0fde7a64ec | ||||
|   device_info_plus: bf2e3232933866d73fe290f2942f2156cdd10342 | ||||
|   DKImagePickerController: 946cec48c7873164274ecc4624d19e3da4c1ef3c | ||||
|   DKPhotoGallery: b3834fecb755ee09a593d7c9e389d8b5d6deed60 | ||||
|   file_picker: 09aa5ec1ab24135ccd7a1621c46c84134bfd6655 | ||||
|   file_saver: 503e386464dbe118f630e17b4c2e1190fa0cf808 | ||||
|   Firebase: cf1b19f21410b029b6786a54e9764a0cacad3c99 | ||||
|   firebase_analytics: 2815af29d49c1a994652abd37a5b001a88bc7b75 | ||||
|   firebase_core: b62a5080210edad3f2934314a8b2c6f5124e8e10 | ||||
|   firebase_messaging: 98619a0572d82cfb3668e78859ba9f1110e268c9 | ||||
|   FirebaseAnalytics: 3feef9ae8733c567866342a1000691baaa7cad49 | ||||
|   FirebaseCore: e0510f1523bc0eb21653cac00792e1e2bd6f1771 | ||||
|   FirebaseCoreInternal: d98ab91e2d80a56d7b246856a8885443b302c0c2 | ||||
|   FirebaseInstallations: 6ef4a1c7eb2a61ee1f74727d7f6ce2e72acf1414 | ||||
|   FirebaseMessaging: f8a160d99c2c2e5babbbcc90c4a3e15db036aee2 | ||||
|   Flutter: e0871f40cf51350855a761d2e70bf5af5b9b5de7 | ||||
|   flutter_native_splash: edf599c81f74d093a4daf8e17bd7a018854bc778 | ||||
|   flutter_secure_storage: 7953c38a04c3fdbb00571bcd87d8e3b5ceb9daec | ||||
|   flutter_native_splash: e8a1e01082d97a8099d973f919f57904c925008a | ||||
|   flutter_udid: a2482c67a61b9c806ef59dd82ed8d007f1b7ac04 | ||||
|   flutter_webrtc: 1a53bd24f97bcfeff512f13699e721897f261563 | ||||
|   gal: 6a522c75909f1244732d4596d11d6a2f86ff37a5 | ||||
|   GoogleAppMeasurement: 987769c4ca6b968f2479fbcc9fe3ce34af454b8e | ||||
|   GoogleDataTransport: aae35b7ea0c09004c3797d53c8c41f66f219d6a7 | ||||
|   GoogleUtilities: 26a3abef001b6533cf678d3eb38fd3f614b7872d | ||||
|   home_widget: 0434835a4c9a75704264feff6be17ea40e0f0d57 | ||||
|   image_picker_ios: c560581cceedb403a6ff17f2f816d7fea1421fc1 | ||||
|   Kingfisher: f2af9028b16baf9dc6c07c570072bc41cbf009ef | ||||
|   livekit_client: 6108dad8b77db3142bafd4c630f471d0a54335cd | ||||
|   media_kit_libs_ios_video: a5fe24bc7875ccd6378a0978c13185e1344651c1 | ||||
|   media_kit_native_event_loop: e6b2ab20cf0746eb1c33be961fcf79667304fa2a | ||||
|   media_kit_video: 5da63f157170e5bf303bf85453b7ef6971218a2e | ||||
|   nanopb: fad817b59e0457d11a5dfbde799381cd727c1275 | ||||
|   package_info_plus: c0502532a26c7662a62a356cebe2692ec5fe4ec4 | ||||
|   pasteboard: 982969ebaa7c78af3e6cc7761e8f5e77565d9ce0 | ||||
|   path_provider_foundation: 2b6b4c569c0fb62ec74538f866245ac84301af46 | ||||
|   SDWebImage: 8a6b7b160b4d710e2a22b6900e25301075c34cb3 | ||||
|   permission_handler_apple: 9878588469a2b0d0fc1e048d9f43605f92e6cec2 | ||||
|   PromisesObjC: f5707f49cb48b9636751c5b2e7d227e43fba9f47 | ||||
|   receive_sharing_intent: 79c848f5b045674ad60b9fea3bafea59962ad2c1 | ||||
|   SAMKeychain: 483e1c9f32984d50ca961e26818a534283b4cd5c | ||||
|   screen_brightness_ios: 715ca807df953bf676d339f11464e438143ee625 | ||||
|   SDWebImage: 73c6079366fea25fa4bb9640d5fb58f0893facd8 | ||||
|   share_plus: 8b6f8b3447e494cca5317c8c3073de39b3600d1f | ||||
|   shared_preferences_foundation: fcdcbc04712aee1108ac7fda236f363274528f78 | ||||
|   sqflite_darwin: 5a7236e3b501866c1c9befc6771dfd73ffb8702d | ||||
|   SwiftyGif: 706c60cf65fa2bc5ee0313beece843c8eb8194d4 | ||||
|   url_launcher_ios: 5334b05cef931de560670eeae103fd3e431ac3fe | ||||
|   volume_controller: 531ddf792994285c9b17f9d8a7e4dcdd29b3eae9 | ||||
|   wakelock_plus: 373cfe59b235a6dd5837d0fb88791d2f13a90d56 | ||||
|   WebRTC-SDK: 79942c006ea64f6fb48d7da8a4786dfc820bc1db | ||||
|   workmanager: 0afdcf5628bbde6924c21af7836fed07b42e30e6 | ||||
|  | ||||
| PODFILE CHECKSUM: 819463e6a0290f5a72f145ba7cde16e8b6ef0796 | ||||
| PODFILE CHECKSUM: f36978bb00ec01cd27f69faaf9a821024de98fcc | ||||
|  | ||||
| COCOAPODS: 1.15.2 | ||||
| COCOAPODS: 1.16.2 | ||||
|   | ||||
| @@ -1,6 +1,8 @@ | ||||
| import Flutter | ||||
| import UIKit | ||||
|  | ||||
| import workmanager | ||||
|  | ||||
| @main | ||||
| @objc class AppDelegate: FlutterAppDelegate { | ||||
|   override func application( | ||||
| @@ -8,6 +10,13 @@ import UIKit | ||||
|     didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey: Any]? | ||||
|   ) -> Bool { | ||||
|     GeneratedPluginRegistrant.register(with: self) | ||||
|        | ||||
|     WorkmanagerPlugin.setPluginRegistrantCallback { registry in | ||||
|         GeneratedPluginRegistrant.register(with: registry) | ||||
|     } | ||||
|      | ||||
|     UIApplication.shared.setMinimumBackgroundFetchInterval(TimeInterval(60*5)) | ||||
|        | ||||
|     return super.application(application, didFinishLaunchingWithOptions: launchOptions) | ||||
|   } | ||||
| } | ||||
|   | ||||
							
								
								
									
										38
									
								
								ios/Runner/Data/Post.swift
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,38 @@ | ||||
| // | ||||
| //  SolarPost.swift | ||||
| //  Runner | ||||
| // | ||||
| //  Created by LittleSheep on 2024/12/14. | ||||
| // | ||||
|  | ||||
|  | ||||
| import Foundation | ||||
|  | ||||
| struct SolarPost : Codable { | ||||
|     let id: Int | ||||
|     let body: SolarPostBody | ||||
|     let publisher: SolarPublisher | ||||
|     let publisherId: Int | ||||
|     let createdAt: Date | ||||
|     let updatedAt: Date | ||||
|     let editedAt: Date? | ||||
|     let publishedAt: Date? | ||||
| } | ||||
|  | ||||
| struct SolarPostBody : Codable { | ||||
|     let content: String? | ||||
|     let title: String? | ||||
|     let description: String? | ||||
|     let attachments: [String]? | ||||
| } | ||||
|  | ||||
| struct SolarPublisher : Codable { | ||||
|     let id: Int | ||||
|     let name: String | ||||
|     let nick: String | ||||
|     let description: String? | ||||
|     let avatar: String? | ||||
|     let banner: String? | ||||
|     let createdAt: Date | ||||
|     let updatedAt: Date | ||||
| } | ||||
							
								
								
									
										21
									
								
								ios/Runner/Data/User.swift
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,21 @@ | ||||
| // | ||||
| //  SolarData.swift | ||||
| //  Runner | ||||
| // | ||||
| //  Created by LittleSheep on 2024/12/14. | ||||
| // | ||||
|  | ||||
| import Foundation | ||||
|  | ||||
| struct SolarUser: Codable { | ||||
|     let id: Int | ||||
|     let name: String | ||||
|     let nick: String | ||||
| } | ||||
|  | ||||
| struct SolarCheckInRecord: Codable { | ||||
|     let id: Int | ||||
|     let resultTier: Int | ||||
|     let resultExperience: Int | ||||
|     let createdAt: Date | ||||
| } | ||||
							
								
								
									
										30
									
								
								ios/Runner/GoogleService-Info.plist
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,30 @@ | ||||
| <?xml version="1.0" encoding="UTF-8"?> | ||||
| <!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd"> | ||||
| <plist version="1.0"> | ||||
| <dict> | ||||
| 	<key>API_KEY</key> | ||||
| 	<string>AIzaSyCzQIyiYKoYHTpGXhN-IjgMML8z797WVD8</string> | ||||
| 	<key>GCM_SENDER_ID</key> | ||||
| 	<string>961776991058</string> | ||||
| 	<key>PLIST_VERSION</key> | ||||
| 	<string>1</string> | ||||
| 	<key>BUNDLE_ID</key> | ||||
| 	<string>dev.solsynth.solian</string> | ||||
| 	<key>PROJECT_ID</key> | ||||
| 	<string>solian-0x001</string> | ||||
| 	<key>STORAGE_BUCKET</key> | ||||
| 	<string>solian-0x001.firebasestorage.app</string> | ||||
| 	<key>IS_ADS_ENABLED</key> | ||||
| 	<false></false> | ||||
| 	<key>IS_ANALYTICS_ENABLED</key> | ||||
| 	<false></false> | ||||
| 	<key>IS_APPINVITE_ENABLED</key> | ||||
| 	<true></true> | ||||
| 	<key>IS_GCM_ENABLED</key> | ||||
| 	<true></true> | ||||
| 	<key>IS_SIGNIN_ENABLED</key> | ||||
| 	<true></true> | ||||
| 	<key>GOOGLE_APP_ID</key> | ||||
| 	<string>1:961776991058:ios:727229d368cc47e1f4188b</string> | ||||
| </dict> | ||||
| </plist> | ||||
| @@ -2,6 +2,10 @@ | ||||
| <!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd"> | ||||
| <plist version="1.0"> | ||||
| <dict> | ||||
| 	<key>AppGroupId</key> | ||||
| 	<string>group.solsynth.solian</string> | ||||
| 	<key>CADisableMinimumFrameDurationOnPhone</key> | ||||
| 	<true/> | ||||
| 	<key>CFBundleDevelopmentRegion</key> | ||||
| 	<string>$(DEVELOPMENT_LANGUAGE)</string> | ||||
| 	<key>CFBundleDisplayName</key> | ||||
| @@ -12,6 +16,11 @@ | ||||
| 	<string>$(PRODUCT_BUNDLE_IDENTIFIER)</string> | ||||
| 	<key>CFBundleInfoDictionaryVersion</key> | ||||
| 	<string>6.0</string> | ||||
| 	<key>CFBundleLocalizations</key> | ||||
| 	<array> | ||||
| 		<string>en</string> | ||||
| 		<string>zh_CN</string> | ||||
| 	</array> | ||||
| 	<key>CFBundleName</key> | ||||
| 	<string>Solian</string> | ||||
| 	<key>CFBundlePackageType</key> | ||||
| @@ -20,14 +29,50 @@ | ||||
| 	<string>$(FLUTTER_BUILD_NAME)</string> | ||||
| 	<key>CFBundleSignature</key> | ||||
| 	<string>????</string> | ||||
| 	<key>CFBundleURLTypes</key> | ||||
| 	<array> | ||||
| 		<dict> | ||||
| 			<key>CFBundleTypeRole</key> | ||||
| 			<string>Editor</string> | ||||
| 			<key>CFBundleURLSchemes</key> | ||||
| 			<array> | ||||
| 				<string>ShareMedia-$(PRODUCT_BUNDLE_IDENTIFIER)</string> | ||||
| 			</array> | ||||
| 		</dict> | ||||
| 	</array> | ||||
| 	<key>CFBundleVersion</key> | ||||
| 	<string>$(FLUTTER_BUILD_NUMBER)</string> | ||||
| 	<key>ITSAppUsesNonExemptEncryption</key> | ||||
| 	<false/> | ||||
| 	<key>LSRequiresIPhoneOS</key> | ||||
| 	<true/> | ||||
| 	<key>NSCameraUsageDescription</key> | ||||
| 	<string>Grant access to Camera will allow Solian take photo or video for your post.</string> | ||||
| 	<key>NSMicrophoneUsageDescription</key> | ||||
| 	<string>Grant access to Microphone will allow Solian record audio for your post.</string> | ||||
| 	<key>NSPhotoLibraryAddUsageDescription</key> | ||||
| 	<string>Grant access to Photo Library will allow Solian download photo to album for you.</string> | ||||
| 	<key>NSPhotoLibraryUsageDescription</key> | ||||
| 	<string>Grant access to Photo Library will allow Solian upload photo or video for your post.</string> | ||||
| 	<key>NSUserActivityTypes</key> | ||||
| 	<array> | ||||
| 		<string>INSendMessageIntent</string> | ||||
| 	</array> | ||||
| 	<key>UIApplicationSupportsIndirectInputEvents</key> | ||||
| 	<true/> | ||||
| 	<key>UIBackgroundModes</key> | ||||
| 	<array> | ||||
| 		<string>fetch</string> | ||||
| 		<string>remote-notification</string> | ||||
| 		<string>audio</string> | ||||
| 		<string>voip</string> | ||||
| 	</array> | ||||
| 	<key>UILaunchStoryboardName</key> | ||||
| 	<string>LaunchScreen</string> | ||||
| 	<key>UIMainStoryboardFile</key> | ||||
| 	<string>Main</string> | ||||
| 	<key>UIStatusBarHidden</key> | ||||
| 	<false/> | ||||
| 	<key>UISupportedInterfaceOrientations</key> | ||||
| 	<array> | ||||
| 		<string>UIInterfaceOrientationPortrait</string> | ||||
| @@ -41,24 +86,5 @@ | ||||
| 		<string>UIInterfaceOrientationLandscapeLeft</string> | ||||
| 		<string>UIInterfaceOrientationLandscapeRight</string> | ||||
| 	</array> | ||||
| 	<key>CADisableMinimumFrameDurationOnPhone</key> | ||||
| 	<true/> | ||||
| 	<key>UIApplicationSupportsIndirectInputEvents</key> | ||||
| 	<true/> | ||||
| 	<key>CFBundleLocalizations</key> | ||||
| 	<array> | ||||
| 		<string>en</string> | ||||
| 		<string>zh_CN</string> | ||||
| 	</array> | ||||
| 	<key>NSPhotoLibraryUsageDescription</key> | ||||
| 	<string>Grant access to Photo Library will allow Solian upload photo or video for your post.</string> | ||||
| 	<key>NSCameraUsageDescription</key> | ||||
| 	<string>Grant access to Photo Library will allow Solian take photo or video for your post.</string> | ||||
| 	<key>NSMicrophoneUsageDescription</key> | ||||
| 	<string>Grant access to Photo Library will allow Solian record audio for your post.</string> | ||||
| 	<key>ITSAppUsesNonExemptEncryption</key> | ||||
| 	<false/> | ||||
| 	<key>UIStatusBarHidden</key> | ||||
| 	<false/> | ||||
| </dict> | ||||
| </plist> | ||||
|   | ||||
							
								
								
									
										19
									
								
								ios/Runner/Runner.entitlements
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,19 @@ | ||||
| <?xml version="1.0" encoding="UTF-8"?> | ||||
| <!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd"> | ||||
| <plist version="1.0"> | ||||
| <dict> | ||||
| 	<key>aps-environment</key> | ||||
| 	<string>development</string> | ||||
| 	<key>com.apple.developer.associated-domains</key> | ||||
| 	<array> | ||||
| 		<string>webcredentials:sn.solsynth.dev</string> | ||||
| 		<string>applinks:sn.solsynth.dev</string> | ||||
| 	</array> | ||||
| 	<key>com.apple.developer.usernotifications.communication</key> | ||||
| 	<true/> | ||||
| 	<key>com.apple.security.application-groups</key> | ||||
| 	<array> | ||||
| 		<string>group.solsynth.solian</string> | ||||
| 	</array> | ||||
| </dict> | ||||
| </plist> | ||||
							
								
								
									
										14
									
								
								ios/Runner/Service/Attachment.swift
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,14 @@ | ||||
| // | ||||
| //  Attachment.swift | ||||
| //  Runner | ||||
| // | ||||
| //  Created by LittleSheep on 2024/12/14. | ||||
| // | ||||
|  | ||||
| import Foundation | ||||
|  | ||||
| func getAttachmentUrl(for identifier: String) -> String { | ||||
|     let serverBaseUrl = "https://api.sn.solsynth.dev" | ||||
|      | ||||
|     return identifier.starts(with: "http") ? identifier : "\(serverBaseUrl)/cgi/uc/attachments/\(identifier)" | ||||
| } | ||||
							
								
								
									
										18
									
								
								ios/SolarNotifyService/Info.plist
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,18 @@ | ||||
| <?xml version="1.0" encoding="UTF-8"?> | ||||
| <!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd"> | ||||
| <plist version="1.0"> | ||||
| <dict> | ||||
| 	<key>NSUserActivityTypes</key> | ||||
| 	<array> | ||||
| 		<string>INStartCallIntent</string> | ||||
| 		<string>INSendMessageIntent</string> | ||||
| 	</array> | ||||
| 	<key>NSExtension</key> | ||||
| 	<dict> | ||||
| 		<key>NSExtensionPointIdentifier</key> | ||||
| 		<string>com.apple.usernotifications.service</string> | ||||
| 		<key>NSExtensionPrincipalClass</key> | ||||
| 		<string>$(PRODUCT_MODULE_NAME).NotificationService</string> | ||||
| 	</dict> | ||||
| </dict> | ||||
| </plist> | ||||
							
								
								
									
										240
									
								
								ios/SolarNotifyService/NotificationService.swift
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,240 @@ | ||||
| // | ||||
| //  NotificationService.swift | ||||
| //  SolarNotifyService | ||||
| // | ||||
| //  Created by LittleSheep on 2024/12/8. | ||||
| // | ||||
|  | ||||
| import UserNotifications | ||||
| import Intents | ||||
|  | ||||
| enum ParseNotificationPayloadError: Error { | ||||
|     case missingMetadata(String) | ||||
|     case missingAvatarUrl(String) | ||||
| } | ||||
|  | ||||
| class NotificationService: UNNotificationServiceExtension { | ||||
|      | ||||
|     private var contentHandler: ((UNNotificationContent) -> Void)? | ||||
|     private var bestAttemptContent: UNMutableNotificationContent? | ||||
|      | ||||
|     private func fetchAvatarImage(from url: String, completion: @escaping (INImage?) -> Void) { | ||||
|         guard let imageURL = URL(string: url) else { | ||||
|             completion(nil) | ||||
|             return | ||||
|         } | ||||
|          | ||||
|         // Define a cache location based on the URL hash | ||||
|         let cacheFileName = imageURL.lastPathComponent | ||||
|         let tempDirectory = FileManager.default.temporaryDirectory | ||||
|         let cachedFileUrl = tempDirectory.appendingPathComponent(cacheFileName) | ||||
|          | ||||
|         // Check if the image is already cached | ||||
|         if FileManager.default.fileExists(atPath: cachedFileUrl.path) { | ||||
|             do { | ||||
|                 let data = try Data(contentsOf: cachedFileUrl) | ||||
|                 let cachedImage = INImage(imageData: data) // No optional binding here | ||||
|                 completion(cachedImage) | ||||
|                 return | ||||
|             } catch { | ||||
|                 print("Failed to load cached avatar image: \(error.localizedDescription)") | ||||
|                 try? FileManager.default.removeItem(at: cachedFileUrl) // Clear corrupted cache | ||||
|             } | ||||
|         } | ||||
|          | ||||
|         // Download the image if not cached | ||||
|         let session = URLSession(configuration: .default) | ||||
|         session.downloadTask(with: imageURL) { localUrl, response, error in | ||||
|             if let error = error { | ||||
|                 print("Failed to fetch avatar image: \(error.localizedDescription)") | ||||
|                 completion(nil) | ||||
|                 return | ||||
|             } | ||||
|              | ||||
|             guard let localUrl = localUrl, let data = try? Data(contentsOf: localUrl) else { | ||||
|                 print("Failed to fetch data for avatar image.") | ||||
|                 completion(nil) | ||||
|                 return | ||||
|             } | ||||
|              | ||||
|             do { | ||||
|                 // Cache the downloaded file | ||||
|                 try FileManager.default.moveItem(at: localUrl, to: cachedFileUrl) | ||||
|             } catch { | ||||
|                 print("Failed to cache avatar image: \(error.localizedDescription)") | ||||
|             } | ||||
|              | ||||
|             // Create INImage from the downloaded data | ||||
|             let inImage = INImage(imageData: data) // Create directly | ||||
|             completion(inImage) | ||||
|         }.resume() | ||||
|     } | ||||
|      | ||||
|     override func didReceive( | ||||
|         _ request: UNNotificationRequest, | ||||
|         withContentHandler contentHandler: @escaping (UNNotificationContent) -> Void | ||||
|     ) { | ||||
|         self.contentHandler = contentHandler | ||||
|         guard let bestAttemptContent = request.content.mutableCopy() as? UNMutableNotificationContent else { | ||||
|             contentHandler(request.content) | ||||
|             return | ||||
|         } | ||||
|         self.bestAttemptContent = bestAttemptContent | ||||
|          | ||||
|         do { | ||||
|             try processNotification(request: request, content: bestAttemptContent) | ||||
|         } catch { | ||||
|             contentHandler(bestAttemptContent) | ||||
|         } | ||||
|     } | ||||
|      | ||||
|     override func serviceExtensionTimeWillExpire() { | ||||
|         if let contentHandler = contentHandler, let bestAttemptContent = bestAttemptContent { | ||||
|             contentHandler(bestAttemptContent) | ||||
|         } | ||||
|     } | ||||
|      | ||||
|     private func processNotification(request: UNNotificationRequest, content: UNMutableNotificationContent) throws { | ||||
|         switch content.categoryIdentifier { | ||||
|         case "messaging.message", "messaging.callStart": | ||||
|             try handleMessagingNotification(request: request, content: content) | ||||
|         default: | ||||
|             try handleDefaultNotification(content: content) | ||||
|         } | ||||
|     } | ||||
|      | ||||
|     private func handleMessagingNotification(request: UNNotificationRequest, content: UNMutableNotificationContent) throws { | ||||
|         guard let metadata = content.userInfo["metadata"] as? [AnyHashable: Any] else { | ||||
|             throw ParseNotificationPayloadError.missingMetadata("The notification has no metadata.") | ||||
|         } | ||||
|          | ||||
|         guard let avatarIdentifier = metadata["avatar"] as? String else { | ||||
|             throw ParseNotificationPayloadError.missingAvatarUrl("The notification has no avatar.") | ||||
|         } | ||||
|          | ||||
|         let avatarUrl = getAttachmentUrl(for: avatarIdentifier) | ||||
|         fetchAvatarImage(from: avatarUrl) { [weak self] inImage in | ||||
|             guard let self = self else { return } | ||||
|              | ||||
|             let handle = INPersonHandle(value: "\(metadata["user_id"] ?? "")", type: .unknown) | ||||
|             let sender = INPerson( | ||||
|                 personHandle: handle, | ||||
|                 nameComponents: nil, | ||||
|                 displayName: content.title, | ||||
|                 image: inImage, | ||||
|                 contactIdentifier: nil, | ||||
|                 customIdentifier: nil | ||||
|             ) | ||||
|              | ||||
|             if content.categoryIdentifier == "messaging.callStart" { | ||||
|                 let intent = self.createCallIntent(with: sender) | ||||
|                 self.donateInteraction(for: intent) | ||||
|                 let updatedContent = try? request.content.updating(from: intent) | ||||
|                 self.contentHandler?(updatedContent ?? content) | ||||
|             } else { | ||||
|                 let intent = self.createMessageIntent(with: sender, metadata: metadata, body: content.body) | ||||
|                 self.donateInteraction(for: intent) | ||||
|                 let updatedContent = try? request.content.updating(from: intent) | ||||
|                 self.contentHandler?(updatedContent ?? content) | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|      | ||||
|     private func handleDefaultNotification(content: UNMutableNotificationContent) throws { | ||||
|         guard let metadata = content.userInfo["metadata"] as? [AnyHashable: Any] else { | ||||
|             throw ParseNotificationPayloadError.missingMetadata("The notification has no metadata.") | ||||
|         } | ||||
|          | ||||
|         if let imageIdentifier = metadata["image"] as? String { | ||||
|             attachMedia(to: content, withIdentifier: imageIdentifier) | ||||
|         } else if let avatarIdentifier = metadata["avatar"] as? String { | ||||
|             attachMedia(to: content, withIdentifier: avatarIdentifier) | ||||
|         } | ||||
|          | ||||
|         contentHandler?(content) | ||||
|     } | ||||
|      | ||||
|     private func attachMedia(to content: UNMutableNotificationContent, withIdentifier identifier: String) { | ||||
|         let attachmentUrl = getAttachmentUrl(for: identifier) | ||||
|          | ||||
|         guard let remoteUrl = URL(string: attachmentUrl) else { | ||||
|             print("Invalid URL for attachment: \(attachmentUrl)") | ||||
|             return | ||||
|         } | ||||
|          | ||||
|         // Define a cache location based on the identifier | ||||
|         let tempDirectory = FileManager.default.temporaryDirectory | ||||
|         let cachedFileUrl = tempDirectory.appendingPathComponent(identifier) | ||||
|          | ||||
|         if FileManager.default.fileExists(atPath: cachedFileUrl.path) { | ||||
|             // Use cached file | ||||
|             attachLocalMedia(to: content, from: cachedFileUrl, withIdentifier: identifier) | ||||
|         } else { | ||||
|             // Download and cache the file | ||||
|             let session = URLSession(configuration: .default) | ||||
|             session.downloadTask(with: remoteUrl) { [weak content] localUrl, response, error in | ||||
|                 guard let content = content else { return } | ||||
|                  | ||||
|                 if let error = error { | ||||
|                     print("Failed to download media: \(error.localizedDescription)") | ||||
|                     self.contentHandler?(content) | ||||
|                     return | ||||
|                 } | ||||
|                  | ||||
|                 guard let localUrl = localUrl else { | ||||
|                     print("No local file URL after download") | ||||
|                     self.contentHandler?(content) | ||||
|                     return | ||||
|                 } | ||||
|                  | ||||
|                 do { | ||||
|                     // Move the downloaded file to the cache | ||||
|                     try FileManager.default.moveItem(at: localUrl, to: cachedFileUrl) | ||||
|                     self.attachLocalMedia(to: content, from: cachedFileUrl, withIdentifier: identifier) | ||||
|                 } catch { | ||||
|                     print("Failed to cache media file: \(error.localizedDescription)") | ||||
|                     self.contentHandler?(content) | ||||
|                 } | ||||
|             }.resume() | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private func attachLocalMedia(to content: UNMutableNotificationContent, from localUrl: URL, withIdentifier identifier: String) { | ||||
|         if let attachment = try? UNNotificationAttachment(identifier: identifier, url: localUrl) { | ||||
|             content.attachments = [attachment] | ||||
|         } else { | ||||
|             print("Failed to create attachment from cached file: \(localUrl.path)") | ||||
|         } | ||||
|         self.contentHandler?(content) | ||||
|     } | ||||
|      | ||||
|     private func createCallIntent(with sender: INPerson) -> INStartCallIntent { | ||||
|         INStartCallIntent( | ||||
|             callRecordFilter: nil, | ||||
|             callRecordToCallBack: nil, | ||||
|             audioRoute: .unknown, | ||||
|             destinationType: .normal, | ||||
|             contacts: [sender], | ||||
|             callCapability: .unknown | ||||
|         ) | ||||
|     } | ||||
|      | ||||
|     private func createMessageIntent(with sender: INPerson, metadata: [AnyHashable: Any], body: String) -> INSendMessageIntent { | ||||
|         INSendMessageIntent( | ||||
|             recipients: nil, | ||||
|             outgoingMessageType: .outgoingMessageText, | ||||
|             content: body, | ||||
|             speakableGroupName: nil, | ||||
|             conversationIdentifier: "\(metadata["channel_id"] ?? "")", | ||||
|             serviceName: nil, | ||||
|             sender: sender, | ||||
|             attachments: nil | ||||
|         ) | ||||
|     } | ||||
|      | ||||
|     private func donateInteraction(for intent: INIntent) { | ||||
|         let interaction = INInteraction(intent: intent, response: nil) | ||||
|         interaction.direction = .incoming | ||||
|         interaction.donate(completion: nil) | ||||
|     } | ||||
| } | ||||
							
								
								
									
										10
									
								
								ios/SolarNotifyService/SolarNotifyService.entitlements
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,10 @@ | ||||
| <?xml version="1.0" encoding="UTF-8"?> | ||||
| <!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd"> | ||||
| <plist version="1.0"> | ||||
| <dict> | ||||
| 	<key>com.apple.security.application-groups</key> | ||||
| 	<array> | ||||
| 		<string>group.solsynth.solian</string> | ||||
| 	</array> | ||||
| </dict> | ||||
| </plist> | ||||
							
								
								
									
										24
									
								
								ios/SolarShare/Base.lproj/MainInterface.storyboard
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,24 @@ | ||||
| <?xml version="1.0" encoding="UTF-8"?> | ||||
| <document type="com.apple.InterfaceBuilder3.CocoaTouch.Storyboard.XIB" version="3.0" toolsVersion="13122.16" targetRuntime="iOS.CocoaTouch" propertyAccessControl="none" useAutolayout="YES" useTraitCollections="YES" useSafeAreas="YES" colorMatched="YES" initialViewController="j1y-V4-xli"> | ||||
|     <dependencies> | ||||
|         <plugIn identifier="com.apple.InterfaceBuilder.IBCocoaTouchPlugin" version="13104.12"/> | ||||
|         <capability name="Safe area layout guides" minToolsVersion="9.0"/> | ||||
|         <capability name="documents saved in the Xcode 8 format" minToolsVersion="8.0"/> | ||||
|     </dependencies> | ||||
|     <scenes> | ||||
|         <!--Share View Controller--> | ||||
|         <scene sceneID="ceB-am-kn3"> | ||||
|             <objects> | ||||
|                 <viewController id="j1y-V4-xli" customClass="ShareViewController" customModuleProvider="target" sceneMemberID="viewController"> | ||||
|                     <view key="view" opaque="NO" contentMode="scaleToFill" id="wbc-yd-nQP"> | ||||
|                         <rect key="frame" x="0.0" y="0.0" width="375" height="667"/> | ||||
|                         <autoresizingMask key="autoresizingMask" widthSizable="YES" heightSizable="YES"/> | ||||
|                         <color key="backgroundColor" red="0.0" green="0.0" blue="0.0" alpha="0.0" colorSpace="custom" customColorSpace="sRGB"/> | ||||
|                         <viewLayoutGuide key="safeArea" id="1Xd-am-t49"/> | ||||
|                     </view> | ||||
|                 </viewController> | ||||
|                 <placeholder placeholderIdentifier="IBFirstResponder" id="CEy-Cv-SGf" userLabel="First Responder" sceneMemberID="firstResponder"/> | ||||
|             </objects> | ||||
|         </scene> | ||||
|     </scenes> | ||||
| </document> | ||||
							
								
								
									
										36
									
								
								ios/SolarShare/Info.plist
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,36 @@ | ||||
| <?xml version="1.0" encoding="UTF-8"?> | ||||
| <!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd"> | ||||
| <plist version="1.0"> | ||||
| <dict> | ||||
| 	<key>PHSupportedMediaTypes</key> | ||||
| 	<array> | ||||
| 		<string>Video</string> | ||||
| 		<string>Image</string> | ||||
| 	</array> | ||||
| 	<key>NSExtension</key> | ||||
| 	<dict> | ||||
| 		<key>NSExtensionAttributes</key> | ||||
| 		<dict> | ||||
|             <key>NSExtensionActivationRule</key> | ||||
|             <dict> | ||||
|                 <key>NSExtensionActivationSupportsText</key> | ||||
|                 <true/> | ||||
|                 <key>NSExtensionActivationSupportsWebURLWithMaxCount</key> | ||||
|                 <integer>15</integer> | ||||
|                 <key>NSExtensionActivationSupportsImageWithMaxCount</key> | ||||
|                 <integer>15</integer> | ||||
|                 <key>NSExtensionActivationSupportsMovieWithMaxCount</key> | ||||
|                 <integer>15</integer> | ||||
|                 <key>NSExtensionActivationSupportsFileWithMaxCount</key> | ||||
|                 <integer>15</integer> | ||||
|             </dict> | ||||
| 		</dict> | ||||
| 		<key>NSExtensionMainStoryboard</key> | ||||
| 		<string>MainInterface</string> | ||||
| 		<key>NSExtensionPointIdentifier</key> | ||||
| 		<string>com.apple.share-services</string> | ||||
| 	</dict> | ||||
| 	<key>AppGroupId</key> | ||||
| 	<string>group.solsynth.solian</string> | ||||
| </dict> | ||||
| </plist> | ||||
							
								
								
									
										18
									
								
								ios/SolarShare/ShareViewController.swift
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,18 @@ | ||||
| // | ||||
| //  ShareViewController.swift | ||||
| //  SolarShare | ||||
| // | ||||
| //  Created by LittleSheep on 2024/12/15. | ||||
| // | ||||
|  | ||||
| import receive_sharing_intent | ||||
|  | ||||
| class ShareViewController: RSIShareViewController { | ||||
|        | ||||
|     // Use this method to return false if you don't want to redirect to host app automatically. | ||||
|     // Default is true | ||||
|     override func shouldAutoRedirect() -> Bool { | ||||
|         return true | ||||
|     } | ||||
|      | ||||
| } | ||||
							
								
								
									
										10
									
								
								ios/SolarShare/SolarShare.entitlements
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,10 @@ | ||||
| <?xml version="1.0" encoding="UTF-8"?> | ||||
| <!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd"> | ||||
| <plist version="1.0"> | ||||
| <dict> | ||||
| 	<key>com.apple.security.application-groups</key> | ||||
| 	<array> | ||||
| 		<string>group.solsynth.solian</string> | ||||
| 	</array> | ||||
| </dict> | ||||
| </plist> | ||||
| @@ -0,0 +1,11 @@ | ||||
| { | ||||
|   "colors" : [ | ||||
|     { | ||||
|       "idiom" : "universal" | ||||
|     } | ||||
|   ], | ||||
|   "info" : { | ||||
|     "author" : "xcode", | ||||
|     "version" : 1 | ||||
|   } | ||||
| } | ||||
| @@ -0,0 +1,35 @@ | ||||
| { | ||||
|   "images" : [ | ||||
|     { | ||||
|       "idiom" : "universal", | ||||
|       "platform" : "ios", | ||||
|       "size" : "1024x1024" | ||||
|     }, | ||||
|     { | ||||
|       "appearances" : [ | ||||
|         { | ||||
|           "appearance" : "luminosity", | ||||
|           "value" : "dark" | ||||
|         } | ||||
|       ], | ||||
|       "idiom" : "universal", | ||||
|       "platform" : "ios", | ||||
|       "size" : "1024x1024" | ||||
|     }, | ||||
|     { | ||||
|       "appearances" : [ | ||||
|         { | ||||
|           "appearance" : "luminosity", | ||||
|           "value" : "tinted" | ||||
|         } | ||||
|       ], | ||||
|       "idiom" : "universal", | ||||
|       "platform" : "ios", | ||||
|       "size" : "1024x1024" | ||||
|     } | ||||
|   ], | ||||
|   "info" : { | ||||
|     "author" : "xcode", | ||||
|     "version" : 1 | ||||
|   } | ||||
| } | ||||
							
								
								
									
										6
									
								
								ios/SolarWidget/Assets.xcassets/Contents.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,6 @@ | ||||
| { | ||||
|   "info" : { | ||||
|     "author" : "xcode", | ||||
|     "version" : 1 | ||||
|   } | ||||
| } | ||||
| @@ -0,0 +1,11 @@ | ||||
| { | ||||
|   "colors" : [ | ||||
|     { | ||||
|       "idiom" : "universal" | ||||
|     } | ||||
|   ], | ||||
|   "info" : { | ||||
|     "author" : "xcode", | ||||
|     "version" : 1 | ||||
|   } | ||||
| } | ||||
							
								
								
									
										144
									
								
								ios/SolarWidget/CheckInWidget.swift
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,144 @@ | ||||
| // | ||||
| //  SolarWidget.swift | ||||
| //  SolarWidget | ||||
| // | ||||
| //  Created by LittleSheep on 2024/12/14. | ||||
| // | ||||
|  | ||||
| import WidgetKit | ||||
| import SwiftUI | ||||
|  | ||||
| struct CheckInProvider: TimelineProvider { | ||||
|     func placeholder(in context: Context) -> CheckInEntry { | ||||
|         CheckInEntry(date: Date(), user: nil, checkIn: nil) | ||||
|     } | ||||
|  | ||||
|     func getSnapshot(in context: Context, completion: @escaping (CheckInEntry) -> ()) { | ||||
|         let prefs = UserDefaults(suiteName: "group.solsynth.solian") | ||||
|          | ||||
|         let dateFormatter = DateFormatter() | ||||
|         dateFormatter.dateFormat = "yyyy-MM-dd'T'HH:mm:ss.SSSSSS'Z'" | ||||
|          | ||||
|         let jsonDecoder = JSONDecoder() | ||||
|         jsonDecoder.dateDecodingStrategy = .formatted(dateFormatter) | ||||
|         jsonDecoder.keyDecodingStrategy = .convertFromSnakeCase | ||||
|          | ||||
|         let userRaw = prefs?.string(forKey: "user") | ||||
|         var user: SolarUser? | ||||
|         if let userRaw = userRaw { | ||||
|             user = try! jsonDecoder.decode(SolarUser.self, from: userRaw.data(using: .utf8)!) | ||||
|         } | ||||
|          | ||||
|         let checkInRaw = prefs?.string(forKey: "pas_check_in_record") | ||||
|         var checkIn: SolarCheckInRecord? | ||||
|         if let checkInRaw = checkInRaw { | ||||
|             checkIn = try! jsonDecoder.decode(SolarCheckInRecord.self, from: checkInRaw.data(using: .utf8)!) | ||||
|             if checkIn != nil && Calendar.current.isDate(checkIn!.createdAt, inSameDayAs: Date()) { | ||||
|                 checkIn = nil | ||||
|             } | ||||
|         } | ||||
|          | ||||
|         let entry = CheckInEntry( | ||||
|             date: Date(), | ||||
|             user: user, | ||||
|             checkIn: checkIn | ||||
|         ) | ||||
|         completion(entry) | ||||
|     } | ||||
|  | ||||
|     func getTimeline(in context: Context, completion: @escaping (Timeline<Entry>) -> ()) { | ||||
|         getSnapshot(in: context) { (entry) in | ||||
|             let timeline = Timeline(entries: [entry], policy: .atEnd) | ||||
|             completion(timeline) | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| struct CheckInEntry: TimelineEntry { | ||||
|     let date: Date | ||||
|     let user: SolarUser? | ||||
|     let checkIn: SolarCheckInRecord? | ||||
| } | ||||
|  | ||||
| struct CheckInWidgetEntryView : View { | ||||
|     var entry: CheckInProvider.Entry | ||||
|      | ||||
|     private let resultTierSymbols: [String] = ["大凶", "凶", "中平", "吉", "大吉"] | ||||
|      | ||||
|     func checkIn() -> Void {} | ||||
|      | ||||
|     func seeDetail() -> Void {} | ||||
|  | ||||
|     var body: some View { | ||||
|         VStack(alignment: .leading) { | ||||
|             if let checkIn = entry.checkIn { | ||||
|                 VStack(alignment: .leading) { | ||||
|                     Text(resultTierSymbols[checkIn.resultTier]).font(.system(size: 27, weight: .bold)) | ||||
|                     Text("+\(checkIn.resultExperience) EXP").font(.system(size: 15, design: .monospaced)) | ||||
|                 }.padding(.horizontal, 4) | ||||
|                  | ||||
|                 Spacer() | ||||
|                  | ||||
|                 HStack { | ||||
|                     VStack(alignment: .leading) { | ||||
|                         Text( | ||||
|                             checkIn.createdAt, | ||||
|                             format: .dateTime.weekday() | ||||
|                         ).font(.system(size: 13)) | ||||
|                         Text( | ||||
|                             checkIn.createdAt, | ||||
|                             format: .dateTime.day().month() | ||||
|                         ).font(.system(size: 13)) | ||||
|                     }.padding(.leading, 4) | ||||
|                      | ||||
|                     Button("See Detail", systemImage: "arrow.right", action: seeDetail) | ||||
|                         .labelStyle(.iconOnly) | ||||
|                         .buttonBorderShape(.circle) | ||||
|                         .frame(maxWidth: .infinity, alignment: .trailing) | ||||
|                 }.frame(alignment: .bottom) | ||||
|             } else { | ||||
|                 VStack(alignment: .leading) { | ||||
|                     Text("Check In").font(.system(size: 19, weight: .bold)) | ||||
|                     Text("You haven't check in today").font(.system(size: 15)) | ||||
|                 }.padding(.horizontal, 4) | ||||
|                  | ||||
|                 Spacer() | ||||
|                  | ||||
|                 HStack(alignment: .bottom) { | ||||
|                     Button("Check In", systemImage: "checkmark", action: checkIn).labelStyle(.iconOnly).buttonBorderShape(.circle).frame(maxWidth: .infinity, alignment: .trailing) | ||||
|                 } | ||||
|             } | ||||
|         }.padding(8).widgetURL(URL(string: "https://sn.solsynth.dev")) | ||||
|     } | ||||
| } | ||||
|  | ||||
| struct CheckInWidget: Widget { | ||||
|     let kind: String = "SolarCheckInWidget" | ||||
|  | ||||
|     var body: some WidgetConfiguration { | ||||
|         StaticConfiguration(kind: kind, provider: CheckInProvider()) { entry in | ||||
|             if #available(iOS 17.0, *) { | ||||
|                 CheckInWidgetEntryView(entry: entry) | ||||
|                     .containerBackground(.fill.tertiary, for: .widget) | ||||
|             } else { | ||||
|                 CheckInWidgetEntryView(entry: entry) | ||||
|                     .padding() | ||||
|                     .background() | ||||
|             } | ||||
|         } | ||||
|         .configurationDisplayName("Check In") | ||||
|         .description("View your today's fortune on your home screen") | ||||
|         .supportedFamilies([.systemSmall, .systemMedium]) | ||||
|     } | ||||
| } | ||||
|  | ||||
| #Preview(as: .systemSmall) { | ||||
|     CheckInWidget() | ||||
| } timeline: { | ||||
|     CheckInEntry(date: .now, user: nil, checkIn: nil) | ||||
|     CheckInEntry( | ||||
|         date: .now, | ||||
|         user: SolarUser(id: 1, name: "demo", nick: "Deemo"), | ||||
|         checkIn: SolarCheckInRecord(id: 1, resultTier: 1, resultExperience: 100, createdAt: Date.now) | ||||
|     ) | ||||
| } | ||||
							
								
								
									
										11
									
								
								ios/SolarWidget/Info.plist
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,11 @@ | ||||
| <?xml version="1.0" encoding="UTF-8"?> | ||||
| <!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd"> | ||||
| <plist version="1.0"> | ||||
| <dict> | ||||
| 	<key>NSExtension</key> | ||||
| 	<dict> | ||||
| 		<key>NSExtensionPointIdentifier</key> | ||||
| 		<string>com.apple.widgetkit-extension</string> | ||||
| 	</dict> | ||||
| </dict> | ||||
| </plist> | ||||
							
								
								
									
										235
									
								
								ios/SolarWidget/RandomPostWidget.swift
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,235 @@ | ||||
| // | ||||
| //  RandomPostWidget.swift | ||||
| //  Runner | ||||
| // | ||||
| //  Created by LittleSheep on 2024/12/14. | ||||
| // | ||||
|  | ||||
| import SwiftUI | ||||
| import WidgetKit | ||||
| import Kingfisher | ||||
|  | ||||
| struct RandomPostProvider: TimelineProvider { | ||||
|     func placeholder(in context: Context) -> RandomPostEntry { | ||||
|         RandomPostEntry(date: Date(), user: nil, randomPost: nil, family: .systemMedium) | ||||
|     } | ||||
|  | ||||
|     func getSnapshot(in context: Context, completion: @escaping (RandomPostEntry) -> ()) { | ||||
|         let prefs = UserDefaults(suiteName: "group.solsynth.solian") | ||||
|          | ||||
|         let dateFormatter = DateFormatter() | ||||
|         dateFormatter.dateFormat = "yyyy-MM-dd'T'HH:mm:ss.SSSSSS'Z'" | ||||
|          | ||||
|         let jsonDecoder = JSONDecoder() | ||||
|         jsonDecoder.dateDecodingStrategy = .formatted(dateFormatter) | ||||
|         jsonDecoder.keyDecodingStrategy = .convertFromSnakeCase | ||||
|          | ||||
|         let userRaw = prefs?.string(forKey: "user") | ||||
|         var user: SolarUser? | ||||
|         if let userRaw = userRaw { | ||||
|             user = try! jsonDecoder.decode(SolarUser.self, from: userRaw.data(using: .utf8)!) | ||||
|         } | ||||
|          | ||||
|         let randomPostRaw = prefs?.string(forKey: "int_random_post") | ||||
|         var randomPost: SolarPost? | ||||
|         if let randomPostRaw = randomPostRaw { | ||||
|             randomPost = try! jsonDecoder.decode(SolarPost.self, from: randomPostRaw.data(using: .utf8)!) | ||||
|         } | ||||
|          | ||||
|          | ||||
|         let entry = RandomPostEntry( | ||||
|             date: Date(), | ||||
|             user: user, | ||||
|             randomPost: randomPost, | ||||
|             family: context.family | ||||
|         ) | ||||
|         completion(entry) | ||||
|     } | ||||
|  | ||||
|     func getTimeline(in context: Context, completion: @escaping (Timeline<Entry>) -> ()) { | ||||
|         getSnapshot(in: context) { (entry) in | ||||
|             let timeline = Timeline(entries: [entry], policy: .atEnd) | ||||
|             completion(timeline) | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| struct RandomPostEntry: TimelineEntry { | ||||
|     let date: Date | ||||
|     let user: SolarUser? | ||||
|     let randomPost: SolarPost? | ||||
|      | ||||
|     let family: WidgetFamily | ||||
| } | ||||
|  | ||||
| struct RandomPostWidgetEntryView : View { | ||||
|     var entry: RandomPostProvider.Entry | ||||
|  | ||||
|     var body: some View { | ||||
|         VStack(alignment: .leading, spacing: 0) { | ||||
|             if let randomPost = entry.randomPost { | ||||
|                 VStack(alignment: .leading, spacing: 0) { | ||||
|                     HStack(alignment: .center) { | ||||
|                         if let avatar = randomPost.publisher.avatar { | ||||
|                             let avatarUrl = getAttachmentUrl(for: avatar) | ||||
|                             let size: CGFloat = 28 | ||||
|                              | ||||
|                             KFImage.url(URL(string: avatarUrl)) | ||||
|                                 .resizable() | ||||
|                                 .setProcessor(ResizingImageProcessor(referenceSize: CGSize(width: size, height: size), mode: .aspectFit)) | ||||
|                                 .aspectRatio(contentMode: .fit) | ||||
|                                 .frame(width: size, height: size) | ||||
|                                 .cornerRadius(size / 2) | ||||
|                                 .frame(width: size, height: size, alignment: .center) | ||||
|                         } | ||||
|                          | ||||
|                         Text("@\(randomPost.publisher.name)") | ||||
|                             .font(.system(size: 13, design: .monospaced)) | ||||
|                             .opacity(0.9) | ||||
|                          | ||||
|                         Spacer() | ||||
|                     }.frame(maxWidth: .infinity).padding(.bottom, 12) | ||||
|                      | ||||
|                     if randomPost.body.title != nil || randomPost.body.description != nil { | ||||
|                         VStack(alignment: .leading) { | ||||
|                             if let title = randomPost.body.title { | ||||
|                                 Text(title) | ||||
|                                     .font(.system(size: 17)) | ||||
|                             } | ||||
|                             if let description = randomPost.body.description { | ||||
|                                 Text(description) | ||||
|                                     .font(.system(size: 15)) | ||||
|                             } | ||||
|                         }.padding(.bottom, 8) | ||||
|                     } | ||||
|                      | ||||
|                     if let content = randomPost.body.content { | ||||
|                         if (randomPost.body.title == nil && randomPost.body.description == nil) || entry.family == .systemLarge || entry.family == .systemExtraLarge { | ||||
|                             Text( | ||||
|                                 (entry.family == .systemLarge || entry.family == .systemExtraLarge) ? content : content.replacingOccurrences(of: "\n", with: " ") | ||||
|                             ) | ||||
|                             .font(.system(size: 15)) | ||||
|                         } else { | ||||
|                             Text("\(Image(systemName: "plus")) total \(content.count) characters") | ||||
|                                 .font(.system(size: 11, design: .monospaced)) | ||||
|                                 .opacity(0.75) | ||||
|                                 .padding(.top, 1) | ||||
|                         } | ||||
|                     } | ||||
|                      | ||||
|                     if let attachment = randomPost.body.attachments { | ||||
|                         if attachment.count == 1 { | ||||
|                             Text("\(Image(systemName: "document.fill")) \(attachment.count) attachment") | ||||
|                                 .font(.system(size: 11, design: .monospaced)) | ||||
|                                 .opacity(0.75) | ||||
|                                 .padding(.top, 2) | ||||
|                         } else if attachment.count > 1 { | ||||
|                             Text("\(Image(systemName: "document.fill")) \(attachment.count) attachments") | ||||
|                                 .font(.system(size: 11, design: .monospaced)) | ||||
|                                 .opacity(0.75) | ||||
|                                 .padding(.top, 2) | ||||
|                         } | ||||
|                     } | ||||
|                      | ||||
|                     Spacer() | ||||
|                      | ||||
|                     Text(randomPost.publishedAt!, format: .dateTime) | ||||
|                         .font(.system(size: 11)) | ||||
|                     Text("#\(randomPost.id)") | ||||
|                         .font(.system(size: 9)) | ||||
|                 }.widgetURL(URL(string: "https://sn.solsynth.dev/posts/\(randomPost.id)")) | ||||
|             } else { | ||||
|                 VStack(alignment: .center) { | ||||
|                     Text("No Recommendations").font(.system(size: 19, weight: .bold)) | ||||
|                     Text("Open the app to load some random post") | ||||
|                         .font(.system(size: 15)) | ||||
|                         .multilineTextAlignment(.center) | ||||
|                 }.frame(alignment: .center) | ||||
|             } | ||||
|         }.padding(8).frame(maxWidth: .infinity) | ||||
|     } | ||||
| } | ||||
|  | ||||
| struct RandomPostWidget: Widget { | ||||
|     let kind: String = "SolarRandomPostWidget" | ||||
|  | ||||
|     var body: some WidgetConfiguration { | ||||
|         StaticConfiguration(kind: kind, provider: RandomPostProvider()) { entry in | ||||
|             if #available(iOS 17.0, *) { | ||||
|                 RandomPostWidgetEntryView(entry: entry) | ||||
|                     .containerBackground(.fill.tertiary, for: .widget) | ||||
|             } else { | ||||
|                 RandomPostWidgetEntryView(entry: entry) | ||||
|                     .padding() | ||||
|                     .background() | ||||
|             } | ||||
|         } | ||||
|         .configurationDisplayName("Random Post") | ||||
|         .description("View the random post on the Solar Network") | ||||
|         .supportedFamilies([.systemSmall, .systemMedium, .systemLarge, .systemExtraLarge]) | ||||
|     } | ||||
| } | ||||
|  | ||||
| #Preview(as: .systemSmall) { | ||||
|     RandomPostWidget() | ||||
| } timeline: { | ||||
|     RandomPostEntry(date: Date.now, user: nil, randomPost: nil, family: .systemLarge) | ||||
|     RandomPostEntry( | ||||
|         date: .now, | ||||
|         user: SolarUser(id: 1, name: "demo", nick: "Deemo"), | ||||
|         randomPost: SolarPost( | ||||
|             id: 1, | ||||
|             body: SolarPostBody( | ||||
|                 content: "Hello, World", | ||||
|                 title: nil, | ||||
|                 description: nil, | ||||
|                 attachments: ["zb2hiUEmYcnpHfVN"] | ||||
|             ), | ||||
|             publisher: SolarPublisher( | ||||
|                 id: 1, | ||||
|                 name: "demo", | ||||
|                 nick: "Deemo", | ||||
|                 description: nil, | ||||
|                 avatar: "IZxCFkJUPKRijFCx", | ||||
|                 banner: nil, | ||||
|                 createdAt: .now, | ||||
|                 updatedAt: .now | ||||
|             ), | ||||
|             publisherId: 1, | ||||
|             createdAt: .now, | ||||
|             updatedAt: .now, | ||||
|             editedAt: nil, | ||||
|             publishedAt: .now | ||||
|         ), | ||||
|         family: .systemSmall | ||||
|     ) | ||||
|     RandomPostEntry( | ||||
|         date: .now, | ||||
|         user: SolarUser(id: 1, name: "demo", nick: "Deemo"), | ||||
|         randomPost: SolarPost( | ||||
|             id: 1, | ||||
|             body: SolarPostBody( | ||||
|                 content: "Hello, World\nOh wow", | ||||
|                 title: "Title", | ||||
|                 description: "Description", | ||||
|                 attachments: ["zb2hiUEmYcnpHfVN"] | ||||
|             ), | ||||
|             publisher: SolarPublisher( | ||||
|                 id: 1, | ||||
|                 name: "demo", | ||||
|                 nick: "Deemo", | ||||
|                 description: nil, | ||||
|                 avatar: "IZxCFkJUPKRijFCx", | ||||
|                 banner: nil, | ||||
|                 createdAt: .now, | ||||
|                 updatedAt: .now | ||||
|             ), | ||||
|             publisherId: 1, | ||||
|             createdAt: .now, | ||||
|             updatedAt: .now, | ||||
|             editedAt: nil, | ||||
|             publishedAt: .now | ||||
|         ), | ||||
|         family: .systemLarge | ||||
|     ) | ||||
| } | ||||
							
								
								
									
										17
									
								
								ios/SolarWidget/SolarWidgetBundle.swift
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,17 @@ | ||||
| // | ||||
| //  SolarWidgetBundle.swift | ||||
| //  SolarWidget | ||||
| // | ||||
| //  Created by LittleSheep on 2024/12/14. | ||||
| // | ||||
|  | ||||
| import WidgetKit | ||||
| import SwiftUI | ||||
|  | ||||
| @main | ||||
| struct SolarWidgetBundle: WidgetBundle { | ||||
|     var body: some Widget { | ||||
|         CheckInWidget() | ||||
|         RandomPostWidget() | ||||
|     } | ||||
| } | ||||
							
								
								
									
										10
									
								
								ios/SolarWidgetExtension.entitlements
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,10 @@ | ||||
| <?xml version="1.0" encoding="UTF-8"?> | ||||
| <!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd"> | ||||
| <plist version="1.0"> | ||||
| <dict> | ||||
| 	<key>com.apple.security.application-groups</key> | ||||
| 	<array> | ||||
| 		<string>group.solsynth.solian</string> | ||||
| 	</array> | ||||
| </dict> | ||||
| </plist> | ||||
							
								
								
									
										432
									
								
								lib/controllers/chat_message_controller.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,432 @@ | ||||
| import 'dart:async'; | ||||
| import 'dart:math' as math; | ||||
|  | ||||
| import 'package:collection/collection.dart'; | ||||
| import 'package:dio/dio.dart'; | ||||
| import 'package:flutter/material.dart'; | ||||
| import 'package:hive/hive.dart'; | ||||
| import 'package:provider/provider.dart'; | ||||
| import 'package:surface/providers/sn_attachment.dart'; | ||||
| 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:uuid/uuid.dart'; | ||||
|  | ||||
| class ChatMessageController extends ChangeNotifier { | ||||
|   static const kChatMessageBoxPrefix = 'nex_chat_messages_'; | ||||
|   static const kSingleBatchLoadLimit = 100; | ||||
|  | ||||
|   late final SnNetworkProvider _sn; | ||||
|   late final UserDirectoryProvider _ud; | ||||
|   late final WebSocketProvider _ws; | ||||
|   late final SnAttachmentProvider _attach; | ||||
|  | ||||
|   StreamSubscription? _wsSubscription; | ||||
|  | ||||
|   ChatMessageController(BuildContext context) { | ||||
|     _sn = context.read<SnNetworkProvider>(); | ||||
|     _ud = context.read<UserDirectoryProvider>(); | ||||
|     _ws = context.read<WebSocketProvider>(); | ||||
|     _attach = context.read<SnAttachmentProvider>(); | ||||
|   } | ||||
|  | ||||
|   bool isPending = true; | ||||
|   bool isLoading = false; | ||||
|  | ||||
|   int? messageTotal; | ||||
|  | ||||
|   bool get isAllLoaded => | ||||
|       messageTotal != null && messages.length >= messageTotal!; | ||||
|  | ||||
|   String? _boxKey; | ||||
|   SnChannel? channel; | ||||
|   SnChannelMember? profile; | ||||
|  | ||||
|   /// Messages are the all the messages that in the channel | ||||
|   final List<SnChatMessage> messages = List.empty(growable: true); | ||||
|  | ||||
|   /// Unconfirmed messages are the messages that sent by client but did not receive the reply from websocket server. | ||||
|   /// 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!); | ||||
|  | ||||
|   Future<void> initialize(SnChannel chan) async { | ||||
|     channel = chan; | ||||
|  | ||||
|     // Initialize local data | ||||
|     _boxKey = '$kChatMessageBoxPrefix${chan.id}'; | ||||
|     await Hive.openBox<SnChatMessage>(_boxKey!); | ||||
|  | ||||
|     // Fetch channel profile | ||||
|     final resp = await _sn.client.get( | ||||
|       '/cgi/im/channels/${chan.keyPath}/me', | ||||
|     ); | ||||
|     profile = SnChannelMember.fromJson( | ||||
|       resp.data as Map<String, dynamic>, | ||||
|     ); | ||||
|  | ||||
|     _wsSubscription = _ws.stream.stream.listen((event) { | ||||
|       switch (event.method) { | ||||
|         case 'events.new': | ||||
|           final payload = SnChatMessage.fromJson(event.payload!); | ||||
|           _addMessage(payload); | ||||
|           break; | ||||
|         case 'status.typing': | ||||
|           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); | ||||
|         //     }); | ||||
|         //   }, | ||||
|         // ); | ||||
|       } | ||||
|     }); | ||||
|  | ||||
|     isPending = false; | ||||
|     notifyListeners(); | ||||
|   } | ||||
|  | ||||
|   Future<void> _saveMessageToLocal(Iterable<SnChatMessage> messages) async { | ||||
|     if (_box == null) return; | ||||
|     await _box!.putAll({ | ||||
|       for (final message in messages) message.id: message, | ||||
|     }); | ||||
|   } | ||||
|  | ||||
|   Future<void> _addUnconfirmedMessage(SnChatMessage message) async { | ||||
|     SnChatMessage? quoteEvent; | ||||
|     if (message.quoteEventId != null) { | ||||
|       quoteEvent = await getMessage(message.quoteEventId as int); | ||||
|     } | ||||
|  | ||||
|     final attachmentRid = List<String>.from( | ||||
|       message.body['attachments']?.cast<String>() ?? [], | ||||
|     ); | ||||
|     final attachments = await _attach.getMultiple(attachmentRid); | ||||
|     message = message.copyWith( | ||||
|       preload: SnChatMessagePreload( | ||||
|         quoteEvent: quoteEvent, | ||||
|         attachments: attachments, | ||||
|       ), | ||||
|     ); | ||||
|  | ||||
|     messages.insert(0, message); | ||||
|     unconfirmedMessages.add(message.uuid); | ||||
|     notifyListeners(); | ||||
|   } | ||||
|  | ||||
|   Future<void> _addMessage(SnChatMessage message) async { | ||||
|     SnChatMessage? quoteEvent; | ||||
|     if (message.quoteEventId != null) { | ||||
|       quoteEvent = await getMessage(message.quoteEventId as int); | ||||
|     } | ||||
|  | ||||
|     final attachmentRid = List<String>.from( | ||||
|       message.body['attachments']?.cast<String>() ?? [], | ||||
|     ); | ||||
|     final attachments = await _attach.getMultiple(attachmentRid); | ||||
|     message = message.copyWith( | ||||
|       preload: SnChatMessagePreload( | ||||
|         quoteEvent: quoteEvent, | ||||
|         attachments: attachments, | ||||
|       ), | ||||
|     ); | ||||
|  | ||||
|     final idx = messages.indexWhere((e) => e.uuid == message.uuid); | ||||
|     if (idx != -1) { | ||||
|       unconfirmedMessages.remove(message.uuid); | ||||
|       messages[idx] = message; | ||||
|     } else { | ||||
|       messages.insert(0, message); | ||||
|     } | ||||
|     await _applyMessage(message); | ||||
|     notifyListeners(); | ||||
|  | ||||
|     if (_box == null) return; | ||||
|     await _box!.put(message.id, message); | ||||
|   } | ||||
|  | ||||
|   Future<void> _applyMessage(SnChatMessage message) async { | ||||
|     if (message.channelId != channel?.id) return; | ||||
|  | ||||
|     switch (message.type) { | ||||
|       case 'messages.edit': | ||||
|         if (message.relatedEventId != null) { | ||||
|           final idx = | ||||
|               messages.indexWhere((x) => x.id == message.relatedEventId); | ||||
|           if (idx != -1) { | ||||
|             final newBody = message.body; | ||||
|             newBody.remove('related_event'); | ||||
|             messages[idx] = messages[idx].copyWith( | ||||
|               body: newBody, | ||||
|               updatedAt: message.updatedAt, | ||||
|             ); | ||||
|             if (_box!.containsKey(message.relatedEventId)) { | ||||
|               await _box!.put(message.relatedEventId, messages[idx]); | ||||
|             } | ||||
|           } | ||||
|         } | ||||
|       case 'messages.delete': | ||||
|         if (message.relatedEventId != null) { | ||||
|           messages.removeWhere((x) => x.id == message.relatedEventId); | ||||
|           if (_box!.containsKey(message.relatedEventId)) { | ||||
|             await _box!.delete(message.relatedEventId); | ||||
|           } | ||||
|         } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   Future<void> sendMessage( | ||||
|     String type, | ||||
|     String content, { | ||||
|     int? quoteId, | ||||
|     int? relatedId, | ||||
|     List<String>? attachments, | ||||
|     SnChatMessage? editingMessage, | ||||
|   }) async { | ||||
|     if (channel == null) return; | ||||
|     const uuid = Uuid(); | ||||
|     final nonce = uuid.v4(); | ||||
|     final body = { | ||||
|       'text': content, | ||||
|       'algorithm': 'plain', | ||||
|       if (quoteId != null) 'quote_event': quoteId, | ||||
|       if (relatedId != null) 'related_event': relatedId, | ||||
|       if (attachments != null && attachments.isNotEmpty) | ||||
|         'attachments': attachments, | ||||
|     }; | ||||
|  | ||||
|     // Mock the message locally | ||||
|     final createdAt = DateTime.now(); | ||||
|     final message = SnChatMessage( | ||||
|       id: 0, | ||||
|       createdAt: createdAt, | ||||
|       updatedAt: createdAt, | ||||
|       deletedAt: null, | ||||
|       uuid: nonce, | ||||
|       body: body, | ||||
|       type: type, | ||||
|       channel: channel!, | ||||
|       channelId: channel!.id, | ||||
|       sender: profile!, | ||||
|       senderId: profile!.id, | ||||
|       quoteEventId: quoteId, | ||||
|       relatedEventId: relatedId, | ||||
|     ); | ||||
|     _addUnconfirmedMessage(message); | ||||
|  | ||||
|     // Send to server | ||||
|     try { | ||||
|       await _sn.client.request( | ||||
|         editingMessage != null | ||||
|             ? '/cgi/im/channels/${channel!.keyPath}/messages/${editingMessage.id}' | ||||
|             : '/cgi/im/channels/${channel!.keyPath}/messages', | ||||
|         data: { | ||||
|           'type': type, | ||||
|           'uuid': nonce, | ||||
|           'body': body, | ||||
|         }, | ||||
|         options: Options( | ||||
|           method: editingMessage != null ? 'PUT' : 'POST', | ||||
|         ), | ||||
|       ); | ||||
|     } catch (err) { | ||||
|       // ignore | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   Future<void> deleteMessage(SnChatMessage message) async { | ||||
|     if (message.channelId != channel?.id) return; | ||||
|  | ||||
|     try { | ||||
|       await _sn.client.delete( | ||||
|         '/cgi/im/channels/${channel!.keyPath}/messages/${message.id}', | ||||
|       ); | ||||
|     } catch (err) { | ||||
|       // ignore | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   /// Check the local storage is up to date with the server. | ||||
|   /// If the local storage is not up to date, it will be updated. | ||||
|   Future<void> checkUpdate() async { | ||||
|     if (_box == null) return; | ||||
|     if (_box!.isEmpty) return; | ||||
|  | ||||
|     isLoading = true; | ||||
|     notifyListeners(); | ||||
|  | ||||
|     try { | ||||
|       final resp = await _sn.client.get( | ||||
|         '/cgi/im/channels/${channel!.keyPath}/events/update', | ||||
|         queryParameters: { | ||||
|           'pivot': _box!.values.last.id, | ||||
|         }, | ||||
|       ); | ||||
|       if (resp.data['up_to_date'] == true) return; | ||||
|       // Only preload the first 100 messages to prevent first time check update cause load to server and waste local storage. | ||||
|       // FIXME If the local is missing more than 100 messages, it won't be fetched, this is a problem, we need to fix it. | ||||
|       final countToFetch = math.min(resp.data['count'] as int, 100); | ||||
|  | ||||
|       for (int idx = 0; idx < countToFetch; idx += kSingleBatchLoadLimit) { | ||||
|         await getMessages(kSingleBatchLoadLimit, idx, forceRemote: true); | ||||
|       } | ||||
|     } catch (err) { | ||||
|       rethrow; | ||||
|     } finally { | ||||
|       await loadMessages(); | ||||
|       isLoading = false; | ||||
|       notifyListeners(); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   /// Get a single event from the current channel | ||||
|   /// If it was not found in local storage we will look it up in remote | ||||
|   Future<SnChatMessage?> getMessage(int id) async { | ||||
|     SnChatMessage? out; | ||||
|     if (_box != null && _box!.containsKey(id)) { | ||||
|       out = _box!.get(id); | ||||
|     } | ||||
|  | ||||
|     if (out == null) { | ||||
|       try { | ||||
|         final resp = await _sn.client | ||||
|             .get('/cgi/im/channels/${channel!.keyPath}/events/$id'); | ||||
|         out = SnChatMessage.fromJson(resp.data); | ||||
|         _saveMessageToLocal([out]); | ||||
|       } catch (_) { | ||||
|         // ignore, maybe not found | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     // Preload some related things if found | ||||
|     if (out != null) { | ||||
|       await _ud.listAccount([out.sender.accountId]); | ||||
|  | ||||
|       final attachments = await _attach.getMultiple( | ||||
|         out.body['attachments']?.cast<String>() ?? [], | ||||
|       ); | ||||
|       out = out.copyWith( | ||||
|         preload: SnChatMessagePreload( | ||||
|           attachments: attachments, | ||||
|         ), | ||||
|       ); | ||||
|     } | ||||
|  | ||||
|     return out; | ||||
|   } | ||||
|  | ||||
|   /// Get message from local storage first, then from the server. | ||||
|   /// Will not check local storage is up to date with the server. | ||||
|   /// If you need to do the sync, do the `checkUpdate` instead. | ||||
|   Future<List<SnChatMessage>> getMessages( | ||||
|     int take, | ||||
|     int offset, { | ||||
|     bool forceLocal = false, | ||||
|     bool forceRemote = false, | ||||
|   }) async { | ||||
|     late List<SnChatMessage> out; | ||||
|     if (_box != null && | ||||
|         (_box!.length >= take + offset || forceLocal) && | ||||
|         !forceRemote) { | ||||
|       out = _box!.keys | ||||
|           .toList() | ||||
|           .cast<int>() | ||||
|           .sorted((a, b) => b.compareTo(a)) | ||||
|           .skip(offset) | ||||
|           .take(take) | ||||
|           .map((key) => _box!.get(key)!) | ||||
|           .toList(); | ||||
|     } else { | ||||
|       final resp = await _sn.client.get( | ||||
|         '/cgi/im/channels/${channel!.keyPath}/events', | ||||
|         queryParameters: { | ||||
|           'take': take, | ||||
|           'offset': offset, | ||||
|         }, | ||||
|       ); | ||||
|       messageTotal = resp.data['count'] as int?; | ||||
|       out = List<SnChatMessage>.from( | ||||
|         resp.data['data']?.map((e) => SnChatMessage.fromJson(e)) ?? [], | ||||
|       ); | ||||
|       _saveMessageToLocal(out); | ||||
|     } | ||||
|  | ||||
|     // Preload attachments | ||||
|     final attachmentRid = List<String>.from( | ||||
|       out.expand((e) => (e.body['attachments'] as List<dynamic>?) ?? []), | ||||
|     ); | ||||
|     final attachments = await _attach.getMultiple(attachmentRid); | ||||
|  | ||||
|     // Putting preload back to data | ||||
|     for (var i = 0; i < out.length; i++) { | ||||
|       // Preload related events (quoted) | ||||
|       SnChatMessage? quoteEvent; | ||||
|       if (out[i].quoteEventId != null) { | ||||
|         quoteEvent = await getMessage(out[i].quoteEventId as int); | ||||
|       } | ||||
|  | ||||
|       out[i] = out[i].copyWith( | ||||
|         preload: SnChatMessagePreload( | ||||
|           quoteEvent: quoteEvent, | ||||
|           attachments: attachments | ||||
|               .where( | ||||
|                 (ele) => | ||||
|                     out[i].body['attachments']?.contains(ele?.rid) ?? false, | ||||
|               ) | ||||
|               .toList(), | ||||
|         ), | ||||
|       ); | ||||
|     } | ||||
|  | ||||
|     // Preload sender accounts | ||||
|     final accountId = out | ||||
|         .where((ele) => ele.sender.accountId >= 0) | ||||
|         .map((ele) => ele.sender.accountId) | ||||
|         .toSet(); | ||||
|     await _ud.listAccount(accountId); | ||||
|  | ||||
|     return out; | ||||
|   } | ||||
|  | ||||
|   /// The load messages method work as same as the `getMessages` method. | ||||
|   /// But it won't return the messages instead append them to the value that controller has. | ||||
|   /// At the same time, this method provide the `isLoading` state. | ||||
|   /// The `skip` parameter is no longer required since it will skip the messages count that already loaded. | ||||
|   Future<void> loadMessages({int take = 20}) async { | ||||
|     isLoading = true; | ||||
|     notifyListeners(); | ||||
|  | ||||
|     try { | ||||
|       final out = await getMessages(take, messages.length); | ||||
|       messages.addAll(out); | ||||
|     } catch (err) { | ||||
|       rethrow; | ||||
|     } finally { | ||||
|       isLoading = false; | ||||
|       notifyListeners(); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   void dispose() { | ||||
|     _box?.close(); | ||||
|     _wsSubscription?.cancel(); | ||||
|     super.dispose(); | ||||
|   } | ||||
| } | ||||
| @@ -6,6 +6,7 @@ import 'package:flutter/material.dart'; | ||||
| import 'package:image_picker/image_picker.dart'; | ||||
| import 'package:mime/mime.dart'; | ||||
| import 'package:provider/provider.dart'; | ||||
| import 'package:surface/providers/post.dart'; | ||||
| import 'package:surface/providers/sn_attachment.dart'; | ||||
| import 'package:surface/providers/sn_network.dart'; | ||||
| import 'package:surface/types/attachment.dart'; | ||||
| @@ -27,6 +28,8 @@ class PostWriteMedia { | ||||
|   final XFile? file; | ||||
|   final Uint8List? raw; | ||||
|  | ||||
|   PostWriteMedia? thumbnail; | ||||
|  | ||||
|   PostWriteMedia(this.attachment, {this.file, this.raw}) { | ||||
|     name = attachment!.name; | ||||
|  | ||||
| @@ -66,8 +69,7 @@ class PostWriteMedia { | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   PostWriteMedia.fromBytes(this.raw, this.name, this.type, | ||||
|       {this.attachment, this.file}); | ||||
|   PostWriteMedia.fromBytes(this.raw, this.name, this.type, {this.attachment, this.file}); | ||||
|  | ||||
|   bool get isEmpty => attachment == null && file == null && raw == null; | ||||
|  | ||||
| @@ -86,7 +88,10 @@ class PostWriteMedia { | ||||
|     if (file != null) { | ||||
|       return file!; | ||||
|     } else if (raw != null) { | ||||
|       return XFile.fromData(raw!, name: name); | ||||
|       return XFile.fromData( | ||||
|         raw!, | ||||
|         name: name, | ||||
|       ); | ||||
|     } | ||||
|     return null; | ||||
|   } | ||||
| @@ -98,8 +103,7 @@ class PostWriteMedia { | ||||
|   }) { | ||||
|     if (attachment != null) { | ||||
|       final sn = context.read<SnNetworkProvider>(); | ||||
|       final ImageProvider provider = | ||||
|           UniversalImage.provider(sn.getAttachmentUrl(attachment!.rid)); | ||||
|       final ImageProvider provider = UniversalImage.provider(sn.getAttachmentUrl(attachment!.rid)); | ||||
|       if (width != null && height != null) { | ||||
|         return ResizeImage( | ||||
|           provider, | ||||
| @@ -110,8 +114,7 @@ class PostWriteMedia { | ||||
|       } | ||||
|       return provider; | ||||
|     } else if (file != null) { | ||||
|       final ImageProvider provider = | ||||
|           kIsWeb ? NetworkImage(file!.path) : FileImage(File(file!.path)); | ||||
|       final ImageProvider provider = kIsWeb ? NetworkImage(file!.path) : FileImage(File(file!.path)); | ||||
|       if (width != null && height != null) { | ||||
|         return ResizeImage( | ||||
|           provider, | ||||
| @@ -158,9 +161,10 @@ class PostWriteController extends ChangeNotifier { | ||||
|   String mode = kTitleMap.keys.first; | ||||
|  | ||||
|   String get title => titleController.text; | ||||
|  | ||||
|   String get description => descriptionController.text; | ||||
|   bool get isRelatedNull => | ||||
|       ![editingPost, repostingPost, replyingPost].any((ele) => ele != null); | ||||
|  | ||||
|   bool get isRelatedNull => ![editingPost, repostingPost, replyingPost].any((ele) => ele != null); | ||||
|  | ||||
|   bool isLoading = false, isBusy = false; | ||||
|   double? progress; | ||||
| @@ -168,6 +172,11 @@ class PostWriteController extends ChangeNotifier { | ||||
|   SnPublisher? publisher; | ||||
|   SnPost? editingPost, repostingPost, replyingPost; | ||||
|  | ||||
|   int visibility = 0; | ||||
|   List<int> visibleUsers = List.empty(); | ||||
|   List<int> invisibleUsers = List.empty(); | ||||
|   List<String> tags = List.empty(); | ||||
|   PostWriteMedia? thumbnail; | ||||
|   List<PostWriteMedia> attachments = List.empty(growable: true); | ||||
|   DateTime? publishedAt, publishedUntil; | ||||
|  | ||||
| @@ -177,53 +186,41 @@ class PostWriteController extends ChangeNotifier { | ||||
|     int? reposting, | ||||
|     int? replying, | ||||
|   }) async { | ||||
|     final sn = context.read<SnNetworkProvider>(); | ||||
|     final attach = context.read<SnAttachmentProvider>(); | ||||
|     final pt = context.read<SnPostContentProvider>(); | ||||
|  | ||||
|     isLoading = true; | ||||
|     notifyListeners(); | ||||
|  | ||||
|     try { | ||||
|       if (editing != null) { | ||||
|         final resp = await sn.client.get('/cgi/co/posts/$editing'); | ||||
|         final post = SnPost.fromJson(resp.data); | ||||
|         final alts = await attach | ||||
|             .getMultiple(post.body['attachments']?.cast<String>() ?? []); | ||||
|         final post = await pt.getPost(editing); | ||||
|         publisher = post.publisher; | ||||
|         titleController.text = post.body['title'] ?? ''; | ||||
|         descriptionController.text = post.body['description'] ?? ''; | ||||
|         contentController.text = post.body['content'] ?? ''; | ||||
|         publishedAt = post.publishedAt; | ||||
|         publishedUntil = post.publishedUntil; | ||||
|         attachments.addAll(alts.map((ele) => PostWriteMedia(ele))); | ||||
|         visibleUsers = List.from(post.visibleUsersList ?? []); | ||||
|         invisibleUsers = List.from(post.invisibleUsersList ?? []); | ||||
|         visibility = post.visibility; | ||||
|         tags = List.from(post.tags.map((ele) => ele.alias)); | ||||
|         attachments.addAll(post.preload?.attachments?.map((ele) => PostWriteMedia(ele)) ?? []); | ||||
|  | ||||
|         editingPost = post.copyWith( | ||||
|           preload: SnPostPreload( | ||||
|             attachments: alts, | ||||
|           ), | ||||
|         ); | ||||
|         if (post.preload?.thumbnail != null && (post.preload?.thumbnail?.rid.isNotEmpty ?? false)) { | ||||
|           thumbnail = PostWriteMedia(post.preload!.thumbnail); | ||||
|         } | ||||
|  | ||||
|         editingPost = post; | ||||
|       } | ||||
|  | ||||
|       if (replying != null) { | ||||
|         final resp = await sn.client.get('/cgi/co/posts/$replying'); | ||||
|         final post = SnPost.fromJson(resp.data); | ||||
|         replyingPost = post.copyWith( | ||||
|           preload: SnPostPreload( | ||||
|             attachments: await attach | ||||
|                 .getMultiple(post.body['attachments']?.cast<String>() ?? []), | ||||
|           ), | ||||
|         ); | ||||
|         final post = await pt.getPost(replying); | ||||
|         replyingPost = post; | ||||
|       } | ||||
|  | ||||
|       if (reposting != null) { | ||||
|         final resp = await sn.client.get('/cgi/co/posts/$reposting'); | ||||
|         final post = SnPost.fromJson(resp.data); | ||||
|         repostingPost = post.copyWith( | ||||
|           preload: SnPostPreload( | ||||
|             attachments: await attach | ||||
|                 .getMultiple(post.body['attachments']?.cast<String>() ?? []), | ||||
|           ), | ||||
|         ); | ||||
|         final post = await pt.getPost(reposting); | ||||
|         repostingPost = post; | ||||
|       } | ||||
|     } catch (err) { | ||||
|       if (!context.mounted) return; | ||||
| @@ -234,6 +231,44 @@ class PostWriteController extends ChangeNotifier { | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   Future<SnAttachment> _uploadAttachment(BuildContext context, PostWriteMedia media) async { | ||||
|     final attach = context.read<SnAttachmentProvider>(); | ||||
|  | ||||
|     final place = await attach.chunkedUploadInitialize( | ||||
|       (await media.length())!, | ||||
|       media.name, | ||||
|       'interactive', | ||||
|       null, | ||||
|       mimetype: media.raw != null && media.type == PostWriteMediaType.image ? 'image/png' : null, | ||||
|     ); | ||||
|  | ||||
|     final item = await attach.chunkedUploadParts( | ||||
|       media.toFile()!, | ||||
|       place.$1, | ||||
|       place.$2, | ||||
|       onProgress: (progress) { | ||||
|         progress = progress; | ||||
|         notifyListeners(); | ||||
|       }, | ||||
|     ); | ||||
|  | ||||
|     return item; | ||||
|   } | ||||
|  | ||||
|   Future<void> uploadSingleAttachment(BuildContext context, int idx) async { | ||||
|     if (isBusy) return; | ||||
|  | ||||
|     final media = idx == -1 ? thumbnail! : attachments[idx]; | ||||
|     isBusy = true; | ||||
|     notifyListeners(); | ||||
|  | ||||
|     final item = await _uploadAttachment(context, media); | ||||
|     attachments[idx] = PostWriteMedia(item); | ||||
|     isBusy = false; | ||||
|  | ||||
|     notifyListeners(); | ||||
|   } | ||||
|  | ||||
|   Future<void> post(BuildContext context) async { | ||||
|     if (isBusy || publisher == null) return; | ||||
|  | ||||
| @@ -246,6 +281,11 @@ class PostWriteController extends ChangeNotifier { | ||||
|  | ||||
|     // Uploading attachments | ||||
|     try { | ||||
|       if (thumbnail != null && thumbnail!.attachment == null) { | ||||
|         final thumb = await _uploadAttachment(context, thumbnail!); | ||||
|         thumbnail = PostWriteMedia(thumb); | ||||
|       } | ||||
|  | ||||
|       for (int i = 0; i < attachments.length; i++) { | ||||
|         final media = attachments[i]; | ||||
|         if (media.attachment != null) continue; // Already uploaded, skip | ||||
| @@ -256,6 +296,7 @@ class PostWriteController extends ChangeNotifier { | ||||
|           media.name, | ||||
|           'interactive', | ||||
|           null, | ||||
|           mimetype: media.raw != null && media.type == PostWriteMediaType.image ? 'image/png' : null, | ||||
|         ); | ||||
|  | ||||
|         final item = await attach.chunkedUploadParts( | ||||
| @@ -264,8 +305,7 @@ class PostWriteController extends ChangeNotifier { | ||||
|           place.$2, | ||||
|           onProgress: (progress) { | ||||
|             // Calculate overall progress for attachments | ||||
|             progress = ((i + progress) / attachments.length) * | ||||
|                 kAttachmentProgressWeight; | ||||
|             progress = ((i + progress) / attachments.length) * kAttachmentProgressWeight; | ||||
|             notifyListeners(); | ||||
|           }, | ||||
|         ); | ||||
| @@ -295,28 +335,24 @@ class PostWriteController extends ChangeNotifier { | ||||
|           'publisher': publisher!.id, | ||||
|           'content': contentController.text, | ||||
|           if (titleController.text.isNotEmpty) 'title': titleController.text, | ||||
|           if (descriptionController.text.isNotEmpty) | ||||
|             'description': descriptionController.text, | ||||
|           'attachments': attachments | ||||
|               .where((e) => e.attachment != null) | ||||
|               .map((e) => e.attachment!.rid) | ||||
|               .toList(), | ||||
|           if (publishedAt != null) | ||||
|             'published_at': publishedAt!.toUtc().toIso8601String(), | ||||
|           if (publishedUntil != null) | ||||
|             'published_until': publishedAt!.toUtc().toIso8601String(), | ||||
|           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(), | ||||
|           '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!.id, | ||||
|           if (repostingPost != null) 'repost_to': repostingPost!.id, | ||||
|         }, | ||||
|         onSendProgress: (count, total) { | ||||
|           progress = | ||||
|               baseProgressVal + (count / total) * (kPostingProgressWeight / 2); | ||||
|           progress = baseProgressVal + (count / total) * (kPostingProgressWeight / 2); | ||||
|           notifyListeners(); | ||||
|         }, | ||||
|         onReceiveProgress: (count, total) { | ||||
|           progress = baseProgressVal + | ||||
|               (kPostingProgressWeight / 2) + | ||||
|               (count / total) * (kPostingProgressWeight / 2); | ||||
|           progress = baseProgressVal + (kPostingProgressWeight / 2) + (count / total) * (kPostingProgressWeight / 2); | ||||
|           notifyListeners(); | ||||
|         }, | ||||
|         options: Options( | ||||
| @@ -338,12 +374,34 @@ class PostWriteController extends ChangeNotifier { | ||||
|   } | ||||
|  | ||||
|   void setAttachmentAt(int idx, PostWriteMedia item) { | ||||
|     attachments[idx] = item; | ||||
|     if (idx == -1) { | ||||
|       thumbnail = item; | ||||
|     } else { | ||||
|       attachments[idx] = item; | ||||
|     } | ||||
|     notifyListeners(); | ||||
|   } | ||||
|  | ||||
|   void removeAttachmentAt(int idx) { | ||||
|     attachments.removeAt(idx); | ||||
|     if (idx == -1) { | ||||
|       thumbnail = null; | ||||
|     } else { | ||||
|       attachments.removeAt(idx); | ||||
|     } | ||||
|     notifyListeners(); | ||||
|   } | ||||
|  | ||||
|   void setThumbnail(int? idx) { | ||||
|     if (idx == null) { | ||||
|       attachments.add(thumbnail!); | ||||
|       thumbnail = null; | ||||
|     } else { | ||||
|       if (thumbnail != null) { | ||||
|         attachments.add(thumbnail!); | ||||
|       } | ||||
|       thumbnail = attachments[idx]; | ||||
|       attachments.removeAt(idx); | ||||
|     } | ||||
|     notifyListeners(); | ||||
|   } | ||||
|  | ||||
| @@ -362,11 +420,41 @@ class PostWriteController extends ChangeNotifier { | ||||
|     notifyListeners(); | ||||
|   } | ||||
|  | ||||
|   void setTags(List<String> value) { | ||||
|     tags = value; | ||||
|     notifyListeners(); | ||||
|   } | ||||
|  | ||||
|   void setVisibility(int value) { | ||||
|     visibility = value; | ||||
|     notifyListeners(); | ||||
|   } | ||||
|  | ||||
|   void setVisibleUsers(List<int> value) { | ||||
|     visibleUsers = value; | ||||
|     notifyListeners(); | ||||
|   } | ||||
|  | ||||
|   void setInvisibleUsers(List<int> value) { | ||||
|     invisibleUsers = value; | ||||
|     notifyListeners(); | ||||
|   } | ||||
|  | ||||
|   void setProgress(double? value) { | ||||
|     progress = value; | ||||
|     notifyListeners(); | ||||
|   } | ||||
|  | ||||
|   void setIsBusy(bool value) { | ||||
|     isBusy = value; | ||||
|     notifyListeners(); | ||||
|   } | ||||
|  | ||||
|   void setMode(String value) { | ||||
|     mode = value; | ||||
|     notifyListeners(); | ||||
|   } | ||||
|  | ||||
|   void reset() { | ||||
|     publishedAt = null; | ||||
|     publishedUntil = null; | ||||
|   | ||||
							
								
								
									
										89
									
								
								lib/firebase_options.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,89 @@ | ||||
| // File generated by FlutterFire CLI. | ||||
| // ignore_for_file: type=lint | ||||
| import 'package:firebase_core/firebase_core.dart' show FirebaseOptions; | ||||
| import 'package:flutter/foundation.dart' | ||||
|     show defaultTargetPlatform, kIsWeb, TargetPlatform; | ||||
|  | ||||
| /// Default [FirebaseOptions] for use with your Firebase apps. | ||||
| /// | ||||
| /// Example: | ||||
| /// ```dart | ||||
| /// import 'firebase_options.dart'; | ||||
| /// // ... | ||||
| /// await Firebase.initializeApp( | ||||
| ///   options: DefaultFirebaseOptions.currentPlatform, | ||||
| /// ); | ||||
| /// ``` | ||||
| class DefaultFirebaseOptions { | ||||
|   static FirebaseOptions get currentPlatform { | ||||
|     if (kIsWeb) { | ||||
|       return web; | ||||
|     } | ||||
|     switch (defaultTargetPlatform) { | ||||
|       case TargetPlatform.android: | ||||
|         return android; | ||||
|       case TargetPlatform.iOS: | ||||
|         return ios; | ||||
|       case TargetPlatform.macOS: | ||||
|         return macos; | ||||
|       case TargetPlatform.windows: | ||||
|         return windows; | ||||
|       case TargetPlatform.linux: | ||||
|         throw UnsupportedError( | ||||
|           'DefaultFirebaseOptions have not been configured for linux - ' | ||||
|           'you can reconfigure this by running the FlutterFire CLI again.', | ||||
|         ); | ||||
|       default: | ||||
|         throw UnsupportedError( | ||||
|           'DefaultFirebaseOptions are not supported for this platform.', | ||||
|         ); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   static const FirebaseOptions web = FirebaseOptions( | ||||
|     apiKey: 'AIzaSyBKfIQpTouj5rXnlzkEieSlbAzepm4mgJE', | ||||
|     appId: '1:961776991058:web:b91d12f2892a5609f4188b', | ||||
|     messagingSenderId: '961776991058', | ||||
|     projectId: 'solian-0x001', | ||||
|     authDomain: 'solian-0x001.firebaseapp.com', | ||||
|     storageBucket: 'solian-0x001.firebasestorage.app', | ||||
|     measurementId: 'G-XY3HHKG0PE', | ||||
|   ); | ||||
|  | ||||
|   static const FirebaseOptions android = FirebaseOptions( | ||||
|     apiKey: 'AIzaSyDvFNudXYs29uDtcCv6pFR8h5tXBs90FYk', | ||||
|     appId: '1:961776991058:android:a8d3f7995b0b8e86f4188b', | ||||
|     messagingSenderId: '961776991058', | ||||
|     projectId: 'solian-0x001', | ||||
|     storageBucket: 'solian-0x001.firebasestorage.app', | ||||
|   ); | ||||
|  | ||||
|   static const FirebaseOptions ios = FirebaseOptions( | ||||
|     apiKey: 'AIzaSyCzQIyiYKoYHTpGXhN-IjgMML8z797WVD8', | ||||
|     appId: '1:961776991058:ios:727229d368cc47e1f4188b', | ||||
|     messagingSenderId: '961776991058', | ||||
|     projectId: 'solian-0x001', | ||||
|     storageBucket: 'solian-0x001.firebasestorage.app', | ||||
|     iosBundleId: 'dev.solsynth.solian', | ||||
|   ); | ||||
|  | ||||
|   static const FirebaseOptions macos = FirebaseOptions( | ||||
|     apiKey: 'AIzaSyCzQIyiYKoYHTpGXhN-IjgMML8z797WVD8', | ||||
|     appId: '1:961776991058:ios:727229d368cc47e1f4188b', | ||||
|     messagingSenderId: '961776991058', | ||||
|     projectId: 'solian-0x001', | ||||
|     storageBucket: 'solian-0x001.firebasestorage.app', | ||||
|     iosBundleId: 'dev.solsynth.solian', | ||||
|   ); | ||||
|  | ||||
|   static const FirebaseOptions windows = FirebaseOptions( | ||||
|     apiKey: 'AIzaSyBKfIQpTouj5rXnlzkEieSlbAzepm4mgJE', | ||||
|     appId: '1:961776991058:web:f152fd119699e13ef4188b', | ||||
|     messagingSenderId: '961776991058', | ||||
|     projectId: 'solian-0x001', | ||||
|     authDomain: 'solian-0x001.firebaseapp.com', | ||||
|     storageBucket: 'solian-0x001.firebasestorage.app', | ||||
|     measurementId: 'G-19FCN0CD9X', | ||||
|   ); | ||||
|  | ||||
| } | ||||
							
								
								
									
										208
									
								
								lib/main.dart
									
									
									
									
									
								
							
							
						
						| @@ -1,24 +1,99 @@ | ||||
| import 'dart:async'; | ||||
| import 'dart:io'; | ||||
|  | ||||
| import 'package:bitsdojo_window/bitsdojo_window.dart'; | ||||
| import 'package:croppy/croppy.dart'; | ||||
| import 'package:easy_localization/easy_localization.dart'; | ||||
| 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:go_router/go_router.dart'; | ||||
| import 'package:hive_flutter/hive_flutter.dart'; | ||||
| import 'package:provider/provider.dart'; | ||||
| import 'package:relative_time/relative_time.dart'; | ||||
| import 'package:responsive_framework/responsive_framework.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'; | ||||
| import 'package:surface/providers/config.dart'; | ||||
| import 'package:surface/providers/link_preview.dart'; | ||||
| import 'package:surface/providers/navigation.dart'; | ||||
| import 'package:surface/providers/notification.dart'; | ||||
| 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/theme.dart'; | ||||
| import 'package:surface/providers/user_directory.dart'; | ||||
| import 'package:surface/providers/userinfo.dart'; | ||||
| import 'package:surface/providers/websocket.dart'; | ||||
| import 'package:surface/providers/widget.dart'; | ||||
| import 'package:surface/router.dart'; | ||||
| 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:workmanager/workmanager.dart'; | ||||
|  | ||||
| @pragma('vm:entry-point') | ||||
| void appBackgroundDispatcher() { | ||||
|   Workmanager().executeTask((task, inputData) async { | ||||
|     print("Native called background task: $task"); | ||||
|     switch (task) { | ||||
|       case Workmanager.iOSBackgroundTask: | ||||
|         await Future.wait([widgetUpdateRandomPost()]); | ||||
|         return true; | ||||
|       case "WidgetUpdateRandomPost": | ||||
|         await widgetUpdateRandomPost(); | ||||
|         return true; | ||||
|       default: | ||||
|         return true; | ||||
|     } | ||||
|   }); | ||||
| } | ||||
|  | ||||
| void main() async { | ||||
|   WidgetsFlutterBinding.ensureInitialized(); | ||||
|   await EasyLocalization.ensureInitialized(); | ||||
|  | ||||
|   if (!kReleaseMode) { | ||||
|     debugInvertOversizedImages = true; | ||||
|   await Hive.initFlutter(); | ||||
|   Hive.registerAdapter(SnChannelImplAdapter()); | ||||
|   Hive.registerAdapter(SnRealmImplAdapter()); | ||||
|   Hive.registerAdapter(SnChannelMemberImplAdapter()); | ||||
|   Hive.registerAdapter(SnChatMessageImplAdapter()); | ||||
|  | ||||
|   await Firebase.initializeApp( | ||||
|     options: DefaultFirebaseOptions.currentPlatform, | ||||
|   ); | ||||
|  | ||||
|   GoRouter.optionURLReflectsImperativeAPIs = true; | ||||
|   usePathUrlStrategy(); | ||||
|  | ||||
|   if (!kIsWeb && (Platform.isWindows || Platform.isLinux || Platform.isMacOS)) { | ||||
|     doWhenWindowReady(() { | ||||
|       appWindow.minSize = Size(480, 640); | ||||
|       appWindow.size = Size(1280, 720); | ||||
|       appWindow.alignment = Alignment.center; | ||||
|       appWindow.show(); | ||||
|     }); | ||||
|   } | ||||
|  | ||||
|   if (!kIsWeb && (Platform.isAndroid || Platform.isIOS)) { | ||||
|     Workmanager().initialize( | ||||
|       appBackgroundDispatcher, | ||||
|       isInDebugMode: kDebugMode, | ||||
|     ); | ||||
|     Workmanager().registerPeriodicTask( | ||||
|       "widget-update-random-post", | ||||
|       "WidgetUpdateRandomPost", | ||||
|       frequency: Duration(minutes: 1), | ||||
|       constraints: Constraints(networkType: NetworkType.connected), | ||||
|       tag: "widget-update", | ||||
|     ); | ||||
|   } | ||||
|  | ||||
|   runApp(const SolianApp()); | ||||
| @@ -32,19 +107,41 @@ class SolianApp extends StatelessWidget { | ||||
|     return ResponsiveBreakpoints.builder( | ||||
|       child: EasyLocalization( | ||||
|         path: 'assets/translations', | ||||
|         supportedLocales: [Locale('en', 'US'), Locale('zh', 'CN')], | ||||
|         supportedLocales: [ | ||||
|           Locale('en', 'US'), | ||||
|           Locale('zh', 'CN'), | ||||
|           Locale('zh', 'TW'), | ||||
|           Locale('zh', 'HK'), | ||||
|         ], | ||||
|         fallbackLocale: Locale('en', 'US'), | ||||
|         useFallbackTranslations: true, | ||||
|         assetLoader: JsonAssetLoader(), | ||||
|         child: MultiProvider( | ||||
|           providers: [ | ||||
|             Provider(create: (_) => SnNetworkProvider()), | ||||
|             Provider(create: (ctx) => SnAttachmentProvider(ctx)), | ||||
|             ChangeNotifierProvider(create: (ctx) => NavigationProvider()), | ||||
|             ChangeNotifierProvider(create: (ctx) => UserProvider(ctx)), | ||||
|             // System extensions layer | ||||
|             Provider(create: (ctx) => HomeWidgetProvider(ctx)), | ||||
|  | ||||
|             // Preferences layer | ||||
|             Provider(create: (ctx) => ConfigProvider(ctx)), | ||||
|  | ||||
|             // Display layer | ||||
|             ChangeNotifierProvider(create: (_) => ThemeProvider()), | ||||
|             ChangeNotifierProvider(create: (ctx) => NavigationProvider()), | ||||
|  | ||||
|             // Data layer | ||||
|             Provider(create: (ctx) => SnNetworkProvider(ctx)), | ||||
|             Provider(create: (ctx) => UserDirectoryProvider(ctx)), | ||||
|             Provider(create: (ctx) => SnAttachmentProvider(ctx)), | ||||
|             Provider(create: (ctx) => SnPostContentProvider(ctx)), | ||||
|             Provider(create: (ctx) => SnRelationshipProvider(ctx)), | ||||
|             Provider(create: (ctx) => SnLinkPreviewProvider(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)), | ||||
|           ], | ||||
|           child: AppMainContent(), | ||||
|           child: _AppDelegate(), | ||||
|         ), | ||||
|       ), | ||||
|       breakpoints: [ | ||||
| @@ -56,13 +153,15 @@ class SolianApp extends StatelessWidget { | ||||
|   } | ||||
| } | ||||
|  | ||||
| class AppMainContent extends StatelessWidget { | ||||
|   const AppMainContent({super.key}); | ||||
| class _AppDelegate extends StatelessWidget { | ||||
|   const _AppDelegate(); | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     context.read<NavigationProvider>(); | ||||
|     context.read<UserProvider>(); | ||||
|     context.read<WebSocketProvider>(); | ||||
|     context.read<ChatChannelProvider>(); | ||||
|     context.read<NotificationProvider>(); | ||||
|  | ||||
|     final th = context.watch<ThemeProvider>(); | ||||
|  | ||||
| @@ -77,6 +176,93 @@ class AppMainContent extends StatelessWidget { | ||||
|         ...context.localizationDelegates, | ||||
|       ], | ||||
|       routerConfig: appRouter, | ||||
|       builder: (context, child) { | ||||
|         return _AppSplashScreen( | ||||
|           key: const Key('global-splash-screen'), | ||||
|           child: child!, | ||||
|         ); | ||||
|       }, | ||||
|     ); | ||||
|   } | ||||
| } | ||||
|  | ||||
| class _AppSplashScreen extends StatefulWidget { | ||||
|   final Widget child; | ||||
|  | ||||
|   const _AppSplashScreen({super.key, required this.child}); | ||||
|  | ||||
|   @override | ||||
|   State<_AppSplashScreen> createState() => _AppSplashScreenState(); | ||||
| } | ||||
|  | ||||
| class _AppSplashScreenState extends State<_AppSplashScreen> { | ||||
|   bool _isReady = false; | ||||
|  | ||||
|   Future<void> _initialize() async { | ||||
|     try { | ||||
|       final home = context.read<HomeWidgetProvider>(); | ||||
|       await home.initialize(); | ||||
|       if (!mounted) return; | ||||
|       // The Network initialization must be done after the HomeWidget initialization | ||||
|       // The Network initialization will save the server url to the HomeWidget | ||||
|       // 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(); | ||||
|       if (!mounted) return; | ||||
|       final ua = context.read<UserProvider>(); | ||||
|       await ua.initialize(); | ||||
|       if (!mounted) return; | ||||
|       final ws = context.read<WebSocketProvider>(); | ||||
|       await ws.tryConnect(); | ||||
|       if (!mounted) return; | ||||
|       final notify = context.read<NotificationProvider>(); | ||||
|       await notify.registerPushNotifications(); | ||||
|     } catch (err) { | ||||
|       if (!mounted) return; | ||||
|       await context.showErrorDialog(err); | ||||
|     } finally { | ||||
|       setState(() => _isReady = true); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   Future<void> _postInitialization() async { | ||||
|     await widgetUpdateRandomPost(); | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   void initState() { | ||||
|     super.initState(); | ||||
|     _initialize().then((_) => _postInitialization()); | ||||
|   } | ||||
|  | ||||
|   @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; | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -1,12 +0,0 @@ | ||||
| import 'dart:io'; | ||||
|  | ||||
| import 'package:dio/dio.dart'; | ||||
| import 'package:native_dio_adapter/native_dio_adapter.dart'; | ||||
|  | ||||
| Dio addClientAdapter(Dio client) { | ||||
|   if (Platform.isAndroid || Platform.isIOS || Platform.isMacOS) { | ||||
|     // Switch to native implementation if possible | ||||
|     client.httpClientAdapter = NativeAdapter(); | ||||
|   } | ||||
|   return client; | ||||
| } | ||||
| @@ -1,2 +0,0 @@ | ||||
| export 'package:surface/providers/adapters/sn_network_web.dart' | ||||
|     if (dart.library.io) 'package:surface/providers/adapters/sn_network_native.dart'; | ||||
| @@ -1,5 +0,0 @@ | ||||
| import 'package:dio/dio.dart'; | ||||
|  | ||||
| Dio addClientAdapter(Dio client) { | ||||
|   return client; | ||||
| } | ||||
							
								
								
									
										144
									
								
								lib/providers/channel.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,144 @@ | ||||
| import 'package:flutter/material.dart'; | ||||
| import 'package:hive_flutter/hive_flutter.dart'; | ||||
| import 'package:provider/provider.dart'; | ||||
| import 'package:surface/controllers/chat_message_controller.dart'; | ||||
| import 'package:surface/providers/sn_network.dart'; | ||||
| import 'package:surface/providers/user_directory.dart'; | ||||
| import 'package:surface/types/chat.dart'; | ||||
| import 'package:surface/types/realm.dart'; | ||||
|  | ||||
| class ChatChannelProvider extends ChangeNotifier { | ||||
|   static const kChatChannelBoxName = 'nex_chat_channels'; | ||||
|  | ||||
|   late final SnNetworkProvider _sn; | ||||
|   late final UserDirectoryProvider _ud; | ||||
|  | ||||
|   Box<SnChannel>? get _channelBox => Hive.box<SnChannel>(kChatChannelBoxName); | ||||
|  | ||||
|   ChatChannelProvider(BuildContext context) { | ||||
|     _sn = context.read<SnNetworkProvider>(); | ||||
|     _ud = context.read<UserDirectoryProvider>(); | ||||
|     _initializeLocalData(); | ||||
|   } | ||||
|  | ||||
|   Future<void> _initializeLocalData() async { | ||||
|     await Hive.openBox<SnChannel>(kChatChannelBoxName); | ||||
|   } | ||||
|  | ||||
|   Future<void> _saveChannelToLocal(Iterable<SnChannel> channels) async { | ||||
|     if (_channelBox == null) return; | ||||
|     await _channelBox!.putAll({ | ||||
|       for (final channel in channels) channel.key: channel, | ||||
|     }); | ||||
|   } | ||||
|  | ||||
|   Future<List<SnChannel>> _fetchChannelsFromServer({ | ||||
|     String scope = 'global', | ||||
|     bool direct = false, | ||||
|     bool doNotSave = false, | ||||
|   }) async { | ||||
|     final resp = await _sn.client.get( | ||||
|       '/cgi/im/channels/$scope/me/available', | ||||
|       queryParameters: { | ||||
|         'direct': direct, | ||||
|       }, | ||||
|     ); | ||||
|     final out = List<SnChannel>.from( | ||||
|       resp.data?.map((e) => SnChannel.fromJson(e)) ?? [], | ||||
|     ); | ||||
|     if (!doNotSave) _saveChannelToLocal(out); | ||||
|     return out; | ||||
|   } | ||||
|  | ||||
|   /// The get channel method will return the channel with the given alias. | ||||
|   /// It will use the local storage as much as possible. | ||||
|   /// The alias should include the scope, formatted as `scope:alias`. | ||||
|   Future<SnChannel> getChannel(String key) async { | ||||
|     if (_channelBox != null) { | ||||
|       final local = _channelBox!.get(key); | ||||
|       if (local != null) return local; | ||||
|     } | ||||
|  | ||||
|     var resp = await _sn.client.get('/cgi/im/channels/$key'); | ||||
|     var out = SnChannel.fromJson(resp.data); | ||||
|  | ||||
|     // Preload realm of the channel | ||||
|     if (out.realmId != null) { | ||||
|       resp = await _sn.client.get('/cgi/id/realms/${out.realmId}'); | ||||
|       out = out.copyWith(realm: SnRealm.fromJson(resp.data)); | ||||
|     } | ||||
|  | ||||
|     _saveChannelToLocal([out]); | ||||
|     return out; | ||||
|   } | ||||
|  | ||||
|   /// The fetch channel method return a stream, which will emit twice. | ||||
|   /// The first time is when the data was fetched from the local storage. | ||||
|   /// And the second time is when the data was fetched from the server. | ||||
|   /// But there is some exception that will only cause one of them to be emitted. | ||||
|   /// Like the local storage is broken or the server is down. | ||||
|   Stream<List<SnChannel>> fetchChannels() async* { | ||||
|     if (_channelBox != null) yield _channelBox!.values.toList(); | ||||
|  | ||||
|     var resp = await _sn.client.get('/cgi/id/realms/me/available'); | ||||
|     final realms = List<SnRealm>.from( | ||||
|       resp.data?.map((e) => SnRealm.fromJson(e)) ?? [], | ||||
|     ); | ||||
|     final realmMap = { | ||||
|       for (final realm in realms) realm.alias: realm, | ||||
|     }; | ||||
|  | ||||
|     final scopeToFetch = {'global', ...realms.map((e) => e.alias)}; | ||||
|  | ||||
|     final List<SnChannel> result = List.empty(growable: true); | ||||
|     final directMessages = await _fetchChannelsFromServer( | ||||
|       scope: scopeToFetch.first, | ||||
|       direct: true, | ||||
|     ); | ||||
|     result.addAll(directMessages); | ||||
|  | ||||
|     final nonBelongsChannels = await _fetchChannelsFromServer( | ||||
|       scope: scopeToFetch.first, | ||||
|       direct: false, | ||||
|     ); | ||||
|     result.addAll(nonBelongsChannels); | ||||
|  | ||||
|     for (final scope in scopeToFetch.skip(1)) { | ||||
|       final channel = await _fetchChannelsFromServer( | ||||
|         scope: scope, | ||||
|         direct: false, | ||||
|         doNotSave: true, | ||||
|       ); | ||||
|       final out = channel.map((ele) => ele.copyWith(realm: realmMap[scope])); | ||||
|       _saveChannelToLocal(out); | ||||
|       result.addAll(out); | ||||
|     } | ||||
|  | ||||
|     yield result; | ||||
|   } | ||||
|  | ||||
|   Future<List<SnChatMessage>> getLastMessages( | ||||
|     Iterable<SnChannel> channels, | ||||
|   ) async { | ||||
|     final result = List<SnChatMessage>.empty(growable: true); | ||||
|     for (final channel in channels) { | ||||
|       final channelBox = await Hive.openBox<SnChatMessage>( | ||||
|         '${ChatMessageController.kChatMessageBoxPrefix}${channel.id}', | ||||
|       ); | ||||
|       final lastMessage = channelBox.isNotEmpty | ||||
|           ? channelBox.values | ||||
|               .reduce((a, b) => a.createdAt.isAfter(b.createdAt) ? a : b) | ||||
|           : null; | ||||
|       if (lastMessage != null) result.add(lastMessage); | ||||
|       channelBox.close(); | ||||
|     } | ||||
|     await _ud.listAccount(result.map((ele) => ele.sender.accountId).toSet()); | ||||
|     return result; | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   void dispose() { | ||||
|     _channelBox?.close(); | ||||
|     super.dispose(); | ||||
|   } | ||||
| } | ||||
							
								
								
									
										459
									
								
								lib/providers/chat_call.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,459 @@ | ||||
| import 'dart:async'; | ||||
|  | ||||
| import 'package:flutter/material.dart'; | ||||
| import 'package:livekit_client/livekit_client.dart'; | ||||
| import 'package:permission_handler/permission_handler.dart'; | ||||
| import 'package:provider/provider.dart'; | ||||
| import 'package:surface/providers/sn_network.dart'; | ||||
| import 'package:surface/types/chat.dart'; | ||||
| import 'package:wakelock_plus/wakelock_plus.dart'; | ||||
|  | ||||
| class ChatCallProvider extends ChangeNotifier { | ||||
|   late final SnNetworkProvider _sn; | ||||
|  | ||||
|   ChatCallProvider(BuildContext context) { | ||||
|     _sn = context.read<SnNetworkProvider>(); | ||||
|   } | ||||
|  | ||||
|   SnChatCall? _current; | ||||
|   SnChannel? _channel; | ||||
|  | ||||
|   bool _isReady = false; | ||||
|   bool _isMounted = false; | ||||
|   bool _isInitialized = false; | ||||
|   bool _isBusy = false; | ||||
|  | ||||
|   String _lastDuration = '00:00:00'; | ||||
|   Timer? _lastDurationUpdateTimer; | ||||
|  | ||||
|   String? token; | ||||
|   String? endpoint; | ||||
|  | ||||
|   StreamSubscription? hwSubscription; | ||||
|   List<MediaDevice> _audioInputs = []; | ||||
|   List<MediaDevice> _videoInputs = []; | ||||
|  | ||||
|   bool _enableAudio = true; | ||||
|   bool _enableVideo = false; | ||||
|   LocalAudioTrack? _audioTrack; | ||||
|   LocalVideoTrack? _videoTrack; | ||||
|   MediaDevice? _videoDevice; | ||||
|   MediaDevice? _audioDevice; | ||||
|  | ||||
|   late Room _room; | ||||
|   late EventsListener<RoomEvent> _listener; | ||||
|  | ||||
|   List<ParticipantTrack> _participantTracks = []; | ||||
|   ParticipantTrack? _focusTrack; | ||||
|  | ||||
|   // Getters for private fields | ||||
|   SnChatCall? get current => _current; | ||||
|   SnChannel? get channel => _channel; | ||||
|   bool get isReady => _isReady; | ||||
|   bool get isMounted => _isMounted; | ||||
|   bool get isInitialized => _isInitialized; | ||||
|   bool get isBusy => _isBusy; | ||||
|   String get lastDuration => _lastDuration; | ||||
|   List<MediaDevice> get audioInputs => _audioInputs; | ||||
|   List<MediaDevice> get videoInputs => _videoInputs; | ||||
|   bool get enableAudio => _enableAudio; | ||||
|   bool get enableVideo => _enableVideo; | ||||
|   LocalAudioTrack? get audioTrack => _audioTrack; | ||||
|   LocalVideoTrack? get videoTrack => _videoTrack; | ||||
|   MediaDevice? get videoDevice => _videoDevice; | ||||
|   MediaDevice? get audioDevice => _audioDevice; | ||||
|   List<ParticipantTrack> get participantTracks => _participantTracks; | ||||
|   ParticipantTrack? get focusTrack => _focusTrack; | ||||
|   Room get room => _room; | ||||
|  | ||||
|   void _updateDuration() { | ||||
|     if (_current == null) { | ||||
|       _lastDuration = '00:00:00'; | ||||
|     } else { | ||||
|       Duration duration = DateTime.now().difference(_current!.createdAt); | ||||
|       String twoDigits(int n) => n.toString().padLeft(2, '0'); | ||||
|       _lastDuration = '${twoDigits(duration.inHours)}:' | ||||
|           '${twoDigits(duration.inMinutes.remainder(60))}:' | ||||
|           '${twoDigits(duration.inSeconds.remainder(60))}'; | ||||
|     } | ||||
|     notifyListeners(); | ||||
|   } | ||||
|  | ||||
|   void enableDurationUpdater() { | ||||
|     _updateDuration(); | ||||
|     _lastDurationUpdateTimer = Timer.periodic( | ||||
|       const Duration(seconds: 1), | ||||
|       (_) => _updateDuration(), | ||||
|     ); | ||||
|   } | ||||
|  | ||||
|   void disableDurationUpdater() { | ||||
|     _lastDurationUpdateTimer?.cancel(); | ||||
|     _lastDurationUpdateTimer = null; | ||||
|   } | ||||
|  | ||||
|   Future<void> checkPermissions() async { | ||||
|     if (lkPlatformIs(PlatformType.macOS) || lkPlatformIs(PlatformType.linux)) { | ||||
|       return; | ||||
|     } | ||||
|  | ||||
|     await Permission.camera.request(); | ||||
|     await Permission.microphone.request(); | ||||
|     await Permission.bluetooth.request(); | ||||
|     await Permission.bluetoothConnect.request(); | ||||
|   } | ||||
|  | ||||
|   void setCall(SnChatCall call, SnChannel related) { | ||||
|     _current = call; | ||||
|     _channel = related; | ||||
|     notifyListeners(); | ||||
|   } | ||||
|  | ||||
|   Future<(String, String)> getRoomToken() async { | ||||
|     final resp = await _sn.client.post( | ||||
|       '/cgi/im/channels/${_channel!.keyPath}/calls/ongoing/token', | ||||
|     ); | ||||
|     token = resp.data['token']; | ||||
|     endpoint = 'wss://${resp.data['endpoint']}'; | ||||
|     return (token!, endpoint!); | ||||
|   } | ||||
|  | ||||
|   void initHardware() { | ||||
|     if (_isReady) return; | ||||
|  | ||||
|     _isReady = true; | ||||
|     hwSubscription = Hardware.instance.onDeviceChange.stream.listen( | ||||
|       _revertDevices, | ||||
|     ); | ||||
|     Hardware.instance.enumerateDevices().then(_revertDevices); | ||||
|     notifyListeners(); | ||||
|   } | ||||
|  | ||||
|   void initRoom() { | ||||
|     initHardware(); | ||||
|     _room = Room( | ||||
|       roomOptions: const RoomOptions( | ||||
|         dynacast: true, | ||||
|         adaptiveStream: true, | ||||
|         defaultAudioPublishOptions: AudioPublishOptions( | ||||
|           name: 'call_voice', | ||||
|           stream: 'call_stream', | ||||
|         ), | ||||
|         defaultVideoPublishOptions: VideoPublishOptions( | ||||
|           name: 'call_video', | ||||
|           stream: 'call_stream', | ||||
|           simulcast: true, | ||||
|           backupVideoCodec: BackupVideoCodec(enabled: true), | ||||
|         ), | ||||
|         defaultScreenShareCaptureOptions: ScreenShareCaptureOptions( | ||||
|           useiOSBroadcastExtension: true, | ||||
|           params: VideoParametersPresets.screenShareH1080FPS30, | ||||
|         ), | ||||
|         defaultCameraCaptureOptions: CameraCaptureOptions( | ||||
|           maxFrameRate: 30, | ||||
|           params: VideoParametersPresets.h1080_169, | ||||
|         ), | ||||
|       ), | ||||
|     ); | ||||
|     _listener = _room.createListener(); | ||||
|     WakelockPlus.enable(); | ||||
|   } | ||||
|  | ||||
|   Future<void> joinRoom(String url, String token) async { | ||||
|     if (_isMounted) return; | ||||
|  | ||||
|     try { | ||||
|       await _room.connect( | ||||
|         url, | ||||
|         token, | ||||
|         fastConnectOptions: FastConnectOptions( | ||||
|           microphone: TrackOption(track: _audioTrack), | ||||
|           camera: TrackOption(track: _videoTrack), | ||||
|         ), | ||||
|       ); | ||||
|     } finally { | ||||
|       _isMounted = true; | ||||
|       notifyListeners(); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   void setupRoom() { | ||||
|     if (isInitialized) return; | ||||
|  | ||||
|     sortParticipants(); | ||||
|     _room.addListener(_onRoomDidUpdate); | ||||
|     WidgetsBindingCompatible.instance?.addPostFrameCallback( | ||||
|       (_) => autoPublish(), | ||||
|     ); | ||||
|  | ||||
|     if (lkPlatformIsMobile()) { | ||||
|       Hardware.instance.setSpeakerphoneOn(true); | ||||
|     } | ||||
|  | ||||
|     _isBusy = false; | ||||
|     _isInitialized = true; | ||||
|     notifyListeners(); | ||||
|   } | ||||
|  | ||||
|   void autoPublish() async { | ||||
|     try { | ||||
|       if (enableVideo) { | ||||
|         await _room.localParticipant?.setCameraEnabled(true); | ||||
|       } | ||||
|       if (enableAudio) { | ||||
|         await _room.localParticipant?.setMicrophoneEnabled(true); | ||||
|       } | ||||
|     } catch (error) { | ||||
|       rethrow; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   Future<void> setEnableAudio(bool value) async { | ||||
|     _enableAudio = value; | ||||
|     if (!_enableAudio) { | ||||
|       await _audioTrack?.stop(); | ||||
|       _audioTrack = null; | ||||
|     } else { | ||||
|       await _changeLocalAudioTrack(); | ||||
|     } | ||||
|     notifyListeners(); | ||||
|   } | ||||
|  | ||||
|   Future<void> setEnableVideo(bool value) async { | ||||
|     _enableVideo = value; | ||||
|     if (!_enableVideo) { | ||||
|       await _videoTrack?.stop(); | ||||
|       _videoTrack = null; | ||||
|     } else { | ||||
|       await _changeLocalVideoTrack(); | ||||
|     } | ||||
|     notifyListeners(); | ||||
|   } | ||||
|  | ||||
|   void setupRoomListeners({ | ||||
|     required Function(DisconnectReason?) onDisconnected, | ||||
|   }) { | ||||
|     _listener | ||||
|       ..on<RoomDisconnectedEvent>((event) async { | ||||
|         onDisconnected(event.reason); | ||||
|       }) | ||||
|       ..on<ParticipantEvent>((event) => sortParticipants()) | ||||
|       ..on<LocalTrackPublishedEvent>((_) => sortParticipants()) | ||||
|       ..on<LocalTrackUnpublishedEvent>((_) => sortParticipants()) | ||||
|       ..on<TrackSubscribedEvent>((_) => sortParticipants()) | ||||
|       ..on<TrackUnsubscribedEvent>((_) => sortParticipants()) | ||||
|       ..on<ParticipantNameUpdatedEvent>((event) { | ||||
|         sortParticipants(); | ||||
|       }); | ||||
|   } | ||||
|  | ||||
|   void sortParticipants() { | ||||
|     Map<String, ParticipantTrack> mediaTracks = {}; | ||||
|     for (var participant in _room.remoteParticipants.values) { | ||||
|       mediaTracks[participant.sid] = ParticipantTrack( | ||||
|         participant: participant, | ||||
|         videoTrack: null, | ||||
|         isScreenShare: false, | ||||
|       ); | ||||
|  | ||||
|       for (var t in participant.videoTrackPublications) { | ||||
|         mediaTracks[participant.sid]?.videoTrack = t.track; | ||||
|         mediaTracks[participant.sid]?.isScreenShare = t.isScreenShare; | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     final newTracks = List<ParticipantTrack>.empty(growable: true); | ||||
|  | ||||
|     final mediaTrackList = mediaTracks.values.toList(); | ||||
|     mediaTrackList.sort((a, b) { | ||||
|       // Loudest people first | ||||
|       if (a.participant.isSpeaking && b.participant.isSpeaking) { | ||||
|         if (a.participant.audioLevel > b.participant.audioLevel) { | ||||
|           return -1; | ||||
|         } else { | ||||
|           return 1; | ||||
|         } | ||||
|       } | ||||
|  | ||||
|       // Last spoke first | ||||
|       final aSpokeAt = a.participant.lastSpokeAt?.millisecondsSinceEpoch ?? 0; | ||||
|       final bSpokeAt = b.participant.lastSpokeAt?.millisecondsSinceEpoch ?? 0; | ||||
|  | ||||
|       if (aSpokeAt != bSpokeAt) { | ||||
|         return aSpokeAt > bSpokeAt ? -1 : 1; | ||||
|       } | ||||
|  | ||||
|       // Has video first | ||||
|       if (a.participant.hasVideo != b.participant.hasVideo) { | ||||
|         return a.participant.hasVideo ? -1 : 1; | ||||
|       } | ||||
|  | ||||
|       // First joined people first | ||||
|       return a.participant.joinedAt.millisecondsSinceEpoch - | ||||
|           b.participant.joinedAt.millisecondsSinceEpoch; | ||||
|     }); | ||||
|  | ||||
|     newTracks.addAll(mediaTrackList); | ||||
|  | ||||
|     if (_room.localParticipant != null) { | ||||
|       ParticipantTrack localTrack = ParticipantTrack( | ||||
|         participant: _room.localParticipant!, | ||||
|         videoTrack: null, | ||||
|         isScreenShare: false, | ||||
|       ); | ||||
|  | ||||
|       final localParticipantTracks = | ||||
|           _room.localParticipant?.videoTrackPublications; | ||||
|       if (localParticipantTracks != null) { | ||||
|         for (var t in localParticipantTracks) { | ||||
|           localTrack.videoTrack = t.track; | ||||
|           localTrack.isScreenShare = t.isScreenShare; | ||||
|         } | ||||
|       } | ||||
|  | ||||
|       newTracks.add(localTrack); | ||||
|     } | ||||
|  | ||||
|     _participantTracks = newTracks; | ||||
|  | ||||
|     if (focusTrack != null) { | ||||
|       final idx = participantTracks | ||||
|           .indexWhere((x) => x.participant.sid == _focusTrack!.participant.sid); | ||||
|       if (idx == -1) { | ||||
|         _focusTrack = null; | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     if (focusTrack == null) { | ||||
|       _focusTrack = participantTracks.firstOrNull; | ||||
|     } else { | ||||
|       final idx = participantTracks.indexWhere( | ||||
|         (x) => _focusTrack!.participant.sid == x.participant.sid, | ||||
|       ); | ||||
|       if (idx > -1) { | ||||
|         _focusTrack = participantTracks[idx]; | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     notifyListeners(); | ||||
|   } | ||||
|  | ||||
|   Future<void> _changeLocalAudioTrack() async { | ||||
|     if (_audioTrack != null) { | ||||
|       await _audioTrack!.stop(); | ||||
|       _audioTrack = null; | ||||
|     } | ||||
|  | ||||
|     if (_audioDevice != null) { | ||||
|       _audioTrack = await LocalAudioTrack.create( | ||||
|         AudioCaptureOptions(deviceId: _audioDevice!.deviceId), | ||||
|       ); | ||||
|       await _audioTrack!.start(); | ||||
|     } | ||||
|     notifyListeners(); | ||||
|   } | ||||
|  | ||||
|   Future<void> _changeLocalVideoTrack() async { | ||||
|     if (_videoTrack != null) { | ||||
|       await _videoTrack!.stop(); | ||||
|       _videoTrack = null; | ||||
|     } | ||||
|  | ||||
|     if (_videoDevice != null) { | ||||
|       _videoTrack = await LocalVideoTrack.createCameraTrack( | ||||
|         CameraCaptureOptions( | ||||
|           deviceId: _videoDevice!.deviceId, | ||||
|           params: VideoParametersPresets.h1080_169, | ||||
|         ), | ||||
|       ); | ||||
|       await _videoTrack!.start(); | ||||
|     } | ||||
|     notifyListeners(); | ||||
|   } | ||||
|  | ||||
|   void _revertDevices(List<MediaDevice> devices) { | ||||
|     _audioInputs = devices.where((d) => d.kind == 'audioinput').toList(); | ||||
|     _videoInputs = devices.where((d) => d.kind == 'videoinput').toList(); | ||||
|     notifyListeners(); | ||||
|   } | ||||
|  | ||||
|   void _onRoomDidUpdate() => sortParticipants(); | ||||
|  | ||||
|   Future<void> changeLocalAudioTrack() async { | ||||
|     if (audioTrack != null) { | ||||
|       await audioTrack!.stop(); | ||||
|       _audioTrack = null; | ||||
|     } | ||||
|  | ||||
|     if (audioDevice != null) { | ||||
|       _audioTrack = await LocalAudioTrack.create( | ||||
|         AudioCaptureOptions( | ||||
|           deviceId: audioDevice!.deviceId, | ||||
|         ), | ||||
|       ); | ||||
|       await audioTrack!.start(); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   Future<void> changeLocalVideoTrack() async { | ||||
|     if (videoTrack != null) { | ||||
|       await _videoTrack!.stop(); | ||||
|       _videoTrack = null; | ||||
|     } | ||||
|  | ||||
|     if (videoDevice != null) { | ||||
|       _videoTrack = await LocalVideoTrack.createCameraTrack( | ||||
|         CameraCaptureOptions( | ||||
|           deviceId: videoDevice!.deviceId, | ||||
|           params: VideoParametersPresets.h1080_169, | ||||
|         ), | ||||
|       ); | ||||
|       await videoTrack!.start(); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   void deactivateHardware() { | ||||
|     hwSubscription?.cancel(); | ||||
|   } | ||||
|  | ||||
|   void disposeRoom() { | ||||
|     _isBusy = false; | ||||
|     _isMounted = false; | ||||
|     _isInitialized = false; | ||||
|     _current = null; | ||||
|     _channel = null; | ||||
|     _room.removeListener(_onRoomDidUpdate); | ||||
|     _room.disconnect(); | ||||
|     _room.dispose(); | ||||
|     _listener.dispose(); | ||||
|     WakelockPlus.disable(); | ||||
|   } | ||||
|  | ||||
|   void disposeHardware() { | ||||
|     _isReady = false; | ||||
|     _audioTrack?.stop(); | ||||
|     _audioTrack = null; | ||||
|     _videoTrack?.stop(); | ||||
|     _videoTrack = null; | ||||
|   } | ||||
|  | ||||
|   void setVideoDevice(MediaDevice? value) { | ||||
|     _videoDevice = value; | ||||
|     notifyListeners(); | ||||
|   } | ||||
|  | ||||
|   void setAudioDevice(MediaDevice? value) { | ||||
|     _audioDevice = value; | ||||
|     notifyListeners(); | ||||
|   } | ||||
|  | ||||
|   void setFocusTrack(ParticipantTrack? value) { | ||||
|     _focusTrack = value; | ||||
|     notifyListeners(); | ||||
|   } | ||||
|  | ||||
|   void setIsBusy(bool value) { | ||||
|     _isBusy = value; | ||||
|     notifyListeners(); | ||||
|   } | ||||
| } | ||||
							
								
								
									
										43
									
								
								lib/providers/config.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,43 @@ | ||||
| import 'package:flutter/material.dart'; | ||||
| import 'package:provider/provider.dart'; | ||||
| import 'package:shared_preferences/shared_preferences.dart'; | ||||
| import 'package:surface/providers/widget.dart'; | ||||
|  | ||||
| const kAtkStoreKey = 'nex_user_atk'; | ||||
| const kRtkStoreKey = 'nex_user_rtk'; | ||||
|  | ||||
| const kNetworkServerDefault = 'https://api.sn.solsynth.dev'; | ||||
| const kNetworkServerStoreKey = 'app_server_url'; | ||||
|  | ||||
| const Map<String, FilterQuality> kImageQualityLevel = { | ||||
|   'settingsImageQualityLowest': FilterQuality.none, | ||||
|   'settingsImageQualityLow': FilterQuality.low, | ||||
|   'settingsImageQualityMedium': FilterQuality.medium, | ||||
|   'settingsImageQualityHigh': FilterQuality.high, | ||||
| }; | ||||
|  | ||||
| class ConfigProvider { | ||||
|   late final SharedPreferences prefs; | ||||
|  | ||||
|   late final HomeWidgetProvider _home; | ||||
|  | ||||
|   ConfigProvider(BuildContext context) { | ||||
|     _home = context.read<HomeWidgetProvider>(); | ||||
|   } | ||||
|  | ||||
|   Future<void> initialize() async { | ||||
|     prefs = await SharedPreferences.getInstance(); | ||||
|   } | ||||
|  | ||||
|   FilterQuality get imageQuality { | ||||
|     return kImageQualityLevel.values.elementAtOrNull(prefs.getInt('app_image_quality') ?? 3) ?? FilterQuality.high; | ||||
|   } | ||||
|  | ||||
|   String get serverUrl { | ||||
|     return prefs.getString(kNetworkServerStoreKey) ?? kNetworkServerDefault; | ||||
|   } | ||||
|   set serverUrl(String url) { | ||||
|     prefs.setString(kNetworkServerStoreKey, url); | ||||
|     _home.saveWidgetData("nex_server_url", url); | ||||
|   } | ||||
| } | ||||
							
								
								
									
										35
									
								
								lib/providers/link_preview.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,35 @@ | ||||
| import 'dart:convert'; | ||||
| import 'dart:developer'; | ||||
|  | ||||
| import 'package:flutter/material.dart'; | ||||
| import 'package:provider/provider.dart'; | ||||
| import 'package:surface/providers/sn_network.dart'; | ||||
| import 'package:surface/types/link.dart'; | ||||
|  | ||||
| class SnLinkPreviewProvider { | ||||
|   late final SnNetworkProvider _sn; | ||||
|  | ||||
|   final Map<String, SnLinkMeta> _cache = {}; | ||||
|  | ||||
|   SnLinkPreviewProvider(BuildContext context) { | ||||
|     _sn = context.read<SnNetworkProvider>(); | ||||
|   } | ||||
|  | ||||
|   Future<SnLinkMeta?> getLinkMeta(String url) async { | ||||
|     final b64 = utf8.fuse(base64Url); | ||||
|     final target = b64.encode(url); | ||||
|     if (_cache.containsKey(target)) return _cache[target]; | ||||
|  | ||||
|     log('[LinkPreview] Fetching $url ($target)'); | ||||
|  | ||||
|     try { | ||||
|       final resp = await _sn.client.get('/cgi/re/link/$target'); | ||||
|       final meta = SnLinkMeta.fromJson(resp.data); | ||||
|       _cache[url] = meta; | ||||
|       return meta; | ||||
|     } catch (err) { | ||||
|       log('[LinkPreview] Failed to fetch $url ($target)...'); | ||||
|       return null; | ||||
|     } | ||||
|   } | ||||
| } | ||||
| @@ -24,6 +24,14 @@ class NavigationProvider extends ChangeNotifier { | ||||
|  | ||||
|   int? get currentIndex => _currentIndex; | ||||
|  | ||||
|   static const List<String> kShowBottomNavScreen = [ | ||||
|     'home', | ||||
|     'explore', | ||||
|     'account', | ||||
|     'album', | ||||
|     'chat', | ||||
|   ]; | ||||
|  | ||||
|   static const List<AppNavDestination> kAllDestination = [ | ||||
|     AppNavDestination( | ||||
|       icon: Icon(Symbols.home, weight: 400, opticalSize: 20), | ||||
| @@ -35,26 +43,42 @@ class NavigationProvider extends ChangeNotifier { | ||||
|       screen: 'explore', | ||||
|       label: 'screenExplore', | ||||
|     ), | ||||
|     AppNavDestination( | ||||
|       icon: Icon(Symbols.chat, weight: 400, opticalSize: 20), | ||||
|       screen: 'chat', | ||||
|       label: 'screenChat', | ||||
|     ), | ||||
|     AppNavDestination( | ||||
|       icon: Icon(Symbols.account_circle, weight: 400, opticalSize: 20), | ||||
|       screen: 'account', | ||||
|       label: 'screenAccount', | ||||
|     ), | ||||
|     AppNavDestination( | ||||
|       icon: Icon(Symbols.album, weight: 400, opticalSize: 20), | ||||
|       icon: Icon(Symbols.group, weight: 400, opticalSize: 20), | ||||
|       screen: 'realm', | ||||
|       label: 'screenRealm', | ||||
|     ), | ||||
|     AppNavDestination( | ||||
|       icon: Icon(Symbols.photo_library, weight: 400, opticalSize: 20), | ||||
|       screen: 'album', | ||||
|       label: 'screenAlbum', | ||||
|     ), | ||||
|     AppNavDestination( | ||||
|       icon: Icon(Symbols.chat, weight: 400, opticalSize: 20), | ||||
|       screen: 'chat', | ||||
|       label: 'screenChat', | ||||
|       icon: Icon(Symbols.diversity_4, weight: 400, opticalSize: 20), | ||||
|       screen: 'friend', | ||||
|       label: 'screenFriend', | ||||
|     ), | ||||
|     AppNavDestination( | ||||
|       icon: Icon(Symbols.notifications, weight: 400, opticalSize: 20), | ||||
|       screen: 'notification', | ||||
|       label: 'screenNotification', | ||||
|     ), | ||||
|   ]; | ||||
|   static const List<String> kDefaultPinnedDestination = [ | ||||
|     'home', | ||||
|     'explore', | ||||
|     'account' | ||||
|     'chat', | ||||
|     'account', | ||||
|   ]; | ||||
|  | ||||
|   List<AppNavDestination> destinations = []; | ||||
|   | ||||
							
								
								
									
										65
									
								
								lib/providers/notification.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,65 @@ | ||||
| import 'dart:developer'; | ||||
| import 'dart:io'; | ||||
|  | ||||
| import 'package:firebase_messaging/firebase_messaging.dart'; | ||||
| import 'package:flutter/foundation.dart'; | ||||
| import 'package:flutter/material.dart'; | ||||
| import 'package:flutter_udid/flutter_udid.dart'; | ||||
| import 'package:provider/provider.dart'; | ||||
| import 'package:surface/providers/sn_network.dart'; | ||||
| import 'package:surface/providers/userinfo.dart'; | ||||
|  | ||||
| class NotificationProvider extends ChangeNotifier { | ||||
|   late final SnNetworkProvider _sn; | ||||
|   late final UserProvider _ua; | ||||
|  | ||||
|   NotificationProvider(BuildContext context) { | ||||
|     _sn = context.read<SnNetworkProvider>(); | ||||
|     _ua = context.read<UserProvider>(); | ||||
|   } | ||||
|  | ||||
|   Future<void> registerPushNotifications() async { | ||||
|     if (kIsWeb || Platform.isWindows || Platform.isLinux) return; | ||||
|     if (!_ua.isAuthorized) return; | ||||
|  | ||||
|     await FirebaseMessaging.instance.requestPermission( | ||||
|       alert: true, | ||||
|       announcement: true, | ||||
|       badge: true, | ||||
|       carPlay: false, | ||||
|       criticalAlert: false, | ||||
|       provisional: false, | ||||
|       sound: true, | ||||
|     ); | ||||
|  | ||||
|     late final String? token; | ||||
|     late final String provider; | ||||
|     var deviceUuid = await FlutterUdid.consistentUdid; | ||||
|  | ||||
|     if (deviceUuid.isEmpty) { | ||||
|       log("Unable to active push notifications, couldn't get device uuid"); | ||||
|       return; | ||||
|     } else { | ||||
|       log('Device UUID is $deviceUuid'); | ||||
|       log('Registering device push notifications...'); | ||||
|     } | ||||
|  | ||||
|     if (Platform.isIOS || Platform.isMacOS) { | ||||
|       provider = 'apns'; | ||||
|       token = await FirebaseMessaging.instance.getAPNSToken(); | ||||
|     } else { | ||||
|       provider = 'fcm'; | ||||
|       token = await FirebaseMessaging.instance.getToken(); | ||||
|     } | ||||
|     log('Device Push Token is $token'); | ||||
|  | ||||
|     await _sn.client.post( | ||||
|       '/cgi/id/notifications/subscription', | ||||
|       data: { | ||||
|         'provider': provider, | ||||
|         'device_token': token, | ||||
|         'device_id': deviceUuid, | ||||
|       }, | ||||
|     ); | ||||
|   } | ||||
| } | ||||