Compare commits
	
		
			69 Commits
		
	
	
		
			3.1.0+114
			...
			3c4a9767e1
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 3c4a9767e1 | |||
| 5df2445f3f | |||
| 56543d7b4c | |||
| 4c6fea1242 | |||
| fff43de9e3 | |||
| b31a915544 | |||
| 8956723ac5 | |||
| ccc3ac415e | |||
| 8c47a59b80 | |||
| a6d869ebf6 | |||
| f3a8699389 | |||
| d345c00e84 | |||
| a706f127b6 | |||
| 680ece0b6a | |||
| b976c6ed37 | |||
| 6ae6b132de | |||
| 95aec7c95b | |||
| edd760fbcb | |||
| ba269dbbb8 | |||
| 1aa45dd9f1 | |||
| 92685d7410 | |||
| c8e351514d | |||
| f3900825e3 | |||
| 2cc6652f75 | |||
| 4d4409de2e | |||
| e1286c797f | |||
| bec037622f | |||
| a0d8c1a9b3 | |||
| 26135d2116 | |||
| 71b67fd22d | |||
| 855072dfea | |||
| b39e2e2d64 | |||
| 84b1d6a346 | |||
| 28335dd548 | |||
| 7253e2d3ef | |||
| 4d489425fa | |||
| 890a8a44cf | |||
| 8e3583f57a | |||
| d0ff14659f | |||
| 1f7caaeaac | |||
| 9f9f42071a | |||
| 6bd6e994cb | |||
| 02e68d76ee | |||
| d04b06089c | |||
| 9be6fea2e0 | |||
| 6b1214a06f | |||
| 4597373ac9 | |||
| 047c8d93aa | |||
| 715f95ca22 | |||
| ba709012d7 | |||
| fd186f8391 | |||
| 262d36cd2d | |||
| f320855348 | |||
| ed90152462 | |||
| 6e5c5f1690 | |||
| 7c92dee097 | |||
| e4bb031138 | |||
| 97226ae96b | |||
| d8cd33e79a | |||
| 5ecd39b6a9 | |||
| 8854305e99 | |||
| 903cade296 | |||
| e48410a528 | |||
| 170ea4f2c0 | |||
| 19f0e11384 | |||
| 80bf6c3bbe | |||
| 8352ce8b5b | |||
| c06abf6e42 | |||
| 37cc0a5291 | 
							
								
								
									
										2
									
								
								.github/workflows/build.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/build.yml
									
									
									
									
										vendored
									
									
								
							| @@ -59,7 +59,7 @@ jobs: | ||||
|           sudo apt-get install -y libnotify-dev | ||||
|           sudo apt-get install -y libgstreamer1.0-dev libgstreamer-plugins-base1.0-dev | ||||
|           sudo apt-get install -y gstreamer-1.0 | ||||
|           sudo apt-get install -y libsecret-1-0 | ||||
|           sudo apt-get install -y libsecret-1-dev | ||||
|       - run: flutter pub get | ||||
|       - run: flutter build linux | ||||
|       - name: Archive production artifacts | ||||
|   | ||||
| @@ -8,6 +8,9 @@ | ||||
|  | ||||
| Hello there! Welcome to the main repository of the DysonNetwork (also known as the Solar Network). The code here is mainly about the front-end app (also known as Solian). But you can still post issues here to get help and request new features! | ||||
|  | ||||
| 如果你看得懂这行字,你可以前往我们的文档来了解更多: | ||||
| [Suki - Solar Network](https://kb.solsynth.dev/zh/solar-network) | ||||
|  | ||||
| ## Server | ||||
|  | ||||
| The backend of the Solar Network project is located at [Solsynth/DysonNetwork](https://github.com/Solsynth/DysonNetwork) | ||||
| @@ -25,8 +28,6 @@ The content below will lead you to the world of Solar Network. | ||||
|  | ||||
| ### For Normal Users | ||||
|  | ||||
| **The v3 Release is not ready, yet.** | ||||
|  | ||||
| 1. Go to the Github Releases page, and download the latest release / pre-release according to your platform. | ||||
|    - **What's the difference between stable and pre-release?** The pre-release is untested by the other users and includes the new cutting-edge features, usually the pre-release is the feature drop. At the same time, due to we're not doing the API versioning, some breaking changes may break the stable release, so use the pre-release one instead. | ||||
| 2. Create an account on the Solar Network | ||||
|   | ||||
| @@ -59,7 +59,6 @@ dependencies { | ||||
|     implementation("com.google.android.material:material:1.12.0") | ||||
|     implementation("com.github.bumptech.glide:glide:4.16.0") | ||||
|     implementation("com.squareup.okhttp3:okhttp:4.12.0") | ||||
|     implementation("com.google.firebase:firebase-messaging-ktx") | ||||
| } | ||||
|  | ||||
| flutter { | ||||
|   | ||||
| @@ -4,6 +4,7 @@ | ||||
|     <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.MODIFY_AUDIO_SETTINGS" /> | ||||
|     <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" /> | ||||
| @@ -89,6 +90,13 @@ | ||||
|             </intent-filter> | ||||
|         </activity> | ||||
|  | ||||
|         <!-- Livekit Screenshare --> | ||||
|         <service | ||||
|             android:name="de.julianassmann.flutter_background.IsolateHolderService" | ||||
|             android:enabled="true" | ||||
|             android:exported="false" | ||||
|             android:foregroundServiceType="mediaProjection" /> | ||||
|  | ||||
|         <!-- Sign in with Apple --> | ||||
|         <activity | ||||
|             android:name="com.aboutyou.dart_packages.sign_in_with_apple.SignInWithAppleCallback" | ||||
| @@ -109,14 +117,6 @@ | ||||
|             android:enabled="true" | ||||
|             android:exported="true" /> | ||||
|  | ||||
|         <service | ||||
|             android:name=".service.MessagingService" | ||||
|             android:exported="false"> | ||||
|             <intent-filter> | ||||
|                 <action android:name="com.google.firebase.MESSAGING_EVENT" /> | ||||
|             </intent-filter> | ||||
|         </service> | ||||
|  | ||||
|         <provider | ||||
|             android:name="androidx.core.content.FileProvider" | ||||
|             android:authorities="dev.solsynth.solian.provider" | ||||
| @@ -143,4 +143,4 @@ | ||||
|             <data android:mimeType="text/plain" /> | ||||
|         </intent> | ||||
|     </queries> | ||||
| </manifest> | ||||
| </manifest> | ||||
|   | ||||
| @@ -1,102 +0,0 @@ | ||||
| package dev.solsynth.solian.service | ||||
|  | ||||
| import android.app.PendingIntent | ||||
| import android.content.Intent | ||||
| import android.graphics.Bitmap | ||||
| import android.graphics.drawable.Drawable | ||||
| import android.os.Build | ||||
| import androidx.core.app.NotificationCompat | ||||
| import androidx.core.app.NotificationManagerCompat | ||||
| import androidx.core.app.RemoteInput | ||||
| import com.bumptech.glide.Glide | ||||
| import com.bumptech.glide.request.target.CustomTarget | ||||
| import com.bumptech.glide.request.transition.Transition | ||||
| import com.google.firebase.messaging.FirebaseMessagingService | ||||
| import com.google.firebase.messaging.RemoteMessage | ||||
| import dev.solsynth.solian.MainActivity | ||||
| import dev.solsynth.solian.receiver.ReplyReceiver | ||||
| import org.json.JSONObject | ||||
|  | ||||
| class MessagingService: FirebaseMessagingService() { | ||||
|     override fun onMessageReceived(remoteMessage: RemoteMessage) { | ||||
|         val type = remoteMessage.data["type"] | ||||
|         if (type == "messages.new") { | ||||
|             handleMessageNotification(remoteMessage) | ||||
|         } else { | ||||
|             // Handle other notification types | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun handleMessageNotification(remoteMessage: RemoteMessage) { | ||||
|         val data = remoteMessage.data | ||||
|         val metaString = data["meta"] ?: return | ||||
|         val meta = JSONObject(metaString) | ||||
|  | ||||
|         val pfp = meta.optString("pfp", null) | ||||
|         val roomId = meta.optString("room_id", null) | ||||
|         val messageId = meta.optString("message_id", null) | ||||
|  | ||||
|         val notificationId = System.currentTimeMillis().toInt() | ||||
|  | ||||
|         val replyLabel = "Reply" | ||||
|         val remoteInput = RemoteInput.Builder("key_text_reply") | ||||
|             .setLabel(replyLabel) | ||||
|             .build() | ||||
|  | ||||
|         val replyIntent = Intent(this, ReplyReceiver::class.java).apply { | ||||
|             putExtra("room_id", roomId) | ||||
|             putExtra("message_id", messageId) | ||||
|             putExtra("notification_id", notificationId) | ||||
|         } | ||||
|  | ||||
|         val pendingIntentFlags = if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.S) { | ||||
|             PendingIntent.FLAG_UPDATE_CURRENT or PendingIntent.FLAG_IMMUTABLE | ||||
|         } else { | ||||
|             PendingIntent.FLAG_UPDATE_CURRENT | ||||
|         } | ||||
|  | ||||
|         val replyPendingIntent = PendingIntent.getBroadcast( | ||||
|             applicationContext, | ||||
|             notificationId, | ||||
|             replyIntent, | ||||
|             pendingIntentFlags | ||||
|         ) | ||||
|  | ||||
|         val action = NotificationCompat.Action.Builder( | ||||
|             android.R.drawable.ic_menu_send, | ||||
|             replyLabel, | ||||
|             replyPendingIntent | ||||
|         ) | ||||
|             .addRemoteInput(remoteInput) | ||||
|             .build() | ||||
|  | ||||
|         val intent = Intent(this, MainActivity::class.java) | ||||
|         intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP) | ||||
|         intent.putExtra("room_id", roomId) | ||||
|         val pendingIntent = PendingIntent.getActivity(this, 0, intent, pendingIntentFlags) | ||||
|  | ||||
|         val notificationBuilder = NotificationCompat.Builder(this, "messages") | ||||
|             .setSmallIcon(android.R.drawable.ic_dialog_info) | ||||
|             .setContentTitle(remoteMessage.notification?.title) | ||||
|             .setContentText(remoteMessage.notification?.body) | ||||
|             .setPriority(NotificationCompat.PRIORITY_HIGH) | ||||
|             .setContentIntent(pendingIntent) | ||||
|             .addAction(action) | ||||
|  | ||||
|         if (pfp != null) { | ||||
|             Glide.with(applicationContext) | ||||
|                 .asBitmap() | ||||
|                 .load(pfp) | ||||
|                 .into(object : CustomTarget<Bitmap>() { | ||||
|                     override fun onResourceReady(resource: Bitmap, transition: Transition<in Bitmap>?) { | ||||
|                         notificationBuilder.setLargeIcon(resource) | ||||
|                         NotificationManagerCompat.from(applicationContext).notify(notificationId, notificationBuilder.build()) | ||||
|                     } | ||||
|  | ||||
|                     override fun onLoadCleared(placeholder: Drawable?) {} | ||||
|                 }) | ||||
|         } else { | ||||
|             NotificationManagerCompat.from(this).notify(notificationId, notificationBuilder.build()) | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @@ -146,7 +146,13 @@ | ||||
|   "edited": "Edited", | ||||
|   "addVideo": "Add video", | ||||
|   "addPhoto": "Add photo", | ||||
|   "addAudio": "Add audio", | ||||
|   "addFile": "Add file", | ||||
|   "recordAudio": "Record Audio", | ||||
|   "linkAttachment": "Link Attachment", | ||||
|   "fileIdCannotBeEmpty": "File ID cannot be empty", | ||||
|   "fileIdLinkHint": "Haven't upload to the Solar Network? Tap here to open Solar Network Drive to customize your uploads.", | ||||
|   "failedToFetchFile": "Failed to fetch file: {}", | ||||
|   "createDirectMessage": "Send new DM", | ||||
|   "gotoDirectMessage": "Go to DM", | ||||
|   "react": "React", | ||||
| @@ -352,6 +358,8 @@ | ||||
|   "postTitle": "Title", | ||||
|   "postDescription": "Description", | ||||
|   "call": "Call", | ||||
|   "callLeave": "Leave", | ||||
|   "callEnd": "End this call", | ||||
|   "done": "Done", | ||||
|   "loginResetPasswordSent": "Password reset link sent, please check your email inbox.", | ||||
|   "accountDeletion": "Delete Account", | ||||
| @@ -622,8 +630,8 @@ | ||||
|   "chatJoin": "Join the Chat", | ||||
|   "realmJoin": "Join the Realm", | ||||
|   "realmJoinSuccess": "Successfully joined the realm.", | ||||
|   "discoverRealms": "Discover Realms", | ||||
|   "discoverPublishers": "Discover Publishers", | ||||
|   "discoverRealms": "Discover realms", | ||||
|   "discoverPublishers": "Discover publishers", | ||||
|   "search": "Search", | ||||
|   "publisherMembers": "Collaborators", | ||||
|   "developerHub": "Developer Hub", | ||||
| @@ -702,5 +710,56 @@ | ||||
|   "aboutDeviceName": "Device Name", | ||||
|   "aboutDeviceIdentifier": "Device Identifier", | ||||
|   "donate": "Donate", | ||||
|   "donateDescription": "Support us to continue developing the Solar Network and keep the server up and running." | ||||
| } | ||||
|   "donateDescription": "Support us to continue developing the Solar Network and keep the server up and running.", | ||||
|   "fileId": "File ID", | ||||
|   "fileIdHint": "The file ID is the ID you get after upload the file via the Solar Network Drive.", | ||||
|   "translate": "Translate", | ||||
|   "translating": "Translating", | ||||
|   "translated": "Translated", | ||||
|   "reactionThumbUp": "Thumbs Up", | ||||
|   "reactionThumbDown": "Thumbs Down", | ||||
|   "reactionJustOkay": "Just Okay", | ||||
|   "reactionCry": "Cry", | ||||
|   "reactionConfuse": "Confused", | ||||
|   "reactionClap": "Clap", | ||||
|   "reactionLaugh": "Laugh", | ||||
|   "reactionAngry": "Angry", | ||||
|   "reactionParty": "Party", | ||||
|   "reactionPray": "Pray", | ||||
|   "reactionHeart": "Heart", | ||||
|   "selectMicrophone": "Select Microphone", | ||||
|   "selectCamera": "Select Camera", | ||||
|   "switchedTo": "Switched to {}", | ||||
|   "connecting": "Connecting", | ||||
|   "reconnecting": "Reconnecting", | ||||
|   "disconnected": "Disconnected", | ||||
|   "connected": "Connected", | ||||
|   "repliesLoadMore": "Load more replies", | ||||
|   "attachmentsRecentUploads": "Recent Uploads", | ||||
|   "attachmentsManualInput": "Manual Input", | ||||
|   "crop": "Crop", | ||||
|   "rename": "Rename", | ||||
|   "markAsSensitive": "Mark as Sensitive", | ||||
|   "fileName": "File name", | ||||
|   "sensitiveCategories.language": "Language", | ||||
|   "sensitiveCategories.sexualContent": "Sexual Content", | ||||
|   "sensitiveCategories.violence": "Violence", | ||||
|   "sensitiveCategories.profanity": "Profanity", | ||||
|   "sensitiveCategories.hateSpeech": "Hate Speech", | ||||
|   "sensitiveCategories.racism": "Racism", | ||||
|   "sensitiveCategories.adultContent": "Adult Content", | ||||
|   "sensitiveCategories.drugAbuse": "Drug Abuse", | ||||
|   "sensitiveCategories.alcoholAbuse": "Alcohol Abuse", | ||||
|   "sensitiveCategories.gambling": "Gambling", | ||||
|   "sensitiveCategories.selfHarm": "Self-harm", | ||||
|   "sensitiveCategories.childAbuse": "Child Abuse", | ||||
|   "sensitiveCategories.other": "Other", | ||||
|   "poll": "Poll", | ||||
|   "pollsRecent": "Recent Polls", | ||||
|   "pollCreateNew": "Create New", | ||||
|   "pollCreateNewHint": "Create a new poll for your post. Pick a publisher and continue.", | ||||
|   "publisher": "Publisher", | ||||
|   "publisherHint": "Enter the publisher name", | ||||
|   "publisherCannotBeEmpty": "Publisher cannot be empty", | ||||
|   "operationFailed": "Operation failed: {}" | ||||
| } | ||||
|   | ||||
| @@ -123,6 +123,10 @@ | ||||
|   "addVideo": "添加视频", | ||||
|   "addPhoto": "添加照片", | ||||
|   "addFile": "添加文件", | ||||
|   "addAttachmentById": "通过 ID 添加附件", | ||||
|   "enterFileId": "输入文件 ID", | ||||
|   "fileIdCannotBeEmpty": "文件 ID 不能为空", | ||||
|   "failedToFetchFile": "获取文件失败: {}", | ||||
|   "createDirectMessage": "创建新私人消息", | ||||
|   "gotoDirectMessage": "前往私信", | ||||
|   "react": "反应", | ||||
|   | ||||
| @@ -123,6 +123,10 @@ | ||||
|     "addVideo": "新增影片", | ||||
|     "addPhoto": "新增照片", | ||||
|     "addFile": "新增檔案", | ||||
|     "addAttachmentById": "透過 ID 新增附件", | ||||
|     "enterFileId": "輸入檔案 ID", | ||||
|     "fileIdCannotBeEmpty": "檔案 ID 不能為空", | ||||
|     "failedToFetchFile": "無法取得檔案: {}", | ||||
|     "createDirectMessage": "建立新私人訊息", | ||||
|     "gotoDirectMessage": "Go to DM", | ||||
|     "react": "反應", | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| # Uncomment this line to define a global platform for your project | ||||
| platform :ios, '13.0' | ||||
| platform :ios, '15.0' | ||||
|  | ||||
| # CocoaPods analytics sends network stats synchronously affecting flutter build latency. | ||||
| ENV['COCOAPODS_DISABLE_STATS'] = 'true' | ||||
|   | ||||
| @@ -40,33 +40,33 @@ PODS: | ||||
|   - file_picker (0.0.1): | ||||
|     - DKImagePickerController/PhotoGallery | ||||
|     - Flutter | ||||
|   - Firebase/CoreOnly (11.15.0): | ||||
|     - FirebaseCore (~> 11.15.0) | ||||
|   - Firebase/Messaging (11.15.0): | ||||
|   - Firebase/CoreOnly (12.0.0): | ||||
|     - FirebaseCore (~> 12.0.0) | ||||
|   - Firebase/Messaging (12.0.0): | ||||
|     - Firebase/CoreOnly | ||||
|     - FirebaseMessaging (~> 11.15.0) | ||||
|   - firebase_core (3.15.1): | ||||
|     - Firebase/CoreOnly (= 11.15.0) | ||||
|     - FirebaseMessaging (~> 12.0.0) | ||||
|   - firebase_core (4.0.0): | ||||
|     - Firebase/CoreOnly (= 12.0.0) | ||||
|     - Flutter | ||||
|   - firebase_messaging (15.2.9): | ||||
|     - Firebase/Messaging (= 11.15.0) | ||||
|   - firebase_messaging (16.0.0): | ||||
|     - Firebase/Messaging (= 12.0.0) | ||||
|     - firebase_core | ||||
|     - Flutter | ||||
|   - FirebaseCore (11.15.0): | ||||
|     - FirebaseCoreInternal (~> 11.15.0) | ||||
|   - FirebaseCore (12.0.0): | ||||
|     - FirebaseCoreInternal (~> 12.0.0) | ||||
|     - GoogleUtilities/Environment (~> 8.1) | ||||
|     - GoogleUtilities/Logger (~> 8.1) | ||||
|   - FirebaseCoreInternal (11.15.0): | ||||
|   - FirebaseCoreInternal (12.0.0): | ||||
|     - "GoogleUtilities/NSData+zlib (~> 8.1)" | ||||
|   - FirebaseInstallations (11.15.0): | ||||
|     - FirebaseCore (~> 11.15.0) | ||||
|   - FirebaseInstallations (12.0.0): | ||||
|     - FirebaseCore (~> 12.0.0) | ||||
|     - GoogleUtilities/Environment (~> 8.1) | ||||
|     - GoogleUtilities/UserDefaults (~> 8.1) | ||||
|     - PromisesObjC (~> 2.4) | ||||
|   - FirebaseMessaging (11.15.0): | ||||
|     - FirebaseCore (~> 11.15.0) | ||||
|     - FirebaseInstallations (~> 11.0) | ||||
|     - GoogleDataTransport (~> 10.0) | ||||
|   - FirebaseMessaging (12.0.0): | ||||
|     - FirebaseCore (~> 12.0.0) | ||||
|     - FirebaseInstallations (~> 12.0.0) | ||||
|     - GoogleDataTransport (~> 10.1) | ||||
|     - GoogleUtilities/AppDelegateSwizzler (~> 8.1) | ||||
|     - GoogleUtilities/Environment (~> 8.1) | ||||
|     - GoogleUtilities/Reachability (~> 8.1) | ||||
| @@ -93,9 +93,9 @@ PODS: | ||||
|   - flutter_udid (0.0.1): | ||||
|     - Flutter | ||||
|     - SAMKeychain | ||||
|   - flutter_webrtc (0.14.0): | ||||
|   - flutter_webrtc (1.0.0): | ||||
|     - Flutter | ||||
|     - WebRTC-SDK (= 125.6422.07) | ||||
|     - WebRTC-SDK (= 137.7151.02) | ||||
|   - gal (1.0.0): | ||||
|     - Flutter | ||||
|     - FlutterMacOS | ||||
| @@ -130,11 +130,11 @@ PODS: | ||||
|     - Flutter | ||||
|   - irondash_engine_context (0.0.1): | ||||
|     - Flutter | ||||
|   - Kingfisher (8.4.0) | ||||
|   - livekit_client (2.4.9): | ||||
|   - Kingfisher (8.5.0) | ||||
|   - livekit_client (2.5.0): | ||||
|     - Flutter | ||||
|     - flutter_webrtc | ||||
|     - WebRTC-SDK (= 125.6422.07) | ||||
|     - WebRTC-SDK (= 137.7151.02) | ||||
|   - local_auth_darwin (0.0.1): | ||||
|     - Flutter | ||||
|     - FlutterMacOS | ||||
| @@ -178,28 +178,31 @@ PODS: | ||||
|   - sqflite_darwin (0.0.4): | ||||
|     - Flutter | ||||
|     - FlutterMacOS | ||||
|   - sqlite3 (3.50.2): | ||||
|     - sqlite3/common (= 3.50.2) | ||||
|   - sqlite3/common (3.50.2) | ||||
|   - sqlite3/dbstatvtab (3.50.2): | ||||
|   - sqlite3 (3.50.3): | ||||
|     - sqlite3/common (= 3.50.3) | ||||
|   - sqlite3/common (3.50.3) | ||||
|   - sqlite3/dbstatvtab (3.50.3): | ||||
|     - sqlite3/common | ||||
|   - sqlite3/fts5 (3.50.2): | ||||
|   - sqlite3/fts5 (3.50.3): | ||||
|     - sqlite3/common | ||||
|   - sqlite3/math (3.50.2): | ||||
|   - sqlite3/math (3.50.3): | ||||
|     - sqlite3/common | ||||
|   - sqlite3/perf-threadsafe (3.50.2): | ||||
|   - sqlite3/perf-threadsafe (3.50.3): | ||||
|     - sqlite3/common | ||||
|   - sqlite3/rtree (3.50.2): | ||||
|   - sqlite3/rtree (3.50.3): | ||||
|     - sqlite3/common | ||||
|   - sqlite3/session (3.50.3): | ||||
|     - sqlite3/common | ||||
|   - sqlite3_flutter_libs (0.0.1): | ||||
|     - Flutter | ||||
|     - FlutterMacOS | ||||
|     - sqlite3 (~> 3.50.1) | ||||
|     - sqlite3 (~> 3.50.3) | ||||
|     - sqlite3/dbstatvtab | ||||
|     - sqlite3/fts5 | ||||
|     - sqlite3/math | ||||
|     - sqlite3/perf-threadsafe | ||||
|     - sqlite3/rtree | ||||
|     - sqlite3/session | ||||
|   - super_native_extensions (0.0.1): | ||||
|     - Flutter | ||||
|   - SwiftyGif (5.4.5) | ||||
| @@ -209,7 +212,7 @@ PODS: | ||||
|     - Flutter | ||||
|   - wakelock_plus (0.0.1): | ||||
|     - Flutter | ||||
|   - WebRTC-SDK (125.6422.07) | ||||
|   - WebRTC-SDK (137.7151.02) | ||||
|  | ||||
| DEPENDENCIES: | ||||
|   - Alamofire | ||||
| @@ -361,13 +364,13 @@ SPEC CHECKSUMS: | ||||
|   DKImagePickerController: 946cec48c7873164274ecc4624d19e3da4c1ef3c | ||||
|   DKPhotoGallery: b3834fecb755ee09a593d7c9e389d8b5d6deed60 | ||||
|   file_picker: a0560bc09d61de87f12d246fc47d2119e6ef37be | ||||
|   Firebase: d99ac19b909cd2c548339c2241ecd0d1599ab02e | ||||
|   firebase_core: ece862f94b2bc72ee0edbeec7ab5c7cb09fe1ab5 | ||||
|   firebase_messaging: e1a5fae495603115be1d0183bc849da748734e2b | ||||
|   FirebaseCore: efb3893e5b94f32b86e331e3bd6dadf18b66568e | ||||
|   FirebaseCoreInternal: 9afa45b1159304c963da48addb78275ef701c6b4 | ||||
|   FirebaseInstallations: 317270fec08a5d418fdbc8429282238cab3ac843 | ||||
|   FirebaseMessaging: 3b26e2cee503815e01c3701236b020aa9b576f09 | ||||
|   Firebase: 800d487043c0557d9faed71477a38d9aafb08a41 | ||||
|   firebase_core: 633e1851ffe1b9ab875f6467a4f574c79cef02e4 | ||||
|   firebase_messaging: d17feef781edc84ebefe62624fb384358ad96361 | ||||
|   FirebaseCore: 055f4ab117d5964158c833f3d5e7ec6d91648d4a | ||||
|   FirebaseCoreInternal: dedc28e569a4be85f38f3d6af1070a2e12018d55 | ||||
|   FirebaseInstallations: d4c7c958f99c8860d7fcece786314ae790e2f988 | ||||
|   FirebaseMessaging: af49f8d7c0a3d2a017d9302c80946f45a7777dde | ||||
|   Flutter: e0871f40cf51350855a761d2e70bf5af5b9b5de7 | ||||
|   flutter_inappwebview_ios: b89ba3482b96fb25e00c967aae065701b66e9b99 | ||||
|   flutter_keyboard_visibility: 4625131e43015dbbe759d9b20daaf77e0e3f6619 | ||||
| @@ -376,14 +379,14 @@ SPEC CHECKSUMS: | ||||
|   flutter_secure_storage: 1ed9476fba7e7a782b22888f956cce43e2c62f13 | ||||
|   flutter_timezone: 7c838e17ffd4645d261e87037e5bebf6d38fe544 | ||||
|   flutter_udid: f7c3884e6ec2951efe4f9de082257fc77c4d15e9 | ||||
|   flutter_webrtc: fd0d3bdef8766a0736dbbe2e5b7e85f1f3c52117 | ||||
|   flutter_webrtc: 6f7da106613d52ade777d5b4875a43f48c28b457 | ||||
|   gal: baecd024ebfd13c441269ca7404792a7152fde89 | ||||
|   GoogleDataTransport: aae35b7ea0c09004c3797d53c8c41f66f219d6a7 | ||||
|   GoogleUtilities: 00c88b9a86066ef77f0da2fab05f65d7768ed8e1 | ||||
|   image_picker_ios: 7fe1ff8e34c1790d6fff70a32484959f563a928a | ||||
|   irondash_engine_context: 8e58ca8e0212ee9d1c7dc6a42121849986c88486 | ||||
|   Kingfisher: b14cc47bbfa7a3c150dd12962ee9c86338545629 | ||||
|   livekit_client: 3f79d79233a5bd13d5b541732624ef959d7c538e | ||||
|   Kingfisher: ff0d31a1f07bdff6a1ebb3ba08b8e6e567b6500c | ||||
|   livekit_client: e3b79b99405428aac439b6b76a254cd9a11dbbfb | ||||
|   local_auth_darwin: d2e8c53ef0c4f43c646462e3415432c4dab3ae19 | ||||
|   media_kit_libs_ios_video: 5a18affdb97d1f5d466dc79988b13eff6c5e2854 | ||||
|   media_kit_video: 1746e198cb697d1ffb734b1d05ec429d1fcd1474 | ||||
| @@ -403,15 +406,15 @@ SPEC CHECKSUMS: | ||||
|   shared_preferences_foundation: 9e1978ff2562383bd5676f64ec4e9aa8fa06a6f7 | ||||
|   sign_in_with_apple: c5dcc141574c8c54d5ac99dd2163c0c72ad22418 | ||||
|   sqflite_darwin: 20b2a3a3b70e43edae938624ce550a3cbf66a3d0 | ||||
|   sqlite3: 3e82a2daae39ba3b41ae6ee84a130494585460fc | ||||
|   sqlite3_flutter_libs: e7fc8c9ea2200ff3271f08f127842131746b70e2 | ||||
|   sqlite3: 83105acd294c9137c026e2da1931c30b4588ab81 | ||||
|   sqlite3_flutter_libs: 616267f2fca40e9c6af8c5d82324e05667040b6e | ||||
|   super_native_extensions: b763c02dc3a8fd078389f410bf15149179020cb4 | ||||
|   SwiftyGif: 706c60cf65fa2bc5ee0313beece843c8eb8194d4 | ||||
|   url_launcher_ios: 694010445543906933d732453a59da0a173ae33d | ||||
|   volume_controller: 3657a1f65bedb98fa41ff7dc5793537919f31b12 | ||||
|   wakelock_plus: e29112ab3ef0b318e58cfa5c32326458be66b556 | ||||
|   WebRTC-SDK: dff00a3892bc570b6014e046297782084071657e | ||||
|   WebRTC-SDK: d20de357dcbf7c9696b124b39f3ff62125107e4b | ||||
|  | ||||
| PODFILE CHECKSUM: f6df17c2a0cbd7af89692fd3877231eaea40230f | ||||
| PODFILE CHECKSUM: c818292390b02fa379036ea099713a332bd7193f | ||||
|  | ||||
| COCOAPODS: 1.16.2 | ||||
|   | ||||
| @@ -10,6 +10,8 @@ | ||||
| 		1498D2341E8E89220040F4C2 /* GeneratedPluginRegistrant.m in Sources */ = {isa = PBXBuildFile; fileRef = 1498D2331E8E89220040F4C2 /* GeneratedPluginRegistrant.m */; }; | ||||
| 		331C808B294A63AB00263BE5 /* RunnerTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 331C807B294A618700263BE5 /* RunnerTests.swift */; }; | ||||
| 		3B3967161E833CAA004F5970 /* AppFrameworkInfo.plist in Resources */ = {isa = PBXBuildFile; fileRef = 3B3967151E833CAA004F5970 /* AppFrameworkInfo.plist */; }; | ||||
| 		73ACDFAD2E3D0E6100B63535 /* ReplayKit.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 73ACDFAC2E3D0E6100B63535 /* ReplayKit.framework */; }; | ||||
| 		73ACDFC32E3D0E6100B63535 /* SolianBroadcastExtension.appex in Embed Foundation Extensions */ = {isa = PBXBuildFile; fileRef = 73ACDFAB2E3D0E6100B63535 /* SolianBroadcastExtension.appex */; settings = {ATTRIBUTES = (RemoveHeadersOnCopy, ); }; }; | ||||
| 		73C305D82E0BE878009035B9 /* SolianShareExtension.appex in Embed Foundation Extensions */ = {isa = PBXBuildFile; fileRef = 73C305CE2E0BE878009035B9 /* SolianShareExtension.appex */; settings = {ATTRIBUTES = (RemoveHeadersOnCopy, ); }; }; | ||||
| 		73CDD6812DEC00480059D95D /* SolianNotificationService.appex in Embed Foundation Extensions */ = {isa = PBXBuildFile; fileRef = 73CDD67A2DEC00480059D95D /* SolianNotificationService.appex */; settings = {ATTRIBUTES = (RemoveHeadersOnCopy, ); }; }; | ||||
| 		73D4264B2DEB815D006C0AAE /* NotifyDelegate.swift in Sources */ = {isa = PBXBuildFile; fileRef = 73D4264A2DEB815D006C0AAE /* NotifyDelegate.swift */; }; | ||||
| @@ -32,6 +34,13 @@ | ||||
| 			remoteGlobalIDString = 97C146ED1CF9000F007C117D; | ||||
| 			remoteInfo = Runner; | ||||
| 		}; | ||||
| 		73ACDFC12E3D0E6100B63535 /* PBXContainerItemProxy */ = { | ||||
| 			isa = PBXContainerItemProxy; | ||||
| 			containerPortal = 97C146E61CF9000F007C117D /* Project object */; | ||||
| 			proxyType = 1; | ||||
| 			remoteGlobalIDString = 73ACDFAA2E3D0E6100B63535; | ||||
| 			remoteInfo = SolianBroadcastExtension; | ||||
| 		}; | ||||
| 		73C305D62E0BE878009035B9 /* PBXContainerItemProxy */ = { | ||||
| 			isa = PBXContainerItemProxy; | ||||
| 			containerPortal = 97C146E61CF9000F007C117D /* Project object */; | ||||
| @@ -55,6 +64,7 @@ | ||||
| 			dstPath = ""; | ||||
| 			dstSubfolderSpec = 13; | ||||
| 			files = ( | ||||
| 				73ACDFC32E3D0E6100B63535 /* SolianBroadcastExtension.appex in Embed Foundation Extensions */, | ||||
| 				73C305D82E0BE878009035B9 /* SolianShareExtension.appex in Embed Foundation Extensions */, | ||||
| 				73CDD6812DEC00480059D95D /* SolianNotificationService.appex in Embed Foundation Extensions */, | ||||
| 			); | ||||
| @@ -91,6 +101,9 @@ | ||||
| 		3A1C47BD29CC6AC2587D4DBE /* Pods-Runner.profile.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-Runner.profile.xcconfig"; path = "Target Support Files/Pods-Runner/Pods-Runner.profile.xcconfig"; sourceTree = "<group>"; }; | ||||
| 		3B3967151E833CAA004F5970 /* AppFrameworkInfo.plist */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.plist.xml; name = AppFrameworkInfo.plist; path = Flutter/AppFrameworkInfo.plist; sourceTree = "<group>"; }; | ||||
| 		737E920B2DB6A9FF00BE9CDB /* Runner.entitlements */ = {isa = PBXFileReference; lastKnownFileType = text.plist.entitlements; path = Runner.entitlements; sourceTree = "<group>"; }; | ||||
| 		73ACDFAB2E3D0E6100B63535 /* SolianBroadcastExtension.appex */ = {isa = PBXFileReference; explicitFileType = "wrapper.app-extension"; includeInIndex = 0; path = SolianBroadcastExtension.appex; sourceTree = BUILT_PRODUCTS_DIR; }; | ||||
| 		73ACDFAC2E3D0E6100B63535 /* ReplayKit.framework */ = {isa = PBXFileReference; lastKnownFileType = wrapper.framework; name = ReplayKit.framework; path = System/Library/Frameworks/ReplayKit.framework; sourceTree = SDKROOT; }; | ||||
| 		73ACDFB82E3D0E6100B63535 /* UIKit.framework */ = {isa = PBXFileReference; lastKnownFileType = wrapper.framework; name = UIKit.framework; path = System/Library/Frameworks/UIKit.framework; sourceTree = SDKROOT; }; | ||||
| 		73C305CE2E0BE878009035B9 /* SolianShareExtension.appex */ = {isa = PBXFileReference; explicitFileType = "wrapper.app-extension"; includeInIndex = 0; path = SolianShareExtension.appex; sourceTree = BUILT_PRODUCTS_DIR; }; | ||||
| 		73CDD67A2DEC00480059D95D /* SolianNotificationService.appex */ = {isa = PBXFileReference; explicitFileType = "wrapper.app-extension"; includeInIndex = 0; path = SolianNotificationService.appex; sourceTree = BUILT_PRODUCTS_DIR; }; | ||||
| 		73D4264A2DEB815D006C0AAE /* NotifyDelegate.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = NotifyDelegate.swift; sourceTree = "<group>"; }; | ||||
| @@ -117,6 +130,13 @@ | ||||
| /* End PBXFileReference section */ | ||||
|  | ||||
| /* Begin PBXFileSystemSynchronizedBuildFileExceptionSet section */ | ||||
| 		73ACDFCA2E3D0E6100B63535 /* Exceptions for "SolianBroadcastExtension" folder in "SolianBroadcastExtension" target */ = { | ||||
| 			isa = PBXFileSystemSynchronizedBuildFileExceptionSet; | ||||
| 			membershipExceptions = ( | ||||
| 				Info.plist, | ||||
| 			); | ||||
| 			target = 73ACDFAA2E3D0E6100B63535 /* SolianBroadcastExtension */; | ||||
| 		}; | ||||
| 		73C305DC2E0BE878009035B9 /* Exceptions for "SolianShareExtension" folder in "SolianShareExtension" target */ = { | ||||
| 			isa = PBXFileSystemSynchronizedBuildFileExceptionSet; | ||||
| 			membershipExceptions = ( | ||||
| @@ -150,6 +170,14 @@ | ||||
| 			path = Services; | ||||
| 			sourceTree = "<group>"; | ||||
| 		}; | ||||
| 		73ACDFAE2E3D0E6100B63535 /* SolianBroadcastExtension */ = { | ||||
| 			isa = PBXFileSystemSynchronizedRootGroup; | ||||
| 			exceptions = ( | ||||
| 				73ACDFCA2E3D0E6100B63535 /* Exceptions for "SolianBroadcastExtension" folder in "SolianBroadcastExtension" target */, | ||||
| 			); | ||||
| 			path = SolianBroadcastExtension; | ||||
| 			sourceTree = "<group>"; | ||||
| 		}; | ||||
| 		73C305CF2E0BE878009035B9 /* SolianShareExtension */ = { | ||||
| 			isa = PBXFileSystemSynchronizedRootGroup; | ||||
| 			exceptions = ( | ||||
| @@ -177,6 +205,14 @@ | ||||
| 			); | ||||
| 			runOnlyForDeploymentPostprocessing = 0; | ||||
| 		}; | ||||
| 		73ACDFA82E3D0E6100B63535 /* Frameworks */ = { | ||||
| 			isa = PBXFrameworksBuildPhase; | ||||
| 			buildActionMask = 2147483647; | ||||
| 			files = ( | ||||
| 				73ACDFAD2E3D0E6100B63535 /* ReplayKit.framework in Frameworks */, | ||||
| 			); | ||||
| 			runOnlyForDeploymentPostprocessing = 0; | ||||
| 		}; | ||||
| 		73C305CB2E0BE878009035B9 /* Frameworks */ = { | ||||
| 			isa = PBXFrameworksBuildPhase; | ||||
| 			buildActionMask = 2147483647; | ||||
| @@ -220,6 +256,8 @@ | ||||
| 				AA0CA8A3E15DEE023BB27438 /* Pods_NotificationService.framework */, | ||||
| 				39FE4CC6223F0D3C0E1FFD04 /* Pods_SolianNotificationService.framework */, | ||||
| 				7B40764A2C4CC0E7DC70A0D3 /* Pods_SolianShareExtension.framework */, | ||||
| 				73ACDFAC2E3D0E6100B63535 /* ReplayKit.framework */, | ||||
| 				73ACDFB82E3D0E6100B63535 /* UIKit.framework */, | ||||
| 			); | ||||
| 			name = Frameworks; | ||||
| 			sourceTree = "<group>"; | ||||
| @@ -264,6 +302,7 @@ | ||||
| 				97C146F01CF9000F007C117D /* Runner */, | ||||
| 				73CDD67B2DEC00480059D95D /* SolianNotificationService */, | ||||
| 				73C305CF2E0BE878009035B9 /* SolianShareExtension */, | ||||
| 				73ACDFAE2E3D0E6100B63535 /* SolianBroadcastExtension */, | ||||
| 				97C146EF1CF9000F007C117D /* Products */, | ||||
| 				331C8082294A63A400263BE5 /* RunnerTests */, | ||||
| 				91E124CE95BCB4DCD890160D /* Pods */, | ||||
| @@ -279,6 +318,7 @@ | ||||
| 				331C8081294A63A400263BE5 /* RunnerTests.xctest */, | ||||
| 				73CDD67A2DEC00480059D95D /* SolianNotificationService.appex */, | ||||
| 				73C305CE2E0BE878009035B9 /* SolianShareExtension.appex */, | ||||
| 				73ACDFAB2E3D0E6100B63535 /* SolianBroadcastExtension.appex */, | ||||
| 			); | ||||
| 			name = Products; | ||||
| 			sourceTree = "<group>"; | ||||
| @@ -323,6 +363,26 @@ | ||||
| 			productReference = 331C8081294A63A400263BE5 /* RunnerTests.xctest */; | ||||
| 			productType = "com.apple.product-type.bundle.unit-test"; | ||||
| 		}; | ||||
| 		73ACDFAA2E3D0E6100B63535 /* SolianBroadcastExtension */ = { | ||||
| 			isa = PBXNativeTarget; | ||||
| 			buildConfigurationList = 73ACDFCB2E3D0E6100B63535 /* Build configuration list for PBXNativeTarget "SolianBroadcastExtension" */; | ||||
| 			buildPhases = ( | ||||
| 				73ACDFA72E3D0E6100B63535 /* Sources */, | ||||
| 				73ACDFA82E3D0E6100B63535 /* Frameworks */, | ||||
| 				73ACDFA92E3D0E6100B63535 /* Resources */, | ||||
| 			); | ||||
| 			buildRules = ( | ||||
| 			); | ||||
| 			dependencies = ( | ||||
| 			); | ||||
| 			fileSystemSynchronizedGroups = ( | ||||
| 				73ACDFAE2E3D0E6100B63535 /* SolianBroadcastExtension */, | ||||
| 			); | ||||
| 			name = SolianBroadcastExtension; | ||||
| 			productName = SolianBroadcastExtension; | ||||
| 			productReference = 73ACDFAB2E3D0E6100B63535 /* SolianBroadcastExtension.appex */; | ||||
| 			productType = "com.apple.product-type.app-extension"; | ||||
| 		}; | ||||
| 		73C305CD2E0BE878009035B9 /* SolianShareExtension */ = { | ||||
| 			isa = PBXNativeTarget; | ||||
| 			buildConfigurationList = 73C305DD2E0BE878009035B9 /* Build configuration list for PBXNativeTarget "SolianShareExtension" */; | ||||
| @@ -385,6 +445,7 @@ | ||||
| 			dependencies = ( | ||||
| 				73CDD6802DEC00480059D95D /* PBXTargetDependency */, | ||||
| 				73C305D72E0BE878009035B9 /* PBXTargetDependency */, | ||||
| 				73ACDFC22E3D0E6100B63535 /* PBXTargetDependency */, | ||||
| 			); | ||||
| 			fileSystemSynchronizedGroups = ( | ||||
| 				73268D272DEB012A0076E970 /* Services */, | ||||
| @@ -409,6 +470,9 @@ | ||||
| 						CreatedOnToolsVersion = 14.0; | ||||
| 						TestTargetID = 97C146ED1CF9000F007C117D; | ||||
| 					}; | ||||
| 					73ACDFAA2E3D0E6100B63535 = { | ||||
| 						CreatedOnToolsVersion = 16.4; | ||||
| 					}; | ||||
| 					73C305CD2E0BE878009035B9 = { | ||||
| 						CreatedOnToolsVersion = 16.4; | ||||
| 					}; | ||||
| @@ -438,6 +502,7 @@ | ||||
| 				331C8080294A63A400263BE5 /* RunnerTests */, | ||||
| 				73CDD6792DEC00480059D95D /* SolianNotificationService */, | ||||
| 				73C305CD2E0BE878009035B9 /* SolianShareExtension */, | ||||
| 				73ACDFAA2E3D0E6100B63535 /* SolianBroadcastExtension */, | ||||
| 			); | ||||
| 		}; | ||||
| /* End PBXProject section */ | ||||
| @@ -450,6 +515,13 @@ | ||||
| 			); | ||||
| 			runOnlyForDeploymentPostprocessing = 0; | ||||
| 		}; | ||||
| 		73ACDFA92E3D0E6100B63535 /* Resources */ = { | ||||
| 			isa = PBXResourcesBuildPhase; | ||||
| 			buildActionMask = 2147483647; | ||||
| 			files = ( | ||||
| 			); | ||||
| 			runOnlyForDeploymentPostprocessing = 0; | ||||
| 		}; | ||||
| 		73C305CC2E0BE878009035B9 /* Resources */ = { | ||||
| 			isa = PBXResourcesBuildPhase; | ||||
| 			buildActionMask = 2147483647; | ||||
| @@ -643,6 +715,13 @@ | ||||
| 			); | ||||
| 			runOnlyForDeploymentPostprocessing = 0; | ||||
| 		}; | ||||
| 		73ACDFA72E3D0E6100B63535 /* Sources */ = { | ||||
| 			isa = PBXSourcesBuildPhase; | ||||
| 			buildActionMask = 2147483647; | ||||
| 			files = ( | ||||
| 			); | ||||
| 			runOnlyForDeploymentPostprocessing = 0; | ||||
| 		}; | ||||
| 		73C305CA2E0BE878009035B9 /* Sources */ = { | ||||
| 			isa = PBXSourcesBuildPhase; | ||||
| 			buildActionMask = 2147483647; | ||||
| @@ -675,6 +754,11 @@ | ||||
| 			target = 97C146ED1CF9000F007C117D /* Runner */; | ||||
| 			targetProxy = 331C8085294A63A400263BE5 /* PBXContainerItemProxy */; | ||||
| 		}; | ||||
| 		73ACDFC22E3D0E6100B63535 /* PBXTargetDependency */ = { | ||||
| 			isa = PBXTargetDependency; | ||||
| 			target = 73ACDFAA2E3D0E6100B63535 /* SolianBroadcastExtension */; | ||||
| 			targetProxy = 73ACDFC12E3D0E6100B63535 /* PBXContainerItemProxy */; | ||||
| 		}; | ||||
| 		73C305D72E0BE878009035B9 /* PBXTargetDependency */ = { | ||||
| 			isa = PBXTargetDependency; | ||||
| 			target = 73C305CD2E0BE878009035B9 /* SolianShareExtension */; | ||||
| @@ -773,7 +857,7 @@ | ||||
| 				INFOPLIST_FILE = Runner/Info.plist; | ||||
| 				INFOPLIST_KEY_CFBundleDisplayName = Solian; | ||||
| 				INFOPLIST_KEY_LSApplicationCategoryType = "public.app-category.social-networking"; | ||||
| 				IPHONEOS_DEPLOYMENT_TARGET = 13.0; | ||||
| 				IPHONEOS_DEPLOYMENT_TARGET = 15.0; | ||||
| 				LD_RUNPATH_SEARCH_PATHS = ( | ||||
| 					"$(inherited)", | ||||
| 					"@executable_path/Frameworks", | ||||
| @@ -836,6 +920,123 @@ | ||||
| 			}; | ||||
| 			name = Profile; | ||||
| 		}; | ||||
| 		73ACDFC42E3D0E6100B63535 /* Debug */ = { | ||||
| 			isa = XCBuildConfiguration; | ||||
| 			buildSettings = { | ||||
| 				ASSETCATALOG_COMPILER_GENERATE_SWIFT_ASSET_SYMBOL_EXTENSIONS = YES; | ||||
| 				CLANG_ANALYZER_NUMBER_OBJECT_CONVERSION = YES_AGGRESSIVE; | ||||
| 				CLANG_CXX_LANGUAGE_STANDARD = "gnu++20"; | ||||
| 				CLANG_ENABLE_OBJC_WEAK = YES; | ||||
| 				CLANG_WARN_DOCUMENTATION_COMMENTS = YES; | ||||
| 				CLANG_WARN_QUOTED_INCLUDE_IN_FRAMEWORK_HEADER = YES; | ||||
| 				CLANG_WARN_UNGUARDED_AVAILABILITY = YES_AGGRESSIVE; | ||||
| 				CODE_SIGN_ENTITLEMENTS = SolianBroadcastExtension/SolianBroadcastExtension.entitlements; | ||||
| 				CODE_SIGN_STYLE = Automatic; | ||||
| 				CURRENT_PROJECT_VERSION = 1; | ||||
| 				DEVELOPMENT_TEAM = W7HPZ53V6B; | ||||
| 				ENABLE_USER_SCRIPT_SANDBOXING = YES; | ||||
| 				GCC_C_LANGUAGE_STANDARD = gnu17; | ||||
| 				GENERATE_INFOPLIST_FILE = YES; | ||||
| 				INFOPLIST_FILE = SolianBroadcastExtension/Info.plist; | ||||
| 				INFOPLIST_KEY_CFBundleDisplayName = SolianBroadcastExtension; | ||||
| 				INFOPLIST_KEY_NSHumanReadableCopyright = ""; | ||||
| 				IPHONEOS_DEPLOYMENT_TARGET = 15.0; | ||||
| 				LD_RUNPATH_SEARCH_PATHS = ( | ||||
| 					"$(inherited)", | ||||
| 					"@executable_path/Frameworks", | ||||
| 					"@executable_path/../../Frameworks", | ||||
| 				); | ||||
| 				LOCALIZATION_PREFERS_STRING_CATALOGS = YES; | ||||
| 				MARKETING_VERSION = 1.0; | ||||
| 				MTL_ENABLE_DEBUG_INFO = INCLUDE_SOURCE; | ||||
| 				MTL_FAST_MATH = YES; | ||||
| 				PRODUCT_BUNDLE_IDENTIFIER = dev.solsynth.solian.SolianBroadcastExtension; | ||||
| 				PRODUCT_NAME = "$(TARGET_NAME)"; | ||||
| 				SKIP_INSTALL = YES; | ||||
| 				SWIFT_ACTIVE_COMPILATION_CONDITIONS = "DEBUG $(inherited)"; | ||||
| 				SWIFT_EMIT_LOC_STRINGS = YES; | ||||
| 				SWIFT_OPTIMIZATION_LEVEL = "-Onone"; | ||||
| 				SWIFT_VERSION = 5.0; | ||||
| 				TARGETED_DEVICE_FAMILY = "1,2"; | ||||
| 			}; | ||||
| 			name = Debug; | ||||
| 		}; | ||||
| 		73ACDFC52E3D0E6100B63535 /* Release */ = { | ||||
| 			isa = XCBuildConfiguration; | ||||
| 			buildSettings = { | ||||
| 				ASSETCATALOG_COMPILER_GENERATE_SWIFT_ASSET_SYMBOL_EXTENSIONS = YES; | ||||
| 				CLANG_ANALYZER_NUMBER_OBJECT_CONVERSION = YES_AGGRESSIVE; | ||||
| 				CLANG_CXX_LANGUAGE_STANDARD = "gnu++20"; | ||||
| 				CLANG_ENABLE_OBJC_WEAK = YES; | ||||
| 				CLANG_WARN_DOCUMENTATION_COMMENTS = YES; | ||||
| 				CLANG_WARN_QUOTED_INCLUDE_IN_FRAMEWORK_HEADER = YES; | ||||
| 				CLANG_WARN_UNGUARDED_AVAILABILITY = YES_AGGRESSIVE; | ||||
| 				CODE_SIGN_ENTITLEMENTS = SolianBroadcastExtension/SolianBroadcastExtension.entitlements; | ||||
| 				CODE_SIGN_STYLE = Automatic; | ||||
| 				CURRENT_PROJECT_VERSION = 1; | ||||
| 				DEVELOPMENT_TEAM = W7HPZ53V6B; | ||||
| 				ENABLE_USER_SCRIPT_SANDBOXING = YES; | ||||
| 				GCC_C_LANGUAGE_STANDARD = gnu17; | ||||
| 				GENERATE_INFOPLIST_FILE = YES; | ||||
| 				INFOPLIST_FILE = SolianBroadcastExtension/Info.plist; | ||||
| 				INFOPLIST_KEY_CFBundleDisplayName = SolianBroadcastExtension; | ||||
| 				INFOPLIST_KEY_NSHumanReadableCopyright = ""; | ||||
| 				IPHONEOS_DEPLOYMENT_TARGET = 15.0; | ||||
| 				LD_RUNPATH_SEARCH_PATHS = ( | ||||
| 					"$(inherited)", | ||||
| 					"@executable_path/Frameworks", | ||||
| 					"@executable_path/../../Frameworks", | ||||
| 				); | ||||
| 				LOCALIZATION_PREFERS_STRING_CATALOGS = YES; | ||||
| 				MARKETING_VERSION = 1.0; | ||||
| 				MTL_FAST_MATH = YES; | ||||
| 				PRODUCT_BUNDLE_IDENTIFIER = dev.solsynth.solian.SolianBroadcastExtension; | ||||
| 				PRODUCT_NAME = "$(TARGET_NAME)"; | ||||
| 				SKIP_INSTALL = YES; | ||||
| 				SWIFT_EMIT_LOC_STRINGS = YES; | ||||
| 				SWIFT_VERSION = 5.0; | ||||
| 				TARGETED_DEVICE_FAMILY = "1,2"; | ||||
| 			}; | ||||
| 			name = Release; | ||||
| 		}; | ||||
| 		73ACDFC62E3D0E6100B63535 /* Profile */ = { | ||||
| 			isa = XCBuildConfiguration; | ||||
| 			buildSettings = { | ||||
| 				ASSETCATALOG_COMPILER_GENERATE_SWIFT_ASSET_SYMBOL_EXTENSIONS = YES; | ||||
| 				CLANG_ANALYZER_NUMBER_OBJECT_CONVERSION = YES_AGGRESSIVE; | ||||
| 				CLANG_CXX_LANGUAGE_STANDARD = "gnu++20"; | ||||
| 				CLANG_ENABLE_OBJC_WEAK = YES; | ||||
| 				CLANG_WARN_DOCUMENTATION_COMMENTS = YES; | ||||
| 				CLANG_WARN_QUOTED_INCLUDE_IN_FRAMEWORK_HEADER = YES; | ||||
| 				CLANG_WARN_UNGUARDED_AVAILABILITY = YES_AGGRESSIVE; | ||||
| 				CODE_SIGN_ENTITLEMENTS = SolianBroadcastExtension/SolianBroadcastExtension.entitlements; | ||||
| 				CODE_SIGN_STYLE = Automatic; | ||||
| 				CURRENT_PROJECT_VERSION = 1; | ||||
| 				DEVELOPMENT_TEAM = W7HPZ53V6B; | ||||
| 				ENABLE_USER_SCRIPT_SANDBOXING = YES; | ||||
| 				GCC_C_LANGUAGE_STANDARD = gnu17; | ||||
| 				GENERATE_INFOPLIST_FILE = YES; | ||||
| 				INFOPLIST_FILE = SolianBroadcastExtension/Info.plist; | ||||
| 				INFOPLIST_KEY_CFBundleDisplayName = SolianBroadcastExtension; | ||||
| 				INFOPLIST_KEY_NSHumanReadableCopyright = ""; | ||||
| 				IPHONEOS_DEPLOYMENT_TARGET = 15.0; | ||||
| 				LD_RUNPATH_SEARCH_PATHS = ( | ||||
| 					"$(inherited)", | ||||
| 					"@executable_path/Frameworks", | ||||
| 					"@executable_path/../../Frameworks", | ||||
| 				); | ||||
| 				LOCALIZATION_PREFERS_STRING_CATALOGS = YES; | ||||
| 				MARKETING_VERSION = 1.0; | ||||
| 				MTL_FAST_MATH = YES; | ||||
| 				PRODUCT_BUNDLE_IDENTIFIER = dev.solsynth.solian.SolianBroadcastExtension; | ||||
| 				PRODUCT_NAME = "$(TARGET_NAME)"; | ||||
| 				SKIP_INSTALL = YES; | ||||
| 				SWIFT_EMIT_LOC_STRINGS = YES; | ||||
| 				SWIFT_VERSION = 5.0; | ||||
| 				TARGETED_DEVICE_FAMILY = "1,2"; | ||||
| 			}; | ||||
| 			name = Profile; | ||||
| 		}; | ||||
| 		73C305D92E0BE878009035B9 /* Debug */ = { | ||||
| 			isa = XCBuildConfiguration; | ||||
| 			baseConfigurationReference = 17FAB080A9C53193ABD9C15B /* Pods-SolianShareExtension.debug.xcconfig */; | ||||
| @@ -1204,7 +1405,7 @@ | ||||
| 				INFOPLIST_FILE = Runner/Info.plist; | ||||
| 				INFOPLIST_KEY_CFBundleDisplayName = Solian; | ||||
| 				INFOPLIST_KEY_LSApplicationCategoryType = "public.app-category.social-networking"; | ||||
| 				IPHONEOS_DEPLOYMENT_TARGET = 13.0; | ||||
| 				IPHONEOS_DEPLOYMENT_TARGET = 15.0; | ||||
| 				LD_RUNPATH_SEARCH_PATHS = ( | ||||
| 					"$(inherited)", | ||||
| 					"@executable_path/Frameworks", | ||||
| @@ -1232,7 +1433,7 @@ | ||||
| 				INFOPLIST_FILE = Runner/Info.plist; | ||||
| 				INFOPLIST_KEY_CFBundleDisplayName = Solian; | ||||
| 				INFOPLIST_KEY_LSApplicationCategoryType = "public.app-category.social-networking"; | ||||
| 				IPHONEOS_DEPLOYMENT_TARGET = 13.0; | ||||
| 				IPHONEOS_DEPLOYMENT_TARGET = 15.0; | ||||
| 				LD_RUNPATH_SEARCH_PATHS = ( | ||||
| 					"$(inherited)", | ||||
| 					"@executable_path/Frameworks", | ||||
| @@ -1258,6 +1459,16 @@ | ||||
| 			defaultConfigurationIsVisible = 0; | ||||
| 			defaultConfigurationName = Release; | ||||
| 		}; | ||||
| 		73ACDFCB2E3D0E6100B63535 /* Build configuration list for PBXNativeTarget "SolianBroadcastExtension" */ = { | ||||
| 			isa = XCConfigurationList; | ||||
| 			buildConfigurations = ( | ||||
| 				73ACDFC42E3D0E6100B63535 /* Debug */, | ||||
| 				73ACDFC52E3D0E6100B63535 /* Release */, | ||||
| 				73ACDFC62E3D0E6100B63535 /* Profile */, | ||||
| 			); | ||||
| 			defaultConfigurationIsVisible = 0; | ||||
| 			defaultConfigurationName = Release; | ||||
| 		}; | ||||
| 		73C305DD2E0BE878009035B9 /* Build configuration list for PBXNativeTarget "SolianShareExtension" */ = { | ||||
| 			isa = XCConfigurationList; | ||||
| 			buildConfigurations = ( | ||||
|   | ||||
							
								
								
									
										37
									
								
								ios/SolianBroadcastExtension/Atomic.swift
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										37
									
								
								ios/SolianBroadcastExtension/Atomic.swift
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,37 @@ | ||||
| // | ||||
| //  Atomic.swift | ||||
| //  Broadcast Extension | ||||
| // | ||||
| //  Created by Maksym Shcheglov. | ||||
| //  https://www.onswiftwings.com/posts/atomic-property-wrapper/ | ||||
| // | ||||
|  | ||||
| import Foundation | ||||
|  | ||||
| @propertyWrapper | ||||
| struct Atomic<Value> { | ||||
|  | ||||
|     private var value: Value | ||||
|     private let lock = NSLock() | ||||
|  | ||||
|     init(wrappedValue value: Value) { | ||||
|         self.value = value | ||||
|     } | ||||
|  | ||||
|     var wrappedValue: Value { | ||||
|         get { load() } | ||||
|         set { store(newValue: newValue) } | ||||
|     } | ||||
|  | ||||
|     func load() -> Value { | ||||
|         lock.lock() | ||||
|         defer { lock.unlock() } | ||||
|         return value | ||||
|     } | ||||
|  | ||||
|     mutating func store(newValue: Value) { | ||||
|         lock.lock() | ||||
|         defer { lock.unlock() } | ||||
|         value = newValue | ||||
|     } | ||||
| } | ||||
							
								
								
									
										29
									
								
								ios/SolianBroadcastExtension/DarwinNotification.swift
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										29
									
								
								ios/SolianBroadcastExtension/DarwinNotification.swift
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,29 @@ | ||||
| // | ||||
| //  DarwinNotificationCenter.swift | ||||
| //  Broadcast Extension | ||||
| // | ||||
| //  Created by Alex-Dan Bumbu on 23/03/2021. | ||||
| //  Copyright © 2021 8x8, Inc. All rights reserved. | ||||
| // | ||||
|  | ||||
| import Foundation | ||||
|  | ||||
| enum DarwinNotification: String { | ||||
|     case broadcastStarted = "iOS_BroadcastStarted" | ||||
|     case broadcastStopped = "iOS_BroadcastStopped" | ||||
| } | ||||
|  | ||||
| class DarwinNotificationCenter { | ||||
|      | ||||
|     static let shared = DarwinNotificationCenter() | ||||
|      | ||||
|     private let notificationCenter: CFNotificationCenter | ||||
|      | ||||
|     init() { | ||||
|         notificationCenter = CFNotificationCenterGetDarwinNotifyCenter() | ||||
|     } | ||||
|      | ||||
|     func postNotification(_ name: DarwinNotification) { | ||||
|         CFNotificationCenterPostNotification(notificationCenter, CFNotificationName(rawValue: name.rawValue as CFString), nil, nil, true) | ||||
|     } | ||||
| } | ||||
							
								
								
									
										15
									
								
								ios/SolianBroadcastExtension/Info.plist
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								ios/SolianBroadcastExtension/Info.plist
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,15 @@ | ||||
| <?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.broadcast-services-upload</string> | ||||
| 		<key>NSExtensionPrincipalClass</key> | ||||
| 		<string>$(PRODUCT_MODULE_NAME).SampleHandler</string> | ||||
| 		<key>RPBroadcastProcessMode</key> | ||||
| 		<string>RPBroadcastProcessModeSampleBuffer</string> | ||||
| 	</dict> | ||||
| </dict> | ||||
| </plist> | ||||
							
								
								
									
										103
									
								
								ios/SolianBroadcastExtension/SampleHandler.swift
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										103
									
								
								ios/SolianBroadcastExtension/SampleHandler.swift
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,103 @@ | ||||
| // | ||||
| //  SampleHandler.swift | ||||
| //  Broadcast Extension | ||||
| // | ||||
| //  Created by Alex-Dan Bumbu on 04.06.2021. | ||||
| // | ||||
|  | ||||
| import ReplayKit | ||||
| import OSLog | ||||
|  | ||||
| let broadcastLogger = OSLog(subsystem: "dev.solsynth.solian", category: "Broadcast") | ||||
| private enum Constants { | ||||
|     // the App Group ID value that the app and the broadcast extension targets are setup with. It differs for each app. | ||||
|     static let appGroupIdentifier = "group.solsynth.solian" | ||||
| } | ||||
|  | ||||
| class SampleHandler: RPBroadcastSampleHandler { | ||||
|  | ||||
|     private var clientConnection: SocketConnection? | ||||
|     private var uploader: SampleUploader? | ||||
|  | ||||
|     private var frameCount: Int = 0 | ||||
|  | ||||
|     var socketFilePath: String { | ||||
|       let sharedContainer = FileManager.default.containerURL(forSecurityApplicationGroupIdentifier: Constants.appGroupIdentifier) | ||||
|         return sharedContainer?.appendingPathComponent("rtc_SSFD").path ?? "" | ||||
|     } | ||||
|  | ||||
|     override init() { | ||||
|       super.init() | ||||
|         if let connection = SocketConnection(filePath: socketFilePath) { | ||||
|           clientConnection = connection | ||||
|           setupConnection() | ||||
|  | ||||
|           uploader = SampleUploader(connection: connection) | ||||
|         } | ||||
|         os_log(.debug, log: broadcastLogger, "%{public}s", socketFilePath) | ||||
|     } | ||||
|  | ||||
|     override func broadcastStarted(withSetupInfo setupInfo: [String: NSObject]?) { | ||||
|         // User has requested to start the broadcast. Setup info from the UI extension can be supplied but optional. | ||||
|         frameCount = 0 | ||||
|  | ||||
|         DarwinNotificationCenter.shared.postNotification(.broadcastStarted) | ||||
|         openConnection() | ||||
|     } | ||||
|  | ||||
|     override func broadcastPaused() { | ||||
|         // User has requested to pause the broadcast. Samples will stop being delivered. | ||||
|     } | ||||
|  | ||||
|     override func broadcastResumed() { | ||||
|         // User has requested to resume the broadcast. Samples delivery will resume. | ||||
|     } | ||||
|  | ||||
|     override func broadcastFinished() { | ||||
|         // User has requested to finish the broadcast. | ||||
|         DarwinNotificationCenter.shared.postNotification(.broadcastStopped) | ||||
|         clientConnection?.close() | ||||
|     } | ||||
|  | ||||
|     override func processSampleBuffer(_ sampleBuffer: CMSampleBuffer, with sampleBufferType: RPSampleBufferType) { | ||||
|         switch sampleBufferType { | ||||
|         case RPSampleBufferType.video: | ||||
|             uploader?.send(sample: sampleBuffer) | ||||
|         default: | ||||
|             break | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| private extension SampleHandler { | ||||
|  | ||||
|     func setupConnection() { | ||||
|         clientConnection?.didClose = { [weak self] error in | ||||
|             os_log(.debug, log: broadcastLogger, "client connection did close \(String(describing: error))") | ||||
|  | ||||
|             if let error = error { | ||||
|                 self?.finishBroadcastWithError(error) | ||||
|             } else { | ||||
|                 // the displayed failure message is more user friendly when using NSError instead of Error | ||||
|                 let JMScreenSharingStopped = 10001 | ||||
|                 let customError = NSError(domain: RPRecordingErrorDomain, code: JMScreenSharingStopped, userInfo: [NSLocalizedDescriptionKey: "Screen sharing stopped"]) | ||||
|                 self?.finishBroadcastWithError(customError) | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     func openConnection() { | ||||
|         let queue = DispatchQueue(label: "broadcast.connectTimer") | ||||
|         let timer = DispatchSource.makeTimerSource(queue: queue) | ||||
|         timer.schedule(deadline: .now(), repeating: .milliseconds(100), leeway: .milliseconds(500)) | ||||
|         timer.setEventHandler { [weak self] in | ||||
|             guard self?.clientConnection?.open() == true else { | ||||
|                 return | ||||
|             } | ||||
|  | ||||
|             timer.cancel() | ||||
|         } | ||||
|  | ||||
|         timer.resume() | ||||
|     } | ||||
| } | ||||
							
								
								
									
										147
									
								
								ios/SolianBroadcastExtension/SampleUploader.swift
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										147
									
								
								ios/SolianBroadcastExtension/SampleUploader.swift
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,147 @@ | ||||
| // | ||||
| //  SampleUploader.swift | ||||
| //  Broadcast Extension | ||||
| // | ||||
| //  Created by Alex-Dan Bumbu on 22/03/2021. | ||||
| //  Copyright © 2021 8x8, Inc. All rights reserved. | ||||
| // | ||||
|  | ||||
| import Foundation | ||||
| import ReplayKit | ||||
| import OSLog | ||||
|  | ||||
| private enum Constants { | ||||
|     static let bufferMaxLength = 10240 | ||||
| } | ||||
|  | ||||
| class SampleUploader { | ||||
|      | ||||
|     private static var imageContext = CIContext(options: nil) | ||||
|      | ||||
|     @Atomic private var isReady = false | ||||
|     private var connection: SocketConnection | ||||
|    | ||||
|     private var dataToSend: Data? | ||||
|     private var byteIndex = 0 | ||||
|    | ||||
|     private let serialQueue: DispatchQueue | ||||
|      | ||||
|     init(connection: SocketConnection) { | ||||
|         self.connection = connection | ||||
|         self.serialQueue = DispatchQueue(label: "org.jitsi.meet.broadcast.sampleUploader") | ||||
|        | ||||
|         setupConnection() | ||||
|     } | ||||
|    | ||||
|     @discardableResult func send(sample buffer: CMSampleBuffer) -> Bool { | ||||
|         guard isReady else { | ||||
|             return false | ||||
|         } | ||||
|          | ||||
|         isReady = false | ||||
|  | ||||
|         dataToSend = prepare(sample: buffer) | ||||
|         byteIndex = 0 | ||||
|  | ||||
|         serialQueue.async { [weak self] in | ||||
|             self?.sendDataChunk() | ||||
|         } | ||||
|          | ||||
|         return true | ||||
|     } | ||||
| } | ||||
|  | ||||
| private extension SampleUploader { | ||||
|      | ||||
|     func setupConnection() { | ||||
|         connection.didOpen = { [weak self] in | ||||
|             self?.isReady = true | ||||
|         } | ||||
|         connection.streamHasSpaceAvailable = { [weak self] in | ||||
|             self?.serialQueue.async { | ||||
|                 if let success = self?.sendDataChunk() { | ||||
|                     self?.isReady = !success | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|      | ||||
|     @discardableResult func sendDataChunk() -> Bool { | ||||
|         guard let dataToSend = dataToSend else { | ||||
|             return false | ||||
|         } | ||||
|        | ||||
|         var bytesLeft = dataToSend.count - byteIndex | ||||
|         var length = bytesLeft > Constants.bufferMaxLength ? Constants.bufferMaxLength : bytesLeft | ||||
|  | ||||
|         length = dataToSend[byteIndex..<(byteIndex + length)].withUnsafeBytes { | ||||
|             guard let ptr = $0.bindMemory(to: UInt8.self).baseAddress else { | ||||
|                 return 0 | ||||
|             } | ||||
|  | ||||
|             return connection.writeToStream(buffer: ptr, maxLength: length) | ||||
|         } | ||||
|  | ||||
|         if length > 0 { | ||||
|             byteIndex += length | ||||
|             bytesLeft -= length | ||||
|  | ||||
|             if bytesLeft == 0 { | ||||
|                 self.dataToSend = nil | ||||
|                 byteIndex = 0 | ||||
|             } | ||||
|         } else { | ||||
|             os_log(.debug, log: broadcastLogger, "writeBufferToStream failure") | ||||
|         } | ||||
|        | ||||
|         return true | ||||
|     } | ||||
|      | ||||
|     func prepare(sample buffer: CMSampleBuffer) -> Data? { | ||||
|         guard let imageBuffer = CMSampleBufferGetImageBuffer(buffer) else { | ||||
|             os_log(.debug, log: broadcastLogger, "image buffer not available") | ||||
|             return nil | ||||
|         } | ||||
|          | ||||
|         CVPixelBufferLockBaseAddress(imageBuffer, .readOnly) | ||||
|          | ||||
|         let scaleFactor = 1.0 | ||||
|         let width = CVPixelBufferGetWidth(imageBuffer)/Int(scaleFactor) | ||||
|         let height = CVPixelBufferGetHeight(imageBuffer)/Int(scaleFactor) | ||||
|         let orientation = CMGetAttachment(buffer, key: RPVideoSampleOrientationKey as CFString, attachmentModeOut: nil)?.uintValue ?? 0 | ||||
|                                      | ||||
|         let scaleTransform = CGAffineTransform(scaleX: CGFloat(1.0/scaleFactor), y: CGFloat(1.0/scaleFactor)) | ||||
|         let bufferData = self.jpegData(from: imageBuffer, scale: scaleTransform) | ||||
|          | ||||
|         CVPixelBufferUnlockBaseAddress(imageBuffer, .readOnly) | ||||
|          | ||||
|         guard let messageData = bufferData else { | ||||
|             os_log(.debug, log: broadcastLogger, "corrupted image buffer") | ||||
|             return nil | ||||
|         } | ||||
|                | ||||
|         let httpResponse = CFHTTPMessageCreateResponse(nil, 200, nil, kCFHTTPVersion1_1).takeRetainedValue() | ||||
|         CFHTTPMessageSetHeaderFieldValue(httpResponse, "Content-Length" as CFString, String(messageData.count) as CFString) | ||||
|         CFHTTPMessageSetHeaderFieldValue(httpResponse, "Buffer-Width" as CFString, String(width) as CFString) | ||||
|         CFHTTPMessageSetHeaderFieldValue(httpResponse, "Buffer-Height" as CFString, String(height) as CFString) | ||||
|         CFHTTPMessageSetHeaderFieldValue(httpResponse, "Buffer-Orientation" as CFString, String(orientation) as CFString) | ||||
|          | ||||
|         CFHTTPMessageSetBody(httpResponse, messageData as CFData) | ||||
|          | ||||
|         let serializedMessage = CFHTTPMessageCopySerializedMessage(httpResponse)?.takeRetainedValue() as Data? | ||||
|        | ||||
|         return serializedMessage | ||||
|     } | ||||
|      | ||||
|     func jpegData(from buffer: CVPixelBuffer, scale scaleTransform: CGAffineTransform) -> Data? { | ||||
|         let image = CIImage(cvPixelBuffer: buffer).transformed(by: scaleTransform) | ||||
|          | ||||
|         guard let colorSpace = image.colorSpace else { | ||||
|             return nil | ||||
|         } | ||||
|        | ||||
|         let options: [CIImageRepresentationOption: Float] = [kCGImageDestinationLossyCompressionQuality as CIImageRepresentationOption: 1.0] | ||||
|  | ||||
|         return SampleUploader.imageContext.jpegRepresentation(of: image, colorSpace: colorSpace, options: options) | ||||
|     } | ||||
| } | ||||
							
								
								
									
										199
									
								
								ios/SolianBroadcastExtension/SocketConnection.swift
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										199
									
								
								ios/SolianBroadcastExtension/SocketConnection.swift
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,199 @@ | ||||
| // | ||||
| //  SocketConnection.swift | ||||
| //  Broadcast Extension | ||||
| // | ||||
| //  Created by Alex-Dan Bumbu on 22/03/2021. | ||||
| //  Copyright © 2021 Atlassian Inc. All rights reserved. | ||||
| // | ||||
|  | ||||
| import Foundation | ||||
| import OSLog | ||||
|  | ||||
| class SocketConnection: NSObject { | ||||
|     var didOpen: (() -> Void)? | ||||
|     var didClose: ((Error?) -> Void)? | ||||
|     var streamHasSpaceAvailable: (() -> Void)? | ||||
|  | ||||
|     private let filePath: String | ||||
|     private var socketHandle: Int32 = -1 | ||||
|     private var address: sockaddr_un? | ||||
|  | ||||
|     private var inputStream: InputStream? | ||||
|     private var outputStream: OutputStream? | ||||
|      | ||||
|     private var networkQueue: DispatchQueue? | ||||
|     private var shouldKeepRunning = false | ||||
|  | ||||
|     init?(filePath path: String) { | ||||
|         filePath = path | ||||
|         socketHandle = Darwin.socket(AF_UNIX, SOCK_STREAM, 0) | ||||
|  | ||||
|         guard socketHandle != -1 else { | ||||
|             os_log(.debug, log: broadcastLogger, "failure: create socket") | ||||
|             return nil | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     func open() -> Bool { | ||||
|         os_log(.debug, log: broadcastLogger, "open socket connection") | ||||
|  | ||||
|         guard FileManager.default.fileExists(atPath: filePath) else { | ||||
|             os_log(.debug, log: broadcastLogger, "failure: socket file missing") | ||||
|             return false | ||||
|         } | ||||
|        | ||||
|         guard setupAddress() == true else { | ||||
|             return false | ||||
|         } | ||||
|          | ||||
|         guard connectSocket() == true else { | ||||
|             return false | ||||
|         } | ||||
|  | ||||
|         setupStreams() | ||||
|          | ||||
|         inputStream?.open() | ||||
|         outputStream?.open() | ||||
|  | ||||
|         return true | ||||
|     } | ||||
|  | ||||
|     func close() { | ||||
|         unscheduleStreams() | ||||
|  | ||||
|         inputStream?.delegate = nil | ||||
|         outputStream?.delegate = nil | ||||
|  | ||||
|         inputStream?.close() | ||||
|         outputStream?.close() | ||||
|          | ||||
|         inputStream = nil | ||||
|         outputStream = nil | ||||
|     } | ||||
|  | ||||
|     func writeToStream(buffer: UnsafePointer<UInt8>, maxLength length: Int) -> Int { | ||||
|         outputStream?.write(buffer, maxLength: length) ?? 0 | ||||
|     } | ||||
| } | ||||
|  | ||||
| extension SocketConnection: StreamDelegate { | ||||
|  | ||||
|     func stream(_ aStream: Stream, handle eventCode: Stream.Event) { | ||||
|         switch eventCode { | ||||
|         case .openCompleted: | ||||
|             os_log(.debug, log: broadcastLogger, "client stream open completed") | ||||
|             if aStream == outputStream { | ||||
|                 didOpen?() | ||||
|             } | ||||
|         case .hasBytesAvailable: | ||||
|             if aStream == inputStream { | ||||
|                 var buffer: UInt8 = 0 | ||||
|                 let numberOfBytesRead = inputStream?.read(&buffer, maxLength: 1) | ||||
|                 if numberOfBytesRead == 0 && aStream.streamStatus == .atEnd { | ||||
|                     os_log(.debug, log: broadcastLogger, "server socket closed") | ||||
|                     close() | ||||
|                     notifyDidClose(error: nil) | ||||
|                 } | ||||
|             } | ||||
|         case .hasSpaceAvailable: | ||||
|             if aStream == outputStream { | ||||
|                 streamHasSpaceAvailable?() | ||||
|             } | ||||
|         case .errorOccurred: | ||||
|             os_log(.debug, log: broadcastLogger, "client stream error occured: \(String(describing: aStream.streamError))") | ||||
|             close() | ||||
|             notifyDidClose(error: aStream.streamError) | ||||
|  | ||||
|         default: | ||||
|             break | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| private extension SocketConnection { | ||||
|    | ||||
|     func setupAddress() -> Bool { | ||||
|         var addr = sockaddr_un() | ||||
|         guard filePath.count < MemoryLayout.size(ofValue: addr.sun_path) else { | ||||
|             os_log(.debug, log: broadcastLogger, "failure: fd path is too long") | ||||
|             return false | ||||
|         } | ||||
|  | ||||
|         _ = withUnsafeMutablePointer(to: &addr.sun_path.0) { ptr in | ||||
|             filePath.withCString { | ||||
|                 strncpy(ptr, $0, filePath.count) | ||||
|             } | ||||
|         } | ||||
|          | ||||
|         address = addr | ||||
|         return true | ||||
|     } | ||||
|  | ||||
|     func connectSocket() -> Bool { | ||||
|         guard var addr = address else { | ||||
|             return false | ||||
|         } | ||||
|          | ||||
|         let status = withUnsafePointer(to: &addr) { ptr in | ||||
|             ptr.withMemoryRebound(to: sockaddr.self, capacity: 1) { | ||||
|                 Darwin.connect(socketHandle, $0, socklen_t(MemoryLayout<sockaddr_un>.size)) | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         guard status == noErr else { | ||||
|             os_log(.debug, log: broadcastLogger, "failure: \(status)") | ||||
|             return false | ||||
|         } | ||||
|          | ||||
|         return true | ||||
|     } | ||||
|  | ||||
|     func setupStreams() { | ||||
|         var readStream: Unmanaged<CFReadStream>? | ||||
|         var writeStream: Unmanaged<CFWriteStream>? | ||||
|  | ||||
|         CFStreamCreatePairWithSocket(kCFAllocatorDefault, socketHandle, &readStream, &writeStream) | ||||
|  | ||||
|         inputStream = readStream?.takeRetainedValue() | ||||
|         inputStream?.delegate = self | ||||
|         inputStream?.setProperty(kCFBooleanTrue, forKey: Stream.PropertyKey(kCFStreamPropertyShouldCloseNativeSocket as String)) | ||||
|  | ||||
|         outputStream = writeStream?.takeRetainedValue() | ||||
|         outputStream?.delegate = self | ||||
|         outputStream?.setProperty(kCFBooleanTrue, forKey: Stream.PropertyKey(kCFStreamPropertyShouldCloseNativeSocket as String)) | ||||
|  | ||||
|         scheduleStreams() | ||||
|     } | ||||
|    | ||||
|     func scheduleStreams() { | ||||
|         shouldKeepRunning = true | ||||
|          | ||||
|         networkQueue = DispatchQueue.global(qos: .userInitiated) | ||||
|         networkQueue?.async { [weak self] in | ||||
|             self?.inputStream?.schedule(in: .current, forMode: .common) | ||||
|             self?.outputStream?.schedule(in: .current, forMode: .common) | ||||
|             RunLoop.current.run() | ||||
|              | ||||
|             var isRunning = false | ||||
|                          | ||||
|             repeat { | ||||
|                 isRunning = self?.shouldKeepRunning ?? false && RunLoop.current.run(mode: .default, before: .distantFuture) | ||||
|             } while (isRunning) | ||||
|         } | ||||
|     } | ||||
|      | ||||
|     func unscheduleStreams() { | ||||
|         networkQueue?.sync { [weak self] in | ||||
|             self?.inputStream?.remove(from: .current, forMode: .common) | ||||
|             self?.outputStream?.remove(from: .current, forMode: .common) | ||||
|         } | ||||
|          | ||||
|         shouldKeepRunning = false | ||||
|     } | ||||
|      | ||||
|     func notifyDidClose(error: Error?) { | ||||
|         if didClose != nil { | ||||
|             didClose?(error) | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @@ -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> | ||||
| @@ -49,8 +49,7 @@ class AppDatabase extends _$AppDatabase { | ||||
|   } | ||||
|  | ||||
|   Future<int> updateMessage(ChatMessagesCompanion message) { | ||||
|     return (update(chatMessages) | ||||
|       ..where((m) => m.id.equals(message.id.value))).write(message); | ||||
|     return into(chatMessages).insert(message, mode: InsertMode.insertOrReplace); | ||||
|   } | ||||
|  | ||||
|   Future<int> updateMessageStatus(String id, MessageStatus status) { | ||||
|   | ||||
| @@ -20,7 +20,6 @@ import 'package:bitsdojo_window/bitsdojo_window.dart'; | ||||
| import 'package:island/pods/userinfo.dart'; | ||||
| import 'package:island/pods/websocket.dart'; | ||||
| import 'package:island/route.dart'; | ||||
|  | ||||
| import 'package:island/services/notify.dart'; | ||||
| import 'package:island/services/timezone.dart'; | ||||
| import 'package:island/widgets/alert.dart'; | ||||
| @@ -30,6 +29,7 @@ import 'package:shared_preferences/shared_preferences.dart'; | ||||
| import 'package:image_picker_platform_interface/image_picker_platform_interface.dart'; | ||||
| import 'package:flutter_native_splash/flutter_native_splash.dart'; | ||||
| import 'package:url_launcher/url_launcher_string.dart'; | ||||
| import 'package:flutter_langdetect/flutter_langdetect.dart' as langdetect; | ||||
|  | ||||
| @pragma('vm:entry-point') | ||||
| Future<void> _firebaseMessagingBackgroundHandler(RemoteMessage message) async { | ||||
| @@ -51,6 +51,7 @@ void main() async { | ||||
|   } | ||||
|  | ||||
|   try { | ||||
|     await langdetect.initLangDetect(); | ||||
|     await EasyLocalization.ensureInitialized(); | ||||
|     await Firebase.initializeApp( | ||||
|       options: DefaultFirebaseOptions.currentPlatform, | ||||
|   | ||||
| @@ -162,8 +162,6 @@ sealed class CallParticipant with _$CallParticipant { | ||||
|     required String identity, | ||||
|     required String name, | ||||
|     required DateTime joinedAt, | ||||
|     required String? accountId, | ||||
|     required SnChatMember? profile, | ||||
|   }) = _CallParticipant; | ||||
|  | ||||
|   factory CallParticipant.fromJson(Map<String, dynamic> json) => | ||||
|   | ||||
| @@ -2498,7 +2498,7 @@ as List<CallParticipant>, | ||||
| /// @nodoc | ||||
| mixin _$CallParticipant { | ||||
|  | ||||
|  String get identity; String get name; DateTime get joinedAt; String? get accountId; SnChatMember? get profile; | ||||
|  String get identity; String get name; DateTime get joinedAt; | ||||
| /// Create a copy of CallParticipant | ||||
| /// with the given fields replaced by the non-null parameter values. | ||||
| @JsonKey(includeFromJson: false, includeToJson: false) | ||||
| @@ -2511,16 +2511,16 @@ $CallParticipantCopyWith<CallParticipant> get copyWith => _$CallParticipantCopyW | ||||
|  | ||||
| @override | ||||
| bool operator ==(Object other) { | ||||
|   return identical(this, other) || (other.runtimeType == runtimeType&&other is CallParticipant&&(identical(other.identity, identity) || other.identity == identity)&&(identical(other.name, name) || other.name == name)&&(identical(other.joinedAt, joinedAt) || other.joinedAt == joinedAt)&&(identical(other.accountId, accountId) || other.accountId == accountId)&&(identical(other.profile, profile) || other.profile == profile)); | ||||
|   return identical(this, other) || (other.runtimeType == runtimeType&&other is CallParticipant&&(identical(other.identity, identity) || other.identity == identity)&&(identical(other.name, name) || other.name == name)&&(identical(other.joinedAt, joinedAt) || other.joinedAt == joinedAt)); | ||||
| } | ||||
|  | ||||
| @JsonKey(includeFromJson: false, includeToJson: false) | ||||
| @override | ||||
| int get hashCode => Object.hash(runtimeType,identity,name,joinedAt,accountId,profile); | ||||
| int get hashCode => Object.hash(runtimeType,identity,name,joinedAt); | ||||
|  | ||||
| @override | ||||
| String toString() { | ||||
|   return 'CallParticipant(identity: $identity, name: $name, joinedAt: $joinedAt, accountId: $accountId, profile: $profile)'; | ||||
|   return 'CallParticipant(identity: $identity, name: $name, joinedAt: $joinedAt)'; | ||||
| } | ||||
|  | ||||
|  | ||||
| @@ -2531,11 +2531,11 @@ abstract mixin class $CallParticipantCopyWith<$Res>  { | ||||
|   factory $CallParticipantCopyWith(CallParticipant value, $Res Function(CallParticipant) _then) = _$CallParticipantCopyWithImpl; | ||||
| @useResult | ||||
| $Res call({ | ||||
|  String identity, String name, DateTime joinedAt, String? accountId, SnChatMember? profile | ||||
|  String identity, String name, DateTime joinedAt | ||||
| }); | ||||
|  | ||||
|  | ||||
| $SnChatMemberCopyWith<$Res>? get profile; | ||||
|  | ||||
|  | ||||
| } | ||||
| /// @nodoc | ||||
| @@ -2548,29 +2548,15 @@ class _$CallParticipantCopyWithImpl<$Res> | ||||
|  | ||||
| /// Create a copy of CallParticipant | ||||
| /// with the given fields replaced by the non-null parameter values. | ||||
| @pragma('vm:prefer-inline') @override $Res call({Object? identity = null,Object? name = null,Object? joinedAt = null,Object? accountId = freezed,Object? profile = freezed,}) { | ||||
| @pragma('vm:prefer-inline') @override $Res call({Object? identity = null,Object? name = null,Object? joinedAt = null,}) { | ||||
|   return _then(_self.copyWith( | ||||
| identity: null == identity ? _self.identity : identity // ignore: cast_nullable_to_non_nullable | ||||
| as String,name: null == name ? _self.name : name // ignore: cast_nullable_to_non_nullable | ||||
| as String,joinedAt: null == joinedAt ? _self.joinedAt : joinedAt // ignore: cast_nullable_to_non_nullable | ||||
| as DateTime,accountId: freezed == accountId ? _self.accountId : accountId // ignore: cast_nullable_to_non_nullable | ||||
| as String?,profile: freezed == profile ? _self.profile : profile // ignore: cast_nullable_to_non_nullable | ||||
| as SnChatMember?, | ||||
| as DateTime, | ||||
|   )); | ||||
| } | ||||
| /// Create a copy of CallParticipant | ||||
| /// with the given fields replaced by the non-null parameter values. | ||||
| @override | ||||
| @pragma('vm:prefer-inline') | ||||
| $SnChatMemberCopyWith<$Res>? get profile { | ||||
|     if (_self.profile == null) { | ||||
|     return null; | ||||
|   } | ||||
|  | ||||
|   return $SnChatMemberCopyWith<$Res>(_self.profile!, (value) { | ||||
|     return _then(_self.copyWith(profile: value)); | ||||
|   }); | ||||
| } | ||||
| } | ||||
|  | ||||
|  | ||||
| @@ -2649,10 +2635,10 @@ return $default(_that);case _: | ||||
| /// } | ||||
| /// ``` | ||||
|  | ||||
| @optionalTypeArgs TResult maybeWhen<TResult extends Object?>(TResult Function( String identity,  String name,  DateTime joinedAt,  String? accountId,  SnChatMember? profile)?  $default,{required TResult orElse(),}) {final _that = this; | ||||
| @optionalTypeArgs TResult maybeWhen<TResult extends Object?>(TResult Function( String identity,  String name,  DateTime joinedAt)?  $default,{required TResult orElse(),}) {final _that = this; | ||||
| switch (_that) { | ||||
| case _CallParticipant() when $default != null: | ||||
| return $default(_that.identity,_that.name,_that.joinedAt,_that.accountId,_that.profile);case _: | ||||
| return $default(_that.identity,_that.name,_that.joinedAt);case _: | ||||
|   return orElse(); | ||||
|  | ||||
| } | ||||
| @@ -2670,10 +2656,10 @@ return $default(_that.identity,_that.name,_that.joinedAt,_that.accountId,_that.p | ||||
| /// } | ||||
| /// ``` | ||||
|  | ||||
| @optionalTypeArgs TResult when<TResult extends Object?>(TResult Function( String identity,  String name,  DateTime joinedAt,  String? accountId,  SnChatMember? profile)  $default,) {final _that = this; | ||||
| @optionalTypeArgs TResult when<TResult extends Object?>(TResult Function( String identity,  String name,  DateTime joinedAt)  $default,) {final _that = this; | ||||
| switch (_that) { | ||||
| case _CallParticipant(): | ||||
| return $default(_that.identity,_that.name,_that.joinedAt,_that.accountId,_that.profile);} | ||||
| return $default(_that.identity,_that.name,_that.joinedAt);} | ||||
| } | ||||
| /// A variant of `when` that fallback to returning `null` | ||||
| /// | ||||
| @@ -2687,10 +2673,10 @@ return $default(_that.identity,_that.name,_that.joinedAt,_that.accountId,_that.p | ||||
| /// } | ||||
| /// ``` | ||||
|  | ||||
| @optionalTypeArgs TResult? whenOrNull<TResult extends Object?>(TResult? Function( String identity,  String name,  DateTime joinedAt,  String? accountId,  SnChatMember? profile)?  $default,) {final _that = this; | ||||
| @optionalTypeArgs TResult? whenOrNull<TResult extends Object?>(TResult? Function( String identity,  String name,  DateTime joinedAt)?  $default,) {final _that = this; | ||||
| switch (_that) { | ||||
| case _CallParticipant() when $default != null: | ||||
| return $default(_that.identity,_that.name,_that.joinedAt,_that.accountId,_that.profile);case _: | ||||
| return $default(_that.identity,_that.name,_that.joinedAt);case _: | ||||
|   return null; | ||||
|  | ||||
| } | ||||
| @@ -2702,14 +2688,12 @@ return $default(_that.identity,_that.name,_that.joinedAt,_that.accountId,_that.p | ||||
| @JsonSerializable() | ||||
|  | ||||
| class _CallParticipant implements CallParticipant { | ||||
|   const _CallParticipant({required this.identity, required this.name, required this.joinedAt, required this.accountId, required this.profile}); | ||||
|   const _CallParticipant({required this.identity, required this.name, required this.joinedAt}); | ||||
|   factory _CallParticipant.fromJson(Map<String, dynamic> json) => _$CallParticipantFromJson(json); | ||||
|  | ||||
| @override final  String identity; | ||||
| @override final  String name; | ||||
| @override final  DateTime joinedAt; | ||||
| @override final  String? accountId; | ||||
| @override final  SnChatMember? profile; | ||||
|  | ||||
| /// Create a copy of CallParticipant | ||||
| /// with the given fields replaced by the non-null parameter values. | ||||
| @@ -2724,16 +2708,16 @@ Map<String, dynamic> toJson() { | ||||
|  | ||||
| @override | ||||
| bool operator ==(Object other) { | ||||
|   return identical(this, other) || (other.runtimeType == runtimeType&&other is _CallParticipant&&(identical(other.identity, identity) || other.identity == identity)&&(identical(other.name, name) || other.name == name)&&(identical(other.joinedAt, joinedAt) || other.joinedAt == joinedAt)&&(identical(other.accountId, accountId) || other.accountId == accountId)&&(identical(other.profile, profile) || other.profile == profile)); | ||||
|   return identical(this, other) || (other.runtimeType == runtimeType&&other is _CallParticipant&&(identical(other.identity, identity) || other.identity == identity)&&(identical(other.name, name) || other.name == name)&&(identical(other.joinedAt, joinedAt) || other.joinedAt == joinedAt)); | ||||
| } | ||||
|  | ||||
| @JsonKey(includeFromJson: false, includeToJson: false) | ||||
| @override | ||||
| int get hashCode => Object.hash(runtimeType,identity,name,joinedAt,accountId,profile); | ||||
| int get hashCode => Object.hash(runtimeType,identity,name,joinedAt); | ||||
|  | ||||
| @override | ||||
| String toString() { | ||||
|   return 'CallParticipant(identity: $identity, name: $name, joinedAt: $joinedAt, accountId: $accountId, profile: $profile)'; | ||||
|   return 'CallParticipant(identity: $identity, name: $name, joinedAt: $joinedAt)'; | ||||
| } | ||||
|  | ||||
|  | ||||
| @@ -2744,11 +2728,11 @@ abstract mixin class _$CallParticipantCopyWith<$Res> implements $CallParticipant | ||||
|   factory _$CallParticipantCopyWith(_CallParticipant value, $Res Function(_CallParticipant) _then) = __$CallParticipantCopyWithImpl; | ||||
| @override @useResult | ||||
| $Res call({ | ||||
|  String identity, String name, DateTime joinedAt, String? accountId, SnChatMember? profile | ||||
|  String identity, String name, DateTime joinedAt | ||||
| }); | ||||
|  | ||||
|  | ||||
| @override $SnChatMemberCopyWith<$Res>? get profile; | ||||
|  | ||||
|  | ||||
| } | ||||
| /// @nodoc | ||||
| @@ -2761,30 +2745,16 @@ class __$CallParticipantCopyWithImpl<$Res> | ||||
|  | ||||
| /// Create a copy of CallParticipant | ||||
| /// with the given fields replaced by the non-null parameter values. | ||||
| @override @pragma('vm:prefer-inline') $Res call({Object? identity = null,Object? name = null,Object? joinedAt = null,Object? accountId = freezed,Object? profile = freezed,}) { | ||||
| @override @pragma('vm:prefer-inline') $Res call({Object? identity = null,Object? name = null,Object? joinedAt = null,}) { | ||||
|   return _then(_CallParticipant( | ||||
| identity: null == identity ? _self.identity : identity // ignore: cast_nullable_to_non_nullable | ||||
| as String,name: null == name ? _self.name : name // ignore: cast_nullable_to_non_nullable | ||||
| as String,joinedAt: null == joinedAt ? _self.joinedAt : joinedAt // ignore: cast_nullable_to_non_nullable | ||||
| as DateTime,accountId: freezed == accountId ? _self.accountId : accountId // ignore: cast_nullable_to_non_nullable | ||||
| as String?,profile: freezed == profile ? _self.profile : profile // ignore: cast_nullable_to_non_nullable | ||||
| as SnChatMember?, | ||||
| as DateTime, | ||||
|   )); | ||||
| } | ||||
|  | ||||
| /// Create a copy of CallParticipant | ||||
| /// with the given fields replaced by the non-null parameter values. | ||||
| @override | ||||
| @pragma('vm:prefer-inline') | ||||
| $SnChatMemberCopyWith<$Res>? get profile { | ||||
|     if (_self.profile == null) { | ||||
|     return null; | ||||
|   } | ||||
|  | ||||
|   return $SnChatMemberCopyWith<$Res>(_self.profile!, (value) { | ||||
|     return _then(_self.copyWith(profile: value)); | ||||
|   }); | ||||
| } | ||||
| } | ||||
|  | ||||
|  | ||||
|   | ||||
| @@ -285,11 +285,6 @@ _CallParticipant _$CallParticipantFromJson(Map<String, dynamic> json) => | ||||
|       identity: json['identity'] as String, | ||||
|       name: json['name'] as String, | ||||
|       joinedAt: DateTime.parse(json['joined_at'] as String), | ||||
|       accountId: json['account_id'] as String?, | ||||
|       profile: | ||||
|           json['profile'] == null | ||||
|               ? null | ||||
|               : SnChatMember.fromJson(json['profile'] as Map<String, dynamic>), | ||||
|     ); | ||||
|  | ||||
| Map<String, dynamic> _$CallParticipantToJson(_CallParticipant instance) => | ||||
| @@ -297,8 +292,6 @@ Map<String, dynamic> _$CallParticipantToJson(_CallParticipant instance) => | ||||
|       'identity': instance.identity, | ||||
|       'name': instance.name, | ||||
|       'joined_at': instance.joinedAt.toIso8601String(), | ||||
|       'account_id': instance.accountId, | ||||
|       'profile': instance.profile?.toJson(), | ||||
|     }; | ||||
|  | ||||
| _SnRealtimeCall _$SnRealtimeCallFromJson(Map<String, dynamic> json) => | ||||
|   | ||||
| @@ -3,25 +3,6 @@ import 'package:freezed_annotation/freezed_annotation.dart'; | ||||
| part 'embed.freezed.dart'; | ||||
| part 'embed.g.dart'; | ||||
|  | ||||
| @freezed | ||||
| sealed class SnEmbedLink with _$SnEmbedLink { | ||||
|   const factory SnEmbedLink({ | ||||
|     @JsonKey(name: 'Type') required String type, | ||||
|     @JsonKey(name: 'Url') required String url, | ||||
|     @JsonKey(name: 'Title') required String title, | ||||
|     @JsonKey(name: 'Description') required String? description, | ||||
|     @JsonKey(name: 'ImageUrl') required String? imageUrl, | ||||
|     @JsonKey(name: 'FaviconUrl') required String faviconUrl, | ||||
|     @JsonKey(name: 'SiteName') required String siteName, | ||||
|     @JsonKey(name: 'ContentType') required String? contentType, | ||||
|     @JsonKey(name: 'Author') required String? author, | ||||
|     @JsonKey(name: 'PublishedDate') required DateTime? publishedDate, | ||||
|   }) = _SnEmbedLink; | ||||
|  | ||||
|   factory SnEmbedLink.fromJson(Map<String, dynamic> json) => | ||||
|       _$SnEmbedLinkFromJson(json); | ||||
| } | ||||
|  | ||||
| @freezed | ||||
| sealed class SnScrappedLink with _$SnScrappedLink { | ||||
|   const factory SnScrappedLink({ | ||||
|   | ||||
| @@ -12,290 +12,6 @@ part of 'embed.dart'; | ||||
| // dart format off | ||||
| T _$identity<T>(T value) => value; | ||||
|  | ||||
| /// @nodoc | ||||
| mixin _$SnEmbedLink { | ||||
|  | ||||
| @JsonKey(name: 'Type') String get type;@JsonKey(name: 'Url') String get url;@JsonKey(name: 'Title') String get title;@JsonKey(name: 'Description') String? get description;@JsonKey(name: 'ImageUrl') String? get imageUrl;@JsonKey(name: 'FaviconUrl') String get faviconUrl;@JsonKey(name: 'SiteName') String get siteName;@JsonKey(name: 'ContentType') String? get contentType;@JsonKey(name: 'Author') String? get author;@JsonKey(name: 'PublishedDate') DateTime? get publishedDate; | ||||
| /// Create a copy of SnEmbedLink | ||||
| /// with the given fields replaced by the non-null parameter values. | ||||
| @JsonKey(includeFromJson: false, includeToJson: false) | ||||
| @pragma('vm:prefer-inline') | ||||
| $SnEmbedLinkCopyWith<SnEmbedLink> get copyWith => _$SnEmbedLinkCopyWithImpl<SnEmbedLink>(this as SnEmbedLink, _$identity); | ||||
|  | ||||
|   /// Serializes this SnEmbedLink to a JSON map. | ||||
|   Map<String, dynamic> toJson(); | ||||
|  | ||||
|  | ||||
| @override | ||||
| bool operator ==(Object other) { | ||||
|   return identical(this, other) || (other.runtimeType == runtimeType&&other is SnEmbedLink&&(identical(other.type, type) || other.type == type)&&(identical(other.url, url) || other.url == url)&&(identical(other.title, title) || other.title == title)&&(identical(other.description, description) || other.description == description)&&(identical(other.imageUrl, imageUrl) || other.imageUrl == imageUrl)&&(identical(other.faviconUrl, faviconUrl) || other.faviconUrl == faviconUrl)&&(identical(other.siteName, siteName) || other.siteName == siteName)&&(identical(other.contentType, contentType) || other.contentType == contentType)&&(identical(other.author, author) || other.author == author)&&(identical(other.publishedDate, publishedDate) || other.publishedDate == publishedDate)); | ||||
| } | ||||
|  | ||||
| @JsonKey(includeFromJson: false, includeToJson: false) | ||||
| @override | ||||
| int get hashCode => Object.hash(runtimeType,type,url,title,description,imageUrl,faviconUrl,siteName,contentType,author,publishedDate); | ||||
|  | ||||
| @override | ||||
| String toString() { | ||||
|   return 'SnEmbedLink(type: $type, url: $url, title: $title, description: $description, imageUrl: $imageUrl, faviconUrl: $faviconUrl, siteName: $siteName, contentType: $contentType, author: $author, publishedDate: $publishedDate)'; | ||||
| } | ||||
|  | ||||
|  | ||||
| } | ||||
|  | ||||
| /// @nodoc | ||||
| abstract mixin class $SnEmbedLinkCopyWith<$Res>  { | ||||
|   factory $SnEmbedLinkCopyWith(SnEmbedLink value, $Res Function(SnEmbedLink) _then) = _$SnEmbedLinkCopyWithImpl; | ||||
| @useResult | ||||
| $Res call({ | ||||
| @JsonKey(name: 'Type') String type,@JsonKey(name: 'Url') String url,@JsonKey(name: 'Title') String title,@JsonKey(name: 'Description') String? description,@JsonKey(name: 'ImageUrl') String? imageUrl,@JsonKey(name: 'FaviconUrl') String faviconUrl,@JsonKey(name: 'SiteName') String siteName,@JsonKey(name: 'ContentType') String? contentType,@JsonKey(name: 'Author') String? author,@JsonKey(name: 'PublishedDate') DateTime? publishedDate | ||||
| }); | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
| } | ||||
| /// @nodoc | ||||
| class _$SnEmbedLinkCopyWithImpl<$Res> | ||||
|     implements $SnEmbedLinkCopyWith<$Res> { | ||||
|   _$SnEmbedLinkCopyWithImpl(this._self, this._then); | ||||
|  | ||||
|   final SnEmbedLink _self; | ||||
|   final $Res Function(SnEmbedLink) _then; | ||||
|  | ||||
| /// Create a copy of SnEmbedLink | ||||
| /// with the given fields replaced by the non-null parameter values. | ||||
| @pragma('vm:prefer-inline') @override $Res call({Object? type = null,Object? url = null,Object? title = null,Object? description = freezed,Object? imageUrl = freezed,Object? faviconUrl = null,Object? siteName = null,Object? contentType = freezed,Object? author = freezed,Object? publishedDate = freezed,}) { | ||||
|   return _then(_self.copyWith( | ||||
| type: null == type ? _self.type : type // ignore: cast_nullable_to_non_nullable | ||||
| as String,url: null == url ? _self.url : url // ignore: cast_nullable_to_non_nullable | ||||
| as String,title: null == title ? _self.title : title // ignore: cast_nullable_to_non_nullable | ||||
| as String,description: freezed == description ? _self.description : description // ignore: cast_nullable_to_non_nullable | ||||
| as String?,imageUrl: freezed == imageUrl ? _self.imageUrl : imageUrl // ignore: cast_nullable_to_non_nullable | ||||
| as String?,faviconUrl: null == faviconUrl ? _self.faviconUrl : faviconUrl // ignore: cast_nullable_to_non_nullable | ||||
| as String,siteName: null == siteName ? _self.siteName : siteName // ignore: cast_nullable_to_non_nullable | ||||
| as String,contentType: freezed == contentType ? _self.contentType : contentType // ignore: cast_nullable_to_non_nullable | ||||
| as String?,author: freezed == author ? _self.author : author // ignore: cast_nullable_to_non_nullable | ||||
| as String?,publishedDate: freezed == publishedDate ? _self.publishedDate : publishedDate // ignore: cast_nullable_to_non_nullable | ||||
| as DateTime?, | ||||
|   )); | ||||
| } | ||||
|  | ||||
| } | ||||
|  | ||||
|  | ||||
| /// Adds pattern-matching-related methods to [SnEmbedLink]. | ||||
| extension SnEmbedLinkPatterns on SnEmbedLink { | ||||
| /// A variant of `map` that fallback to returning `orElse`. | ||||
| /// | ||||
| /// It is equivalent to doing: | ||||
| /// ```dart | ||||
| /// switch (sealedClass) { | ||||
| ///   case final Subclass value: | ||||
| ///     return ...; | ||||
| ///   case _: | ||||
| ///     return orElse(); | ||||
| /// } | ||||
| /// ``` | ||||
|  | ||||
| @optionalTypeArgs TResult maybeMap<TResult extends Object?>(TResult Function( _SnEmbedLink value)?  $default,{required TResult orElse(),}){ | ||||
| final _that = this; | ||||
| switch (_that) { | ||||
| case _SnEmbedLink() when $default != null: | ||||
| return $default(_that);case _: | ||||
|   return orElse(); | ||||
|  | ||||
| } | ||||
| } | ||||
| /// A `switch`-like method, using callbacks. | ||||
| /// | ||||
| /// Callbacks receives the raw object, upcasted. | ||||
| /// It is equivalent to doing: | ||||
| /// ```dart | ||||
| /// switch (sealedClass) { | ||||
| ///   case final Subclass value: | ||||
| ///     return ...; | ||||
| ///   case final Subclass2 value: | ||||
| ///     return ...; | ||||
| /// } | ||||
| /// ``` | ||||
|  | ||||
| @optionalTypeArgs TResult map<TResult extends Object?>(TResult Function( _SnEmbedLink value)  $default,){ | ||||
| final _that = this; | ||||
| switch (_that) { | ||||
| case _SnEmbedLink(): | ||||
| return $default(_that);} | ||||
| } | ||||
| /// A variant of `map` that fallback to returning `null`. | ||||
| /// | ||||
| /// It is equivalent to doing: | ||||
| /// ```dart | ||||
| /// switch (sealedClass) { | ||||
| ///   case final Subclass value: | ||||
| ///     return ...; | ||||
| ///   case _: | ||||
| ///     return null; | ||||
| /// } | ||||
| /// ``` | ||||
|  | ||||
| @optionalTypeArgs TResult? mapOrNull<TResult extends Object?>(TResult? Function( _SnEmbedLink value)?  $default,){ | ||||
| final _that = this; | ||||
| switch (_that) { | ||||
| case _SnEmbedLink() when $default != null: | ||||
| return $default(_that);case _: | ||||
|   return null; | ||||
|  | ||||
| } | ||||
| } | ||||
| /// A variant of `when` that fallback to an `orElse` callback. | ||||
| /// | ||||
| /// It is equivalent to doing: | ||||
| /// ```dart | ||||
| /// switch (sealedClass) { | ||||
| ///   case Subclass(:final field): | ||||
| ///     return ...; | ||||
| ///   case _: | ||||
| ///     return orElse(); | ||||
| /// } | ||||
| /// ``` | ||||
|  | ||||
| @optionalTypeArgs TResult maybeWhen<TResult extends Object?>(TResult Function(@JsonKey(name: 'Type')  String type, @JsonKey(name: 'Url')  String url, @JsonKey(name: 'Title')  String title, @JsonKey(name: 'Description')  String? description, @JsonKey(name: 'ImageUrl')  String? imageUrl, @JsonKey(name: 'FaviconUrl')  String faviconUrl, @JsonKey(name: 'SiteName')  String siteName, @JsonKey(name: 'ContentType')  String? contentType, @JsonKey(name: 'Author')  String? author, @JsonKey(name: 'PublishedDate')  DateTime? publishedDate)?  $default,{required TResult orElse(),}) {final _that = this; | ||||
| switch (_that) { | ||||
| case _SnEmbedLink() when $default != null: | ||||
| return $default(_that.type,_that.url,_that.title,_that.description,_that.imageUrl,_that.faviconUrl,_that.siteName,_that.contentType,_that.author,_that.publishedDate);case _: | ||||
|   return orElse(); | ||||
|  | ||||
| } | ||||
| } | ||||
| /// A `switch`-like method, using callbacks. | ||||
| /// | ||||
| /// As opposed to `map`, this offers destructuring. | ||||
| /// It is equivalent to doing: | ||||
| /// ```dart | ||||
| /// switch (sealedClass) { | ||||
| ///   case Subclass(:final field): | ||||
| ///     return ...; | ||||
| ///   case Subclass2(:final field2): | ||||
| ///     return ...; | ||||
| /// } | ||||
| /// ``` | ||||
|  | ||||
| @optionalTypeArgs TResult when<TResult extends Object?>(TResult Function(@JsonKey(name: 'Type')  String type, @JsonKey(name: 'Url')  String url, @JsonKey(name: 'Title')  String title, @JsonKey(name: 'Description')  String? description, @JsonKey(name: 'ImageUrl')  String? imageUrl, @JsonKey(name: 'FaviconUrl')  String faviconUrl, @JsonKey(name: 'SiteName')  String siteName, @JsonKey(name: 'ContentType')  String? contentType, @JsonKey(name: 'Author')  String? author, @JsonKey(name: 'PublishedDate')  DateTime? publishedDate)  $default,) {final _that = this; | ||||
| switch (_that) { | ||||
| case _SnEmbedLink(): | ||||
| return $default(_that.type,_that.url,_that.title,_that.description,_that.imageUrl,_that.faviconUrl,_that.siteName,_that.contentType,_that.author,_that.publishedDate);} | ||||
| } | ||||
| /// A variant of `when` that fallback to returning `null` | ||||
| /// | ||||
| /// It is equivalent to doing: | ||||
| /// ```dart | ||||
| /// switch (sealedClass) { | ||||
| ///   case Subclass(:final field): | ||||
| ///     return ...; | ||||
| ///   case _: | ||||
| ///     return null; | ||||
| /// } | ||||
| /// ``` | ||||
|  | ||||
| @optionalTypeArgs TResult? whenOrNull<TResult extends Object?>(TResult? Function(@JsonKey(name: 'Type')  String type, @JsonKey(name: 'Url')  String url, @JsonKey(name: 'Title')  String title, @JsonKey(name: 'Description')  String? description, @JsonKey(name: 'ImageUrl')  String? imageUrl, @JsonKey(name: 'FaviconUrl')  String faviconUrl, @JsonKey(name: 'SiteName')  String siteName, @JsonKey(name: 'ContentType')  String? contentType, @JsonKey(name: 'Author')  String? author, @JsonKey(name: 'PublishedDate')  DateTime? publishedDate)?  $default,) {final _that = this; | ||||
| switch (_that) { | ||||
| case _SnEmbedLink() when $default != null: | ||||
| return $default(_that.type,_that.url,_that.title,_that.description,_that.imageUrl,_that.faviconUrl,_that.siteName,_that.contentType,_that.author,_that.publishedDate);case _: | ||||
|   return null; | ||||
|  | ||||
| } | ||||
| } | ||||
|  | ||||
| } | ||||
|  | ||||
| /// @nodoc | ||||
| @JsonSerializable() | ||||
|  | ||||
| class _SnEmbedLink implements SnEmbedLink { | ||||
|   const _SnEmbedLink({@JsonKey(name: 'Type') required this.type, @JsonKey(name: 'Url') required this.url, @JsonKey(name: 'Title') required this.title, @JsonKey(name: 'Description') required this.description, @JsonKey(name: 'ImageUrl') required this.imageUrl, @JsonKey(name: 'FaviconUrl') required this.faviconUrl, @JsonKey(name: 'SiteName') required this.siteName, @JsonKey(name: 'ContentType') required this.contentType, @JsonKey(name: 'Author') required this.author, @JsonKey(name: 'PublishedDate') required this.publishedDate}); | ||||
|   factory _SnEmbedLink.fromJson(Map<String, dynamic> json) => _$SnEmbedLinkFromJson(json); | ||||
|  | ||||
| @override@JsonKey(name: 'Type') final  String type; | ||||
| @override@JsonKey(name: 'Url') final  String url; | ||||
| @override@JsonKey(name: 'Title') final  String title; | ||||
| @override@JsonKey(name: 'Description') final  String? description; | ||||
| @override@JsonKey(name: 'ImageUrl') final  String? imageUrl; | ||||
| @override@JsonKey(name: 'FaviconUrl') final  String faviconUrl; | ||||
| @override@JsonKey(name: 'SiteName') final  String siteName; | ||||
| @override@JsonKey(name: 'ContentType') final  String? contentType; | ||||
| @override@JsonKey(name: 'Author') final  String? author; | ||||
| @override@JsonKey(name: 'PublishedDate') final  DateTime? publishedDate; | ||||
|  | ||||
| /// Create a copy of SnEmbedLink | ||||
| /// with the given fields replaced by the non-null parameter values. | ||||
| @override @JsonKey(includeFromJson: false, includeToJson: false) | ||||
| @pragma('vm:prefer-inline') | ||||
| _$SnEmbedLinkCopyWith<_SnEmbedLink> get copyWith => __$SnEmbedLinkCopyWithImpl<_SnEmbedLink>(this, _$identity); | ||||
|  | ||||
| @override | ||||
| Map<String, dynamic> toJson() { | ||||
|   return _$SnEmbedLinkToJson(this, ); | ||||
| } | ||||
|  | ||||
| @override | ||||
| bool operator ==(Object other) { | ||||
|   return identical(this, other) || (other.runtimeType == runtimeType&&other is _SnEmbedLink&&(identical(other.type, type) || other.type == type)&&(identical(other.url, url) || other.url == url)&&(identical(other.title, title) || other.title == title)&&(identical(other.description, description) || other.description == description)&&(identical(other.imageUrl, imageUrl) || other.imageUrl == imageUrl)&&(identical(other.faviconUrl, faviconUrl) || other.faviconUrl == faviconUrl)&&(identical(other.siteName, siteName) || other.siteName == siteName)&&(identical(other.contentType, contentType) || other.contentType == contentType)&&(identical(other.author, author) || other.author == author)&&(identical(other.publishedDate, publishedDate) || other.publishedDate == publishedDate)); | ||||
| } | ||||
|  | ||||
| @JsonKey(includeFromJson: false, includeToJson: false) | ||||
| @override | ||||
| int get hashCode => Object.hash(runtimeType,type,url,title,description,imageUrl,faviconUrl,siteName,contentType,author,publishedDate); | ||||
|  | ||||
| @override | ||||
| String toString() { | ||||
|   return 'SnEmbedLink(type: $type, url: $url, title: $title, description: $description, imageUrl: $imageUrl, faviconUrl: $faviconUrl, siteName: $siteName, contentType: $contentType, author: $author, publishedDate: $publishedDate)'; | ||||
| } | ||||
|  | ||||
|  | ||||
| } | ||||
|  | ||||
| /// @nodoc | ||||
| abstract mixin class _$SnEmbedLinkCopyWith<$Res> implements $SnEmbedLinkCopyWith<$Res> { | ||||
|   factory _$SnEmbedLinkCopyWith(_SnEmbedLink value, $Res Function(_SnEmbedLink) _then) = __$SnEmbedLinkCopyWithImpl; | ||||
| @override @useResult | ||||
| $Res call({ | ||||
| @JsonKey(name: 'Type') String type,@JsonKey(name: 'Url') String url,@JsonKey(name: 'Title') String title,@JsonKey(name: 'Description') String? description,@JsonKey(name: 'ImageUrl') String? imageUrl,@JsonKey(name: 'FaviconUrl') String faviconUrl,@JsonKey(name: 'SiteName') String siteName,@JsonKey(name: 'ContentType') String? contentType,@JsonKey(name: 'Author') String? author,@JsonKey(name: 'PublishedDate') DateTime? publishedDate | ||||
| }); | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
| } | ||||
| /// @nodoc | ||||
| class __$SnEmbedLinkCopyWithImpl<$Res> | ||||
|     implements _$SnEmbedLinkCopyWith<$Res> { | ||||
|   __$SnEmbedLinkCopyWithImpl(this._self, this._then); | ||||
|  | ||||
|   final _SnEmbedLink _self; | ||||
|   final $Res Function(_SnEmbedLink) _then; | ||||
|  | ||||
| /// Create a copy of SnEmbedLink | ||||
| /// with the given fields replaced by the non-null parameter values. | ||||
| @override @pragma('vm:prefer-inline') $Res call({Object? type = null,Object? url = null,Object? title = null,Object? description = freezed,Object? imageUrl = freezed,Object? faviconUrl = null,Object? siteName = null,Object? contentType = freezed,Object? author = freezed,Object? publishedDate = freezed,}) { | ||||
|   return _then(_SnEmbedLink( | ||||
| type: null == type ? _self.type : type // ignore: cast_nullable_to_non_nullable | ||||
| as String,url: null == url ? _self.url : url // ignore: cast_nullable_to_non_nullable | ||||
| as String,title: null == title ? _self.title : title // ignore: cast_nullable_to_non_nullable | ||||
| as String,description: freezed == description ? _self.description : description // ignore: cast_nullable_to_non_nullable | ||||
| as String?,imageUrl: freezed == imageUrl ? _self.imageUrl : imageUrl // ignore: cast_nullable_to_non_nullable | ||||
| as String?,faviconUrl: null == faviconUrl ? _self.faviconUrl : faviconUrl // ignore: cast_nullable_to_non_nullable | ||||
| as String,siteName: null == siteName ? _self.siteName : siteName // ignore: cast_nullable_to_non_nullable | ||||
| as String,contentType: freezed == contentType ? _self.contentType : contentType // ignore: cast_nullable_to_non_nullable | ||||
| as String?,author: freezed == author ? _self.author : author // ignore: cast_nullable_to_non_nullable | ||||
| as String?,publishedDate: freezed == publishedDate ? _self.publishedDate : publishedDate // ignore: cast_nullable_to_non_nullable | ||||
| as DateTime?, | ||||
|   )); | ||||
| } | ||||
|  | ||||
|  | ||||
| } | ||||
|  | ||||
|  | ||||
| /// @nodoc | ||||
| mixin _$SnScrappedLink { | ||||
|  | ||||
|   | ||||
| @@ -6,36 +6,6 @@ part of 'embed.dart'; | ||||
| // JsonSerializableGenerator | ||||
| // ************************************************************************** | ||||
|  | ||||
| _SnEmbedLink _$SnEmbedLinkFromJson(Map<String, dynamic> json) => _SnEmbedLink( | ||||
|   type: json['Type'] as String, | ||||
|   url: json['Url'] as String, | ||||
|   title: json['Title'] as String, | ||||
|   description: json['Description'] as String?, | ||||
|   imageUrl: json['ImageUrl'] as String?, | ||||
|   faviconUrl: json['FaviconUrl'] as String, | ||||
|   siteName: json['SiteName'] as String, | ||||
|   contentType: json['ContentType'] as String?, | ||||
|   author: json['Author'] as String?, | ||||
|   publishedDate: | ||||
|       json['PublishedDate'] == null | ||||
|           ? null | ||||
|           : DateTime.parse(json['PublishedDate'] as String), | ||||
| ); | ||||
|  | ||||
| Map<String, dynamic> _$SnEmbedLinkToJson(_SnEmbedLink instance) => | ||||
|     <String, dynamic>{ | ||||
|       'Type': instance.type, | ||||
|       'Url': instance.url, | ||||
|       'Title': instance.title, | ||||
|       'Description': instance.description, | ||||
|       'ImageUrl': instance.imageUrl, | ||||
|       'FaviconUrl': instance.faviconUrl, | ||||
|       'SiteName': instance.siteName, | ||||
|       'ContentType': instance.contentType, | ||||
|       'Author': instance.author, | ||||
|       'PublishedDate': instance.publishedDate?.toIso8601String(), | ||||
|     }; | ||||
|  | ||||
| _SnScrappedLink _$SnScrappedLinkFromJson(Map<String, dynamic> json) => | ||||
|     _SnScrappedLink( | ||||
|       type: json['type'] as String, | ||||
|   | ||||
| @@ -12,6 +12,7 @@ sealed class UniversalFile with _$UniversalFile { | ||||
|   const factory UniversalFile({ | ||||
|     required dynamic data, | ||||
|     required UniversalFileType type, | ||||
|     @Default(false) bool isLink, | ||||
|   }) = _UniversalFile; | ||||
|  | ||||
|   factory UniversalFile.fromJson(Map<String, dynamic> json) => | ||||
| @@ -41,6 +42,7 @@ sealed class SnCloudFile with _$SnCloudFile { | ||||
|     required String? description, | ||||
|     required Map<String, dynamic>? fileMeta, | ||||
|     required Map<String, dynamic>? userMeta, | ||||
|     @Default([]) List<int> sensitiveMarks, | ||||
|     required String? mimeType, | ||||
|     required String? hash, | ||||
|     required int size, | ||||
|   | ||||
| @@ -15,7 +15,7 @@ T _$identity<T>(T value) => value; | ||||
| /// @nodoc | ||||
| mixin _$UniversalFile { | ||||
|  | ||||
|  dynamic get data; UniversalFileType get type; | ||||
|  dynamic get data; UniversalFileType get type; bool get isLink; | ||||
| /// Create a copy of UniversalFile | ||||
| /// with the given fields replaced by the non-null parameter values. | ||||
| @JsonKey(includeFromJson: false, includeToJson: false) | ||||
| @@ -28,16 +28,16 @@ $UniversalFileCopyWith<UniversalFile> get copyWith => _$UniversalFileCopyWithImp | ||||
|  | ||||
| @override | ||||
| bool operator ==(Object other) { | ||||
|   return identical(this, other) || (other.runtimeType == runtimeType&&other is UniversalFile&&const DeepCollectionEquality().equals(other.data, data)&&(identical(other.type, type) || other.type == type)); | ||||
|   return identical(this, other) || (other.runtimeType == runtimeType&&other is UniversalFile&&const DeepCollectionEquality().equals(other.data, data)&&(identical(other.type, type) || other.type == type)&&(identical(other.isLink, isLink) || other.isLink == isLink)); | ||||
| } | ||||
|  | ||||
| @JsonKey(includeFromJson: false, includeToJson: false) | ||||
| @override | ||||
| int get hashCode => Object.hash(runtimeType,const DeepCollectionEquality().hash(data),type); | ||||
| int get hashCode => Object.hash(runtimeType,const DeepCollectionEquality().hash(data),type,isLink); | ||||
|  | ||||
| @override | ||||
| String toString() { | ||||
|   return 'UniversalFile(data: $data, type: $type)'; | ||||
|   return 'UniversalFile(data: $data, type: $type, isLink: $isLink)'; | ||||
| } | ||||
|  | ||||
|  | ||||
| @@ -48,7 +48,7 @@ abstract mixin class $UniversalFileCopyWith<$Res>  { | ||||
|   factory $UniversalFileCopyWith(UniversalFile value, $Res Function(UniversalFile) _then) = _$UniversalFileCopyWithImpl; | ||||
| @useResult | ||||
| $Res call({ | ||||
|  dynamic data, UniversalFileType type | ||||
|  dynamic data, UniversalFileType type, bool isLink | ||||
| }); | ||||
|  | ||||
|  | ||||
| @@ -65,11 +65,12 @@ class _$UniversalFileCopyWithImpl<$Res> | ||||
|  | ||||
| /// Create a copy of UniversalFile | ||||
| /// with the given fields replaced by the non-null parameter values. | ||||
| @pragma('vm:prefer-inline') @override $Res call({Object? data = freezed,Object? type = null,}) { | ||||
| @pragma('vm:prefer-inline') @override $Res call({Object? data = freezed,Object? type = null,Object? isLink = null,}) { | ||||
|   return _then(_self.copyWith( | ||||
| data: freezed == data ? _self.data : data // ignore: cast_nullable_to_non_nullable | ||||
| as dynamic,type: null == type ? _self.type : type // ignore: cast_nullable_to_non_nullable | ||||
| as UniversalFileType, | ||||
| as UniversalFileType,isLink: null == isLink ? _self.isLink : isLink // ignore: cast_nullable_to_non_nullable | ||||
| as bool, | ||||
|   )); | ||||
| } | ||||
|  | ||||
| @@ -151,10 +152,10 @@ return $default(_that);case _: | ||||
| /// } | ||||
| /// ``` | ||||
|  | ||||
| @optionalTypeArgs TResult maybeWhen<TResult extends Object?>(TResult Function( dynamic data,  UniversalFileType type)?  $default,{required TResult orElse(),}) {final _that = this; | ||||
| @optionalTypeArgs TResult maybeWhen<TResult extends Object?>(TResult Function( dynamic data,  UniversalFileType type,  bool isLink)?  $default,{required TResult orElse(),}) {final _that = this; | ||||
| switch (_that) { | ||||
| case _UniversalFile() when $default != null: | ||||
| return $default(_that.data,_that.type);case _: | ||||
| return $default(_that.data,_that.type,_that.isLink);case _: | ||||
|   return orElse(); | ||||
|  | ||||
| } | ||||
| @@ -172,10 +173,10 @@ return $default(_that.data,_that.type);case _: | ||||
| /// } | ||||
| /// ``` | ||||
|  | ||||
| @optionalTypeArgs TResult when<TResult extends Object?>(TResult Function( dynamic data,  UniversalFileType type)  $default,) {final _that = this; | ||||
| @optionalTypeArgs TResult when<TResult extends Object?>(TResult Function( dynamic data,  UniversalFileType type,  bool isLink)  $default,) {final _that = this; | ||||
| switch (_that) { | ||||
| case _UniversalFile(): | ||||
| return $default(_that.data,_that.type);} | ||||
| return $default(_that.data,_that.type,_that.isLink);} | ||||
| } | ||||
| /// A variant of `when` that fallback to returning `null` | ||||
| /// | ||||
| @@ -189,10 +190,10 @@ return $default(_that.data,_that.type);} | ||||
| /// } | ||||
| /// ``` | ||||
|  | ||||
| @optionalTypeArgs TResult? whenOrNull<TResult extends Object?>(TResult? Function( dynamic data,  UniversalFileType type)?  $default,) {final _that = this; | ||||
| @optionalTypeArgs TResult? whenOrNull<TResult extends Object?>(TResult? Function( dynamic data,  UniversalFileType type,  bool isLink)?  $default,) {final _that = this; | ||||
| switch (_that) { | ||||
| case _UniversalFile() when $default != null: | ||||
| return $default(_that.data,_that.type);case _: | ||||
| return $default(_that.data,_that.type,_that.isLink);case _: | ||||
|   return null; | ||||
|  | ||||
| } | ||||
| @@ -204,11 +205,12 @@ return $default(_that.data,_that.type);case _: | ||||
| @JsonSerializable() | ||||
|  | ||||
| class _UniversalFile extends UniversalFile { | ||||
|   const _UniversalFile({required this.data, required this.type}): super._(); | ||||
|   const _UniversalFile({required this.data, required this.type, this.isLink = false}): super._(); | ||||
|   factory _UniversalFile.fromJson(Map<String, dynamic> json) => _$UniversalFileFromJson(json); | ||||
|  | ||||
| @override final  dynamic data; | ||||
| @override final  UniversalFileType type; | ||||
| @override@JsonKey() final  bool isLink; | ||||
|  | ||||
| /// Create a copy of UniversalFile | ||||
| /// with the given fields replaced by the non-null parameter values. | ||||
| @@ -223,16 +225,16 @@ Map<String, dynamic> toJson() { | ||||
|  | ||||
| @override | ||||
| bool operator ==(Object other) { | ||||
|   return identical(this, other) || (other.runtimeType == runtimeType&&other is _UniversalFile&&const DeepCollectionEquality().equals(other.data, data)&&(identical(other.type, type) || other.type == type)); | ||||
|   return identical(this, other) || (other.runtimeType == runtimeType&&other is _UniversalFile&&const DeepCollectionEquality().equals(other.data, data)&&(identical(other.type, type) || other.type == type)&&(identical(other.isLink, isLink) || other.isLink == isLink)); | ||||
| } | ||||
|  | ||||
| @JsonKey(includeFromJson: false, includeToJson: false) | ||||
| @override | ||||
| int get hashCode => Object.hash(runtimeType,const DeepCollectionEquality().hash(data),type); | ||||
| int get hashCode => Object.hash(runtimeType,const DeepCollectionEquality().hash(data),type,isLink); | ||||
|  | ||||
| @override | ||||
| String toString() { | ||||
|   return 'UniversalFile(data: $data, type: $type)'; | ||||
|   return 'UniversalFile(data: $data, type: $type, isLink: $isLink)'; | ||||
| } | ||||
|  | ||||
|  | ||||
| @@ -243,7 +245,7 @@ abstract mixin class _$UniversalFileCopyWith<$Res> implements $UniversalFileCopy | ||||
|   factory _$UniversalFileCopyWith(_UniversalFile value, $Res Function(_UniversalFile) _then) = __$UniversalFileCopyWithImpl; | ||||
| @override @useResult | ||||
| $Res call({ | ||||
|  dynamic data, UniversalFileType type | ||||
|  dynamic data, UniversalFileType type, bool isLink | ||||
| }); | ||||
|  | ||||
|  | ||||
| @@ -260,11 +262,12 @@ class __$UniversalFileCopyWithImpl<$Res> | ||||
|  | ||||
| /// Create a copy of UniversalFile | ||||
| /// with the given fields replaced by the non-null parameter values. | ||||
| @override @pragma('vm:prefer-inline') $Res call({Object? data = freezed,Object? type = null,}) { | ||||
| @override @pragma('vm:prefer-inline') $Res call({Object? data = freezed,Object? type = null,Object? isLink = null,}) { | ||||
|   return _then(_UniversalFile( | ||||
| data: freezed == data ? _self.data : data // ignore: cast_nullable_to_non_nullable | ||||
| as dynamic,type: null == type ? _self.type : type // ignore: cast_nullable_to_non_nullable | ||||
| as UniversalFileType, | ||||
| as UniversalFileType,isLink: null == isLink ? _self.isLink : isLink // ignore: cast_nullable_to_non_nullable | ||||
| as bool, | ||||
|   )); | ||||
| } | ||||
|  | ||||
| @@ -275,7 +278,7 @@ as UniversalFileType, | ||||
| /// @nodoc | ||||
| mixin _$SnCloudFile { | ||||
|  | ||||
|  String get id; String get name; String? get description; Map<String, dynamic>? get fileMeta; Map<String, dynamic>? get userMeta; String? get mimeType; String? get hash; int get size; DateTime? get uploadedAt; String? get uploadedTo; DateTime get createdAt; DateTime get updatedAt; DateTime? get deletedAt; | ||||
|  String get id; String get name; String? get description; Map<String, dynamic>? get fileMeta; Map<String, dynamic>? get userMeta; List<int> get sensitiveMarks; String? get mimeType; String? get hash; int get size; DateTime? get uploadedAt; String? get uploadedTo; DateTime get createdAt; DateTime get updatedAt; DateTime? get deletedAt; | ||||
| /// Create a copy of SnCloudFile | ||||
| /// with the given fields replaced by the non-null parameter values. | ||||
| @JsonKey(includeFromJson: false, includeToJson: false) | ||||
| @@ -288,16 +291,16 @@ $SnCloudFileCopyWith<SnCloudFile> get copyWith => _$SnCloudFileCopyWithImpl<SnCl | ||||
|  | ||||
| @override | ||||
| bool operator ==(Object other) { | ||||
|   return identical(this, other) || (other.runtimeType == runtimeType&&other is SnCloudFile&&(identical(other.id, id) || other.id == id)&&(identical(other.name, name) || other.name == name)&&(identical(other.description, description) || other.description == description)&&const DeepCollectionEquality().equals(other.fileMeta, fileMeta)&&const DeepCollectionEquality().equals(other.userMeta, userMeta)&&(identical(other.mimeType, mimeType) || other.mimeType == mimeType)&&(identical(other.hash, hash) || other.hash == hash)&&(identical(other.size, size) || other.size == size)&&(identical(other.uploadedAt, uploadedAt) || other.uploadedAt == uploadedAt)&&(identical(other.uploadedTo, uploadedTo) || other.uploadedTo == uploadedTo)&&(identical(other.createdAt, createdAt) || other.createdAt == createdAt)&&(identical(other.updatedAt, updatedAt) || other.updatedAt == updatedAt)&&(identical(other.deletedAt, deletedAt) || other.deletedAt == deletedAt)); | ||||
|   return identical(this, other) || (other.runtimeType == runtimeType&&other is SnCloudFile&&(identical(other.id, id) || other.id == id)&&(identical(other.name, name) || other.name == name)&&(identical(other.description, description) || other.description == description)&&const DeepCollectionEquality().equals(other.fileMeta, fileMeta)&&const DeepCollectionEquality().equals(other.userMeta, userMeta)&&const DeepCollectionEquality().equals(other.sensitiveMarks, sensitiveMarks)&&(identical(other.mimeType, mimeType) || other.mimeType == mimeType)&&(identical(other.hash, hash) || other.hash == hash)&&(identical(other.size, size) || other.size == size)&&(identical(other.uploadedAt, uploadedAt) || other.uploadedAt == uploadedAt)&&(identical(other.uploadedTo, uploadedTo) || other.uploadedTo == uploadedTo)&&(identical(other.createdAt, createdAt) || other.createdAt == createdAt)&&(identical(other.updatedAt, updatedAt) || other.updatedAt == updatedAt)&&(identical(other.deletedAt, deletedAt) || other.deletedAt == deletedAt)); | ||||
| } | ||||
|  | ||||
| @JsonKey(includeFromJson: false, includeToJson: false) | ||||
| @override | ||||
| int get hashCode => Object.hash(runtimeType,id,name,description,const DeepCollectionEquality().hash(fileMeta),const DeepCollectionEquality().hash(userMeta),mimeType,hash,size,uploadedAt,uploadedTo,createdAt,updatedAt,deletedAt); | ||||
| int get hashCode => Object.hash(runtimeType,id,name,description,const DeepCollectionEquality().hash(fileMeta),const DeepCollectionEquality().hash(userMeta),const DeepCollectionEquality().hash(sensitiveMarks),mimeType,hash,size,uploadedAt,uploadedTo,createdAt,updatedAt,deletedAt); | ||||
|  | ||||
| @override | ||||
| String toString() { | ||||
|   return 'SnCloudFile(id: $id, name: $name, description: $description, fileMeta: $fileMeta, userMeta: $userMeta, mimeType: $mimeType, hash: $hash, size: $size, uploadedAt: $uploadedAt, uploadedTo: $uploadedTo, createdAt: $createdAt, updatedAt: $updatedAt, deletedAt: $deletedAt)'; | ||||
|   return 'SnCloudFile(id: $id, name: $name, description: $description, fileMeta: $fileMeta, userMeta: $userMeta, sensitiveMarks: $sensitiveMarks, mimeType: $mimeType, hash: $hash, size: $size, uploadedAt: $uploadedAt, uploadedTo: $uploadedTo, createdAt: $createdAt, updatedAt: $updatedAt, deletedAt: $deletedAt)'; | ||||
| } | ||||
|  | ||||
|  | ||||
| @@ -308,7 +311,7 @@ abstract mixin class $SnCloudFileCopyWith<$Res>  { | ||||
|   factory $SnCloudFileCopyWith(SnCloudFile value, $Res Function(SnCloudFile) _then) = _$SnCloudFileCopyWithImpl; | ||||
| @useResult | ||||
| $Res call({ | ||||
|  String id, String name, String? description, Map<String, dynamic>? fileMeta, Map<String, dynamic>? userMeta, String? mimeType, String? hash, int size, DateTime? uploadedAt, String? uploadedTo, DateTime createdAt, DateTime updatedAt, DateTime? deletedAt | ||||
|  String id, String name, String? description, Map<String, dynamic>? fileMeta, Map<String, dynamic>? userMeta, List<int> sensitiveMarks, String? mimeType, String? hash, int size, DateTime? uploadedAt, String? uploadedTo, DateTime createdAt, DateTime updatedAt, DateTime? deletedAt | ||||
| }); | ||||
|  | ||||
|  | ||||
| @@ -325,14 +328,15 @@ class _$SnCloudFileCopyWithImpl<$Res> | ||||
|  | ||||
| /// Create a copy of SnCloudFile | ||||
| /// with the given fields replaced by the non-null parameter values. | ||||
| @pragma('vm:prefer-inline') @override $Res call({Object? id = null,Object? name = null,Object? description = freezed,Object? fileMeta = freezed,Object? userMeta = freezed,Object? mimeType = freezed,Object? hash = freezed,Object? size = null,Object? uploadedAt = freezed,Object? uploadedTo = freezed,Object? createdAt = null,Object? updatedAt = null,Object? deletedAt = freezed,}) { | ||||
| @pragma('vm:prefer-inline') @override $Res call({Object? id = null,Object? name = null,Object? description = freezed,Object? fileMeta = freezed,Object? userMeta = freezed,Object? sensitiveMarks = null,Object? mimeType = freezed,Object? hash = freezed,Object? size = null,Object? uploadedAt = freezed,Object? uploadedTo = freezed,Object? createdAt = null,Object? updatedAt = null,Object? deletedAt = freezed,}) { | ||||
|   return _then(_self.copyWith( | ||||
| id: null == id ? _self.id : id // ignore: cast_nullable_to_non_nullable | ||||
| as String,name: null == name ? _self.name : name // ignore: cast_nullable_to_non_nullable | ||||
| as String,description: freezed == description ? _self.description : description // ignore: cast_nullable_to_non_nullable | ||||
| as String?,fileMeta: freezed == fileMeta ? _self.fileMeta : fileMeta // ignore: cast_nullable_to_non_nullable | ||||
| as Map<String, dynamic>?,userMeta: freezed == userMeta ? _self.userMeta : userMeta // ignore: cast_nullable_to_non_nullable | ||||
| as Map<String, dynamic>?,mimeType: freezed == mimeType ? _self.mimeType : mimeType // ignore: cast_nullable_to_non_nullable | ||||
| as Map<String, dynamic>?,sensitiveMarks: null == sensitiveMarks ? _self.sensitiveMarks : sensitiveMarks // ignore: cast_nullable_to_non_nullable | ||||
| as List<int>,mimeType: freezed == mimeType ? _self.mimeType : mimeType // ignore: cast_nullable_to_non_nullable | ||||
| as String?,hash: freezed == hash ? _self.hash : hash // ignore: cast_nullable_to_non_nullable | ||||
| as String?,size: null == size ? _self.size : size // ignore: cast_nullable_to_non_nullable | ||||
| as int,uploadedAt: freezed == uploadedAt ? _self.uploadedAt : uploadedAt // ignore: cast_nullable_to_non_nullable | ||||
| @@ -422,10 +426,10 @@ return $default(_that);case _: | ||||
| /// } | ||||
| /// ``` | ||||
|  | ||||
| @optionalTypeArgs TResult maybeWhen<TResult extends Object?>(TResult Function( String id,  String name,  String? description,  Map<String, dynamic>? fileMeta,  Map<String, dynamic>? userMeta,  String? mimeType,  String? hash,  int size,  DateTime? uploadedAt,  String? uploadedTo,  DateTime createdAt,  DateTime updatedAt,  DateTime? deletedAt)?  $default,{required TResult orElse(),}) {final _that = this; | ||||
| @optionalTypeArgs TResult maybeWhen<TResult extends Object?>(TResult Function( String id,  String name,  String? description,  Map<String, dynamic>? fileMeta,  Map<String, dynamic>? userMeta,  List<int> sensitiveMarks,  String? mimeType,  String? hash,  int size,  DateTime? uploadedAt,  String? uploadedTo,  DateTime createdAt,  DateTime updatedAt,  DateTime? deletedAt)?  $default,{required TResult orElse(),}) {final _that = this; | ||||
| switch (_that) { | ||||
| case _SnCloudFile() when $default != null: | ||||
| return $default(_that.id,_that.name,_that.description,_that.fileMeta,_that.userMeta,_that.mimeType,_that.hash,_that.size,_that.uploadedAt,_that.uploadedTo,_that.createdAt,_that.updatedAt,_that.deletedAt);case _: | ||||
| return $default(_that.id,_that.name,_that.description,_that.fileMeta,_that.userMeta,_that.sensitiveMarks,_that.mimeType,_that.hash,_that.size,_that.uploadedAt,_that.uploadedTo,_that.createdAt,_that.updatedAt,_that.deletedAt);case _: | ||||
|   return orElse(); | ||||
|  | ||||
| } | ||||
| @@ -443,10 +447,10 @@ return $default(_that.id,_that.name,_that.description,_that.fileMeta,_that.userM | ||||
| /// } | ||||
| /// ``` | ||||
|  | ||||
| @optionalTypeArgs TResult when<TResult extends Object?>(TResult Function( String id,  String name,  String? description,  Map<String, dynamic>? fileMeta,  Map<String, dynamic>? userMeta,  String? mimeType,  String? hash,  int size,  DateTime? uploadedAt,  String? uploadedTo,  DateTime createdAt,  DateTime updatedAt,  DateTime? deletedAt)  $default,) {final _that = this; | ||||
| @optionalTypeArgs TResult when<TResult extends Object?>(TResult Function( String id,  String name,  String? description,  Map<String, dynamic>? fileMeta,  Map<String, dynamic>? userMeta,  List<int> sensitiveMarks,  String? mimeType,  String? hash,  int size,  DateTime? uploadedAt,  String? uploadedTo,  DateTime createdAt,  DateTime updatedAt,  DateTime? deletedAt)  $default,) {final _that = this; | ||||
| switch (_that) { | ||||
| case _SnCloudFile(): | ||||
| return $default(_that.id,_that.name,_that.description,_that.fileMeta,_that.userMeta,_that.mimeType,_that.hash,_that.size,_that.uploadedAt,_that.uploadedTo,_that.createdAt,_that.updatedAt,_that.deletedAt);} | ||||
| return $default(_that.id,_that.name,_that.description,_that.fileMeta,_that.userMeta,_that.sensitiveMarks,_that.mimeType,_that.hash,_that.size,_that.uploadedAt,_that.uploadedTo,_that.createdAt,_that.updatedAt,_that.deletedAt);} | ||||
| } | ||||
| /// A variant of `when` that fallback to returning `null` | ||||
| /// | ||||
| @@ -460,10 +464,10 @@ return $default(_that.id,_that.name,_that.description,_that.fileMeta,_that.userM | ||||
| /// } | ||||
| /// ``` | ||||
|  | ||||
| @optionalTypeArgs TResult? whenOrNull<TResult extends Object?>(TResult? Function( String id,  String name,  String? description,  Map<String, dynamic>? fileMeta,  Map<String, dynamic>? userMeta,  String? mimeType,  String? hash,  int size,  DateTime? uploadedAt,  String? uploadedTo,  DateTime createdAt,  DateTime updatedAt,  DateTime? deletedAt)?  $default,) {final _that = this; | ||||
| @optionalTypeArgs TResult? whenOrNull<TResult extends Object?>(TResult? Function( String id,  String name,  String? description,  Map<String, dynamic>? fileMeta,  Map<String, dynamic>? userMeta,  List<int> sensitiveMarks,  String? mimeType,  String? hash,  int size,  DateTime? uploadedAt,  String? uploadedTo,  DateTime createdAt,  DateTime updatedAt,  DateTime? deletedAt)?  $default,) {final _that = this; | ||||
| switch (_that) { | ||||
| case _SnCloudFile() when $default != null: | ||||
| return $default(_that.id,_that.name,_that.description,_that.fileMeta,_that.userMeta,_that.mimeType,_that.hash,_that.size,_that.uploadedAt,_that.uploadedTo,_that.createdAt,_that.updatedAt,_that.deletedAt);case _: | ||||
| return $default(_that.id,_that.name,_that.description,_that.fileMeta,_that.userMeta,_that.sensitiveMarks,_that.mimeType,_that.hash,_that.size,_that.uploadedAt,_that.uploadedTo,_that.createdAt,_that.updatedAt,_that.deletedAt);case _: | ||||
|   return null; | ||||
|  | ||||
| } | ||||
| @@ -475,7 +479,7 @@ return $default(_that.id,_that.name,_that.description,_that.fileMeta,_that.userM | ||||
| @JsonSerializable() | ||||
|  | ||||
| class _SnCloudFile implements SnCloudFile { | ||||
|   const _SnCloudFile({required this.id, required this.name, required this.description, required final  Map<String, dynamic>? fileMeta, required final  Map<String, dynamic>? userMeta, required this.mimeType, required this.hash, required this.size, required this.uploadedAt, required this.uploadedTo, required this.createdAt, required this.updatedAt, required this.deletedAt}): _fileMeta = fileMeta,_userMeta = userMeta; | ||||
|   const _SnCloudFile({required this.id, required this.name, required this.description, required final  Map<String, dynamic>? fileMeta, required final  Map<String, dynamic>? userMeta, final  List<int> sensitiveMarks = const [], required this.mimeType, required this.hash, required this.size, required this.uploadedAt, required this.uploadedTo, required this.createdAt, required this.updatedAt, required this.deletedAt}): _fileMeta = fileMeta,_userMeta = userMeta,_sensitiveMarks = sensitiveMarks; | ||||
|   factory _SnCloudFile.fromJson(Map<String, dynamic> json) => _$SnCloudFileFromJson(json); | ||||
|  | ||||
| @override final  String id; | ||||
| @@ -499,6 +503,13 @@ class _SnCloudFile implements SnCloudFile { | ||||
|   return EqualUnmodifiableMapView(value); | ||||
| } | ||||
|  | ||||
|  final  List<int> _sensitiveMarks; | ||||
| @override@JsonKey() List<int> get sensitiveMarks { | ||||
|   if (_sensitiveMarks is EqualUnmodifiableListView) return _sensitiveMarks; | ||||
|   // ignore: implicit_dynamic_type | ||||
|   return EqualUnmodifiableListView(_sensitiveMarks); | ||||
| } | ||||
|  | ||||
| @override final  String? mimeType; | ||||
| @override final  String? hash; | ||||
| @override final  int size; | ||||
| @@ -521,16 +532,16 @@ Map<String, dynamic> toJson() { | ||||
|  | ||||
| @override | ||||
| bool operator ==(Object other) { | ||||
|   return identical(this, other) || (other.runtimeType == runtimeType&&other is _SnCloudFile&&(identical(other.id, id) || other.id == id)&&(identical(other.name, name) || other.name == name)&&(identical(other.description, description) || other.description == description)&&const DeepCollectionEquality().equals(other._fileMeta, _fileMeta)&&const DeepCollectionEquality().equals(other._userMeta, _userMeta)&&(identical(other.mimeType, mimeType) || other.mimeType == mimeType)&&(identical(other.hash, hash) || other.hash == hash)&&(identical(other.size, size) || other.size == size)&&(identical(other.uploadedAt, uploadedAt) || other.uploadedAt == uploadedAt)&&(identical(other.uploadedTo, uploadedTo) || other.uploadedTo == uploadedTo)&&(identical(other.createdAt, createdAt) || other.createdAt == createdAt)&&(identical(other.updatedAt, updatedAt) || other.updatedAt == updatedAt)&&(identical(other.deletedAt, deletedAt) || other.deletedAt == deletedAt)); | ||||
|   return identical(this, other) || (other.runtimeType == runtimeType&&other is _SnCloudFile&&(identical(other.id, id) || other.id == id)&&(identical(other.name, name) || other.name == name)&&(identical(other.description, description) || other.description == description)&&const DeepCollectionEquality().equals(other._fileMeta, _fileMeta)&&const DeepCollectionEquality().equals(other._userMeta, _userMeta)&&const DeepCollectionEquality().equals(other._sensitiveMarks, _sensitiveMarks)&&(identical(other.mimeType, mimeType) || other.mimeType == mimeType)&&(identical(other.hash, hash) || other.hash == hash)&&(identical(other.size, size) || other.size == size)&&(identical(other.uploadedAt, uploadedAt) || other.uploadedAt == uploadedAt)&&(identical(other.uploadedTo, uploadedTo) || other.uploadedTo == uploadedTo)&&(identical(other.createdAt, createdAt) || other.createdAt == createdAt)&&(identical(other.updatedAt, updatedAt) || other.updatedAt == updatedAt)&&(identical(other.deletedAt, deletedAt) || other.deletedAt == deletedAt)); | ||||
| } | ||||
|  | ||||
| @JsonKey(includeFromJson: false, includeToJson: false) | ||||
| @override | ||||
| int get hashCode => Object.hash(runtimeType,id,name,description,const DeepCollectionEquality().hash(_fileMeta),const DeepCollectionEquality().hash(_userMeta),mimeType,hash,size,uploadedAt,uploadedTo,createdAt,updatedAt,deletedAt); | ||||
| int get hashCode => Object.hash(runtimeType,id,name,description,const DeepCollectionEquality().hash(_fileMeta),const DeepCollectionEquality().hash(_userMeta),const DeepCollectionEquality().hash(_sensitiveMarks),mimeType,hash,size,uploadedAt,uploadedTo,createdAt,updatedAt,deletedAt); | ||||
|  | ||||
| @override | ||||
| String toString() { | ||||
|   return 'SnCloudFile(id: $id, name: $name, description: $description, fileMeta: $fileMeta, userMeta: $userMeta, mimeType: $mimeType, hash: $hash, size: $size, uploadedAt: $uploadedAt, uploadedTo: $uploadedTo, createdAt: $createdAt, updatedAt: $updatedAt, deletedAt: $deletedAt)'; | ||||
|   return 'SnCloudFile(id: $id, name: $name, description: $description, fileMeta: $fileMeta, userMeta: $userMeta, sensitiveMarks: $sensitiveMarks, mimeType: $mimeType, hash: $hash, size: $size, uploadedAt: $uploadedAt, uploadedTo: $uploadedTo, createdAt: $createdAt, updatedAt: $updatedAt, deletedAt: $deletedAt)'; | ||||
| } | ||||
|  | ||||
|  | ||||
| @@ -541,7 +552,7 @@ abstract mixin class _$SnCloudFileCopyWith<$Res> implements $SnCloudFileCopyWith | ||||
|   factory _$SnCloudFileCopyWith(_SnCloudFile value, $Res Function(_SnCloudFile) _then) = __$SnCloudFileCopyWithImpl; | ||||
| @override @useResult | ||||
| $Res call({ | ||||
|  String id, String name, String? description, Map<String, dynamic>? fileMeta, Map<String, dynamic>? userMeta, String? mimeType, String? hash, int size, DateTime? uploadedAt, String? uploadedTo, DateTime createdAt, DateTime updatedAt, DateTime? deletedAt | ||||
|  String id, String name, String? description, Map<String, dynamic>? fileMeta, Map<String, dynamic>? userMeta, List<int> sensitiveMarks, String? mimeType, String? hash, int size, DateTime? uploadedAt, String? uploadedTo, DateTime createdAt, DateTime updatedAt, DateTime? deletedAt | ||||
| }); | ||||
|  | ||||
|  | ||||
| @@ -558,14 +569,15 @@ class __$SnCloudFileCopyWithImpl<$Res> | ||||
|  | ||||
| /// Create a copy of SnCloudFile | ||||
| /// with the given fields replaced by the non-null parameter values. | ||||
| @override @pragma('vm:prefer-inline') $Res call({Object? id = null,Object? name = null,Object? description = freezed,Object? fileMeta = freezed,Object? userMeta = freezed,Object? mimeType = freezed,Object? hash = freezed,Object? size = null,Object? uploadedAt = freezed,Object? uploadedTo = freezed,Object? createdAt = null,Object? updatedAt = null,Object? deletedAt = freezed,}) { | ||||
| @override @pragma('vm:prefer-inline') $Res call({Object? id = null,Object? name = null,Object? description = freezed,Object? fileMeta = freezed,Object? userMeta = freezed,Object? sensitiveMarks = null,Object? mimeType = freezed,Object? hash = freezed,Object? size = null,Object? uploadedAt = freezed,Object? uploadedTo = freezed,Object? createdAt = null,Object? updatedAt = null,Object? deletedAt = freezed,}) { | ||||
|   return _then(_SnCloudFile( | ||||
| id: null == id ? _self.id : id // ignore: cast_nullable_to_non_nullable | ||||
| as String,name: null == name ? _self.name : name // ignore: cast_nullable_to_non_nullable | ||||
| as String,description: freezed == description ? _self.description : description // ignore: cast_nullable_to_non_nullable | ||||
| as String?,fileMeta: freezed == fileMeta ? _self._fileMeta : fileMeta // ignore: cast_nullable_to_non_nullable | ||||
| as Map<String, dynamic>?,userMeta: freezed == userMeta ? _self._userMeta : userMeta // ignore: cast_nullable_to_non_nullable | ||||
| as Map<String, dynamic>?,mimeType: freezed == mimeType ? _self.mimeType : mimeType // ignore: cast_nullable_to_non_nullable | ||||
| as Map<String, dynamic>?,sensitiveMarks: null == sensitiveMarks ? _self._sensitiveMarks : sensitiveMarks // ignore: cast_nullable_to_non_nullable | ||||
| as List<int>,mimeType: freezed == mimeType ? _self.mimeType : mimeType // ignore: cast_nullable_to_non_nullable | ||||
| as String?,hash: freezed == hash ? _self.hash : hash // ignore: cast_nullable_to_non_nullable | ||||
| as String?,size: null == size ? _self.size : size // ignore: cast_nullable_to_non_nullable | ||||
| as int,uploadedAt: freezed == uploadedAt ? _self.uploadedAt : uploadedAt // ignore: cast_nullable_to_non_nullable | ||||
|   | ||||
| @@ -10,12 +10,14 @@ _UniversalFile _$UniversalFileFromJson(Map<String, dynamic> json) => | ||||
|     _UniversalFile( | ||||
|       data: json['data'], | ||||
|       type: $enumDecode(_$UniversalFileTypeEnumMap, json['type']), | ||||
|       isLink: json['is_link'] as bool? ?? false, | ||||
|     ); | ||||
|  | ||||
| Map<String, dynamic> _$UniversalFileToJson(_UniversalFile instance) => | ||||
|     <String, dynamic>{ | ||||
|       'data': instance.data, | ||||
|       'type': _$UniversalFileTypeEnumMap[instance.type]!, | ||||
|       'is_link': instance.isLink, | ||||
|     }; | ||||
|  | ||||
| const _$UniversalFileTypeEnumMap = { | ||||
| @@ -31,6 +33,11 @@ _SnCloudFile _$SnCloudFileFromJson(Map<String, dynamic> json) => _SnCloudFile( | ||||
|   description: json['description'] as String?, | ||||
|   fileMeta: json['file_meta'] as Map<String, dynamic>?, | ||||
|   userMeta: json['user_meta'] as Map<String, dynamic>?, | ||||
|   sensitiveMarks: | ||||
|       (json['sensitive_marks'] as List<dynamic>?) | ||||
|           ?.map((e) => (e as num).toInt()) | ||||
|           .toList() ?? | ||||
|       const [], | ||||
|   mimeType: json['mime_type'] as String?, | ||||
|   hash: json['hash'] as String?, | ||||
|   size: (json['size'] as num).toInt(), | ||||
| @@ -54,6 +61,7 @@ Map<String, dynamic> _$SnCloudFileToJson(_SnCloudFile instance) => | ||||
|       'description': instance.description, | ||||
|       'file_meta': instance.fileMeta, | ||||
|       'user_meta': instance.userMeta, | ||||
|       'sensitive_marks': instance.sensitiveMarks, | ||||
|       'mime_type': instance.mimeType, | ||||
|       'hash': instance.hash, | ||||
|       'size': instance.size, | ||||
|   | ||||
							
								
								
									
										92
									
								
								lib/models/poll.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										92
									
								
								lib/models/poll.dart
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,92 @@ | ||||
| import 'package:freezed_annotation/freezed_annotation.dart'; | ||||
| import 'package:island/models/publisher.dart'; | ||||
|  | ||||
| part 'poll.freezed.dart'; | ||||
| part 'poll.g.dart'; | ||||
|  | ||||
| @freezed | ||||
| sealed class SnPollWithStats with _$SnPollWithStats { | ||||
|   const factory SnPollWithStats({ | ||||
|     required Map<String, dynamic>? userAnswer, | ||||
|     required Map<String, dynamic> stats, | ||||
|     required String id, | ||||
|     required List<SnPollQuestion> questions, | ||||
|     String? title, | ||||
|     String? description, | ||||
|     DateTime? endedAt, | ||||
|     required String publisherId, | ||||
|     required DateTime createdAt, | ||||
|     required DateTime updatedAt, | ||||
|     DateTime? deletedAt, | ||||
|   }) = _SnPollWithStats; | ||||
|  | ||||
|   factory SnPollWithStats.fromJson(Map<String, dynamic> json) => | ||||
|       _$SnPollWithStatsFromJson(json); | ||||
| } | ||||
|  | ||||
| @freezed | ||||
| sealed class SnPoll with _$SnPoll { | ||||
|   const factory SnPoll({ | ||||
|     required String id, | ||||
|     required List<SnPollQuestion> questions, | ||||
|  | ||||
|     String? title, | ||||
|     String? description, | ||||
|  | ||||
|     DateTime? endedAt, | ||||
|  | ||||
|     required String publisherId, | ||||
|     SnPublisher? publisher, | ||||
|  | ||||
|     // ModelBase fields | ||||
|     required DateTime createdAt, | ||||
|     required DateTime updatedAt, | ||||
|     DateTime? deletedAt, | ||||
|   }) = _SnPoll; | ||||
|  | ||||
|   factory SnPoll.fromJson(Map<String, dynamic> json) => _$SnPollFromJson(json); | ||||
| } | ||||
|  | ||||
| @freezed | ||||
| sealed class SnPollQuestion with _$SnPollQuestion { | ||||
|   const factory SnPollQuestion({ | ||||
|     required String id, | ||||
|  | ||||
|     required SnPollQuestionType type, | ||||
|     List<SnPollOption>? options, | ||||
|  | ||||
|     required String title, | ||||
|     String? description, | ||||
|     required int order, | ||||
|     required bool isRequired, | ||||
|   }) = _SnPollQuestion; | ||||
|  | ||||
|   factory SnPollQuestion.fromJson(Map<String, dynamic> json) => | ||||
|       _$SnPollQuestionFromJson(json); | ||||
| } | ||||
|  | ||||
| @freezed | ||||
| sealed class SnPollOption with _$SnPollOption { | ||||
|   const factory SnPollOption({ | ||||
|     required String id, | ||||
|     required String label, | ||||
|     String? description, | ||||
|     required int order, | ||||
|   }) = _SnPollOption; | ||||
|  | ||||
|   factory SnPollOption.fromJson(Map<String, dynamic> json) => | ||||
|       _$SnPollOptionFromJson(json); | ||||
| } | ||||
|  | ||||
| enum SnPollQuestionType { | ||||
|   @JsonValue(0) | ||||
|   singleChoice, | ||||
|   @JsonValue(1) | ||||
|   multipleChoice, | ||||
|   @JsonValue(2) | ||||
|   yesNo, | ||||
|   @JsonValue(3) | ||||
|   rating, | ||||
|   @JsonValue(4) | ||||
|   freeText, | ||||
| } | ||||
							
								
								
									
										1186
									
								
								lib/models/poll.freezed.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1186
									
								
								lib/models/poll.freezed.dart
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										133
									
								
								lib/models/poll.g.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										133
									
								
								lib/models/poll.g.dart
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,133 @@ | ||||
| // GENERATED CODE - DO NOT MODIFY BY HAND | ||||
|  | ||||
| part of 'poll.dart'; | ||||
|  | ||||
| // ************************************************************************** | ||||
| // JsonSerializableGenerator | ||||
| // ************************************************************************** | ||||
|  | ||||
| _SnPollWithStats _$SnPollWithStatsFromJson(Map<String, dynamic> json) => | ||||
|     _SnPollWithStats( | ||||
|       userAnswer: json['user_answer'] as Map<String, dynamic>?, | ||||
|       stats: json['stats'] as Map<String, dynamic>, | ||||
|       id: json['id'] as String, | ||||
|       questions: | ||||
|           (json['questions'] as List<dynamic>) | ||||
|               .map((e) => SnPollQuestion.fromJson(e as Map<String, dynamic>)) | ||||
|               .toList(), | ||||
|       title: json['title'] as String?, | ||||
|       description: json['description'] as String?, | ||||
|       endedAt: | ||||
|           json['ended_at'] == null | ||||
|               ? null | ||||
|               : DateTime.parse(json['ended_at'] as String), | ||||
|       publisherId: json['publisher_id'] as String, | ||||
|       createdAt: DateTime.parse(json['created_at'] as String), | ||||
|       updatedAt: DateTime.parse(json['updated_at'] as String), | ||||
|       deletedAt: | ||||
|           json['deleted_at'] == null | ||||
|               ? null | ||||
|               : DateTime.parse(json['deleted_at'] as String), | ||||
|     ); | ||||
|  | ||||
| Map<String, dynamic> _$SnPollWithStatsToJson(_SnPollWithStats instance) => | ||||
|     <String, dynamic>{ | ||||
|       'user_answer': instance.userAnswer, | ||||
|       'stats': instance.stats, | ||||
|       'id': instance.id, | ||||
|       'questions': instance.questions.map((e) => e.toJson()).toList(), | ||||
|       'title': instance.title, | ||||
|       'description': instance.description, | ||||
|       'ended_at': instance.endedAt?.toIso8601String(), | ||||
|       'publisher_id': instance.publisherId, | ||||
|       'created_at': instance.createdAt.toIso8601String(), | ||||
|       'updated_at': instance.updatedAt.toIso8601String(), | ||||
|       'deleted_at': instance.deletedAt?.toIso8601String(), | ||||
|     }; | ||||
|  | ||||
| _SnPoll _$SnPollFromJson(Map<String, dynamic> json) => _SnPoll( | ||||
|   id: json['id'] as String, | ||||
|   questions: | ||||
|       (json['questions'] as List<dynamic>) | ||||
|           .map((e) => SnPollQuestion.fromJson(e as Map<String, dynamic>)) | ||||
|           .toList(), | ||||
|   title: json['title'] as String?, | ||||
|   description: json['description'] as String?, | ||||
|   endedAt: | ||||
|       json['ended_at'] == null | ||||
|           ? null | ||||
|           : DateTime.parse(json['ended_at'] as String), | ||||
|   publisherId: json['publisher_id'] as String, | ||||
|   publisher: | ||||
|       json['publisher'] == null | ||||
|           ? null | ||||
|           : SnPublisher.fromJson(json['publisher'] as Map<String, dynamic>), | ||||
|   createdAt: DateTime.parse(json['created_at'] as String), | ||||
|   updatedAt: DateTime.parse(json['updated_at'] as String), | ||||
|   deletedAt: | ||||
|       json['deleted_at'] == null | ||||
|           ? null | ||||
|           : DateTime.parse(json['deleted_at'] as String), | ||||
| ); | ||||
|  | ||||
| Map<String, dynamic> _$SnPollToJson(_SnPoll instance) => <String, dynamic>{ | ||||
|   'id': instance.id, | ||||
|   'questions': instance.questions.map((e) => e.toJson()).toList(), | ||||
|   'title': instance.title, | ||||
|   'description': instance.description, | ||||
|   'ended_at': instance.endedAt?.toIso8601String(), | ||||
|   'publisher_id': instance.publisherId, | ||||
|   'publisher': instance.publisher?.toJson(), | ||||
|   'created_at': instance.createdAt.toIso8601String(), | ||||
|   'updated_at': instance.updatedAt.toIso8601String(), | ||||
|   'deleted_at': instance.deletedAt?.toIso8601String(), | ||||
| }; | ||||
|  | ||||
| _SnPollQuestion _$SnPollQuestionFromJson(Map<String, dynamic> json) => | ||||
|     _SnPollQuestion( | ||||
|       id: json['id'] as String, | ||||
|       type: $enumDecode(_$SnPollQuestionTypeEnumMap, json['type']), | ||||
|       options: | ||||
|           (json['options'] as List<dynamic>?) | ||||
|               ?.map((e) => SnPollOption.fromJson(e as Map<String, dynamic>)) | ||||
|               .toList(), | ||||
|       title: json['title'] as String, | ||||
|       description: json['description'] as String?, | ||||
|       order: (json['order'] as num).toInt(), | ||||
|       isRequired: json['is_required'] as bool, | ||||
|     ); | ||||
|  | ||||
| Map<String, dynamic> _$SnPollQuestionToJson(_SnPollQuestion instance) => | ||||
|     <String, dynamic>{ | ||||
|       'id': instance.id, | ||||
|       'type': _$SnPollQuestionTypeEnumMap[instance.type]!, | ||||
|       'options': instance.options?.map((e) => e.toJson()).toList(), | ||||
|       'title': instance.title, | ||||
|       'description': instance.description, | ||||
|       'order': instance.order, | ||||
|       'is_required': instance.isRequired, | ||||
|     }; | ||||
|  | ||||
| const _$SnPollQuestionTypeEnumMap = { | ||||
|   SnPollQuestionType.singleChoice: 0, | ||||
|   SnPollQuestionType.multipleChoice: 1, | ||||
|   SnPollQuestionType.yesNo: 2, | ||||
|   SnPollQuestionType.rating: 3, | ||||
|   SnPollQuestionType.freeText: 4, | ||||
| }; | ||||
|  | ||||
| _SnPollOption _$SnPollOptionFromJson(Map<String, dynamic> json) => | ||||
|     _SnPollOption( | ||||
|       id: json['id'] as String, | ||||
|       label: json['label'] as String, | ||||
|       description: json['description'] as String?, | ||||
|       order: (json['order'] as num).toInt(), | ||||
|     ); | ||||
|  | ||||
| Map<String, dynamic> _$SnPollOptionToJson(_SnPollOption instance) => | ||||
|     <String, dynamic>{ | ||||
|       'id': instance.id, | ||||
|       'label': instance.label, | ||||
|       'description': instance.description, | ||||
|       'order': instance.order, | ||||
|     }; | ||||
| @@ -34,6 +34,7 @@ sealed class SnPost with _$SnPost { | ||||
|     @Default([]) List<SnCloudFile> attachments, | ||||
|     required SnPublisher publisher, | ||||
|     @Default({}) Map<String, int> reactionsCount, | ||||
|     @Default({}) Map<String, bool> reactionsMade, | ||||
|     @Default([]) List<dynamic> reactions, | ||||
|     @Default([]) List<PostTag> tags, | ||||
|     @Default([]) List<PostCategory> categories, | ||||
| @@ -77,6 +78,13 @@ sealed class SnSubscriptionStatus with _$SnSubscriptionStatus { | ||||
| sealed class ReactInfo with _$ReactInfo { | ||||
|   const factory ReactInfo({required String icon, required int attitude}) = | ||||
|       _ReactInfo; | ||||
|  | ||||
|   static String getTranslationKey(String templateKey) { | ||||
|     final parts = templateKey.split('_'); | ||||
|     final camelCase = | ||||
|         parts.map((p) => p[0].toUpperCase() + p.substring(1)).join(); | ||||
|     return 'reaction$camelCase'; | ||||
|   } | ||||
| } | ||||
|  | ||||
| const Map<String, ReactInfo> kReactionTemplates = { | ||||
|   | ||||
| @@ -15,7 +15,7 @@ T _$identity<T>(T value) => value; | ||||
| /// @nodoc | ||||
| mixin _$SnPost { | ||||
|  | ||||
|  String get id; String? get title; String? get description; String? get language; DateTime? get editedAt; DateTime? get publishedAt; int get visibility; String? get content; int get type; Map<String, dynamic>? get meta; int get viewsUnique; int get viewsTotal; int get upvotes; int get downvotes; int get repliesCount; String? get threadedPostId; SnPost? get threadedPost; String? get repliedPostId; SnPost? get repliedPost; String? get forwardedPostId; SnPost? get forwardedPost; List<SnCloudFile> get attachments; SnPublisher get publisher; Map<String, int> get reactionsCount; List<dynamic> get reactions; List<PostTag> get tags; List<PostCategory> get categories; List<dynamic> get collections; DateTime? get createdAt; DateTime? get updatedAt; DateTime? get deletedAt; bool get isTruncated; | ||||
|  String get id; String? get title; String? get description; String? get language; DateTime? get editedAt; DateTime? get publishedAt; int get visibility; String? get content; int get type; Map<String, dynamic>? get meta; int get viewsUnique; int get viewsTotal; int get upvotes; int get downvotes; int get repliesCount; String? get threadedPostId; SnPost? get threadedPost; String? get repliedPostId; SnPost? get repliedPost; String? get forwardedPostId; SnPost? get forwardedPost; List<SnCloudFile> get attachments; SnPublisher get publisher; Map<String, int> get reactionsCount; Map<String, bool> get reactionsMade; List<dynamic> get reactions; List<PostTag> get tags; List<PostCategory> get categories; List<dynamic> get collections; DateTime? get createdAt; DateTime? get updatedAt; DateTime? get deletedAt; bool get isTruncated; | ||||
| /// Create a copy of SnPost | ||||
| /// with the given fields replaced by the non-null parameter values. | ||||
| @JsonKey(includeFromJson: false, includeToJson: false) | ||||
| @@ -28,16 +28,16 @@ $SnPostCopyWith<SnPost> get copyWith => _$SnPostCopyWithImpl<SnPost>(this as SnP | ||||
|  | ||||
| @override | ||||
| bool operator ==(Object other) { | ||||
|   return identical(this, other) || (other.runtimeType == runtimeType&&other is SnPost&&(identical(other.id, id) || other.id == id)&&(identical(other.title, title) || other.title == title)&&(identical(other.description, description) || other.description == description)&&(identical(other.language, language) || other.language == language)&&(identical(other.editedAt, editedAt) || other.editedAt == editedAt)&&(identical(other.publishedAt, publishedAt) || other.publishedAt == publishedAt)&&(identical(other.visibility, visibility) || other.visibility == visibility)&&(identical(other.content, content) || other.content == content)&&(identical(other.type, type) || other.type == type)&&const DeepCollectionEquality().equals(other.meta, meta)&&(identical(other.viewsUnique, viewsUnique) || other.viewsUnique == viewsUnique)&&(identical(other.viewsTotal, viewsTotal) || other.viewsTotal == viewsTotal)&&(identical(other.upvotes, upvotes) || other.upvotes == upvotes)&&(identical(other.downvotes, downvotes) || other.downvotes == downvotes)&&(identical(other.repliesCount, repliesCount) || other.repliesCount == repliesCount)&&(identical(other.threadedPostId, threadedPostId) || other.threadedPostId == threadedPostId)&&(identical(other.threadedPost, threadedPost) || other.threadedPost == threadedPost)&&(identical(other.repliedPostId, repliedPostId) || other.repliedPostId == repliedPostId)&&(identical(other.repliedPost, repliedPost) || other.repliedPost == repliedPost)&&(identical(other.forwardedPostId, forwardedPostId) || other.forwardedPostId == forwardedPostId)&&(identical(other.forwardedPost, forwardedPost) || other.forwardedPost == forwardedPost)&&const DeepCollectionEquality().equals(other.attachments, attachments)&&(identical(other.publisher, publisher) || other.publisher == publisher)&&const DeepCollectionEquality().equals(other.reactionsCount, reactionsCount)&&const DeepCollectionEquality().equals(other.reactions, reactions)&&const DeepCollectionEquality().equals(other.tags, tags)&&const DeepCollectionEquality().equals(other.categories, categories)&&const DeepCollectionEquality().equals(other.collections, collections)&&(identical(other.createdAt, createdAt) || other.createdAt == createdAt)&&(identical(other.updatedAt, updatedAt) || other.updatedAt == updatedAt)&&(identical(other.deletedAt, deletedAt) || other.deletedAt == deletedAt)&&(identical(other.isTruncated, isTruncated) || other.isTruncated == isTruncated)); | ||||
|   return identical(this, other) || (other.runtimeType == runtimeType&&other is SnPost&&(identical(other.id, id) || other.id == id)&&(identical(other.title, title) || other.title == title)&&(identical(other.description, description) || other.description == description)&&(identical(other.language, language) || other.language == language)&&(identical(other.editedAt, editedAt) || other.editedAt == editedAt)&&(identical(other.publishedAt, publishedAt) || other.publishedAt == publishedAt)&&(identical(other.visibility, visibility) || other.visibility == visibility)&&(identical(other.content, content) || other.content == content)&&(identical(other.type, type) || other.type == type)&&const DeepCollectionEquality().equals(other.meta, meta)&&(identical(other.viewsUnique, viewsUnique) || other.viewsUnique == viewsUnique)&&(identical(other.viewsTotal, viewsTotal) || other.viewsTotal == viewsTotal)&&(identical(other.upvotes, upvotes) || other.upvotes == upvotes)&&(identical(other.downvotes, downvotes) || other.downvotes == downvotes)&&(identical(other.repliesCount, repliesCount) || other.repliesCount == repliesCount)&&(identical(other.threadedPostId, threadedPostId) || other.threadedPostId == threadedPostId)&&(identical(other.threadedPost, threadedPost) || other.threadedPost == threadedPost)&&(identical(other.repliedPostId, repliedPostId) || other.repliedPostId == repliedPostId)&&(identical(other.repliedPost, repliedPost) || other.repliedPost == repliedPost)&&(identical(other.forwardedPostId, forwardedPostId) || other.forwardedPostId == forwardedPostId)&&(identical(other.forwardedPost, forwardedPost) || other.forwardedPost == forwardedPost)&&const DeepCollectionEquality().equals(other.attachments, attachments)&&(identical(other.publisher, publisher) || other.publisher == publisher)&&const DeepCollectionEquality().equals(other.reactionsCount, reactionsCount)&&const DeepCollectionEquality().equals(other.reactionsMade, reactionsMade)&&const DeepCollectionEquality().equals(other.reactions, reactions)&&const DeepCollectionEquality().equals(other.tags, tags)&&const DeepCollectionEquality().equals(other.categories, categories)&&const DeepCollectionEquality().equals(other.collections, collections)&&(identical(other.createdAt, createdAt) || other.createdAt == createdAt)&&(identical(other.updatedAt, updatedAt) || other.updatedAt == updatedAt)&&(identical(other.deletedAt, deletedAt) || other.deletedAt == deletedAt)&&(identical(other.isTruncated, isTruncated) || other.isTruncated == isTruncated)); | ||||
| } | ||||
|  | ||||
| @JsonKey(includeFromJson: false, includeToJson: false) | ||||
| @override | ||||
| int get hashCode => Object.hashAll([runtimeType,id,title,description,language,editedAt,publishedAt,visibility,content,type,const DeepCollectionEquality().hash(meta),viewsUnique,viewsTotal,upvotes,downvotes,repliesCount,threadedPostId,threadedPost,repliedPostId,repliedPost,forwardedPostId,forwardedPost,const DeepCollectionEquality().hash(attachments),publisher,const DeepCollectionEquality().hash(reactionsCount),const DeepCollectionEquality().hash(reactions),const DeepCollectionEquality().hash(tags),const DeepCollectionEquality().hash(categories),const DeepCollectionEquality().hash(collections),createdAt,updatedAt,deletedAt,isTruncated]); | ||||
| int get hashCode => Object.hashAll([runtimeType,id,title,description,language,editedAt,publishedAt,visibility,content,type,const DeepCollectionEquality().hash(meta),viewsUnique,viewsTotal,upvotes,downvotes,repliesCount,threadedPostId,threadedPost,repliedPostId,repliedPost,forwardedPostId,forwardedPost,const DeepCollectionEquality().hash(attachments),publisher,const DeepCollectionEquality().hash(reactionsCount),const DeepCollectionEquality().hash(reactionsMade),const DeepCollectionEquality().hash(reactions),const DeepCollectionEquality().hash(tags),const DeepCollectionEquality().hash(categories),const DeepCollectionEquality().hash(collections),createdAt,updatedAt,deletedAt,isTruncated]); | ||||
|  | ||||
| @override | ||||
| String toString() { | ||||
|   return 'SnPost(id: $id, title: $title, description: $description, language: $language, editedAt: $editedAt, publishedAt: $publishedAt, visibility: $visibility, content: $content, type: $type, meta: $meta, viewsUnique: $viewsUnique, viewsTotal: $viewsTotal, upvotes: $upvotes, downvotes: $downvotes, repliesCount: $repliesCount, threadedPostId: $threadedPostId, threadedPost: $threadedPost, repliedPostId: $repliedPostId, repliedPost: $repliedPost, forwardedPostId: $forwardedPostId, forwardedPost: $forwardedPost, attachments: $attachments, publisher: $publisher, reactionsCount: $reactionsCount, reactions: $reactions, tags: $tags, categories: $categories, collections: $collections, createdAt: $createdAt, updatedAt: $updatedAt, deletedAt: $deletedAt, isTruncated: $isTruncated)'; | ||||
|   return 'SnPost(id: $id, title: $title, description: $description, language: $language, editedAt: $editedAt, publishedAt: $publishedAt, visibility: $visibility, content: $content, type: $type, meta: $meta, viewsUnique: $viewsUnique, viewsTotal: $viewsTotal, upvotes: $upvotes, downvotes: $downvotes, repliesCount: $repliesCount, threadedPostId: $threadedPostId, threadedPost: $threadedPost, repliedPostId: $repliedPostId, repliedPost: $repliedPost, forwardedPostId: $forwardedPostId, forwardedPost: $forwardedPost, attachments: $attachments, publisher: $publisher, reactionsCount: $reactionsCount, reactionsMade: $reactionsMade, reactions: $reactions, tags: $tags, categories: $categories, collections: $collections, createdAt: $createdAt, updatedAt: $updatedAt, deletedAt: $deletedAt, isTruncated: $isTruncated)'; | ||||
| } | ||||
|  | ||||
|  | ||||
| @@ -48,7 +48,7 @@ abstract mixin class $SnPostCopyWith<$Res>  { | ||||
|   factory $SnPostCopyWith(SnPost value, $Res Function(SnPost) _then) = _$SnPostCopyWithImpl; | ||||
| @useResult | ||||
| $Res call({ | ||||
|  String id, String? title, String? description, String? language, DateTime? editedAt, DateTime? publishedAt, int visibility, String? content, int type, Map<String, dynamic>? meta, int viewsUnique, int viewsTotal, int upvotes, int downvotes, int repliesCount, String? threadedPostId, SnPost? threadedPost, String? repliedPostId, SnPost? repliedPost, String? forwardedPostId, SnPost? forwardedPost, List<SnCloudFile> attachments, SnPublisher publisher, Map<String, int> reactionsCount, List<dynamic> reactions, List<PostTag> tags, List<PostCategory> categories, List<dynamic> collections, DateTime? createdAt, DateTime? updatedAt, DateTime? deletedAt, bool isTruncated | ||||
|  String id, String? title, String? description, String? language, DateTime? editedAt, DateTime? publishedAt, int visibility, String? content, int type, Map<String, dynamic>? meta, int viewsUnique, int viewsTotal, int upvotes, int downvotes, int repliesCount, String? threadedPostId, SnPost? threadedPost, String? repliedPostId, SnPost? repliedPost, String? forwardedPostId, SnPost? forwardedPost, List<SnCloudFile> attachments, SnPublisher publisher, Map<String, int> reactionsCount, Map<String, bool> reactionsMade, List<dynamic> reactions, List<PostTag> tags, List<PostCategory> categories, List<dynamic> collections, DateTime? createdAt, DateTime? updatedAt, DateTime? deletedAt, bool isTruncated | ||||
| }); | ||||
|  | ||||
|  | ||||
| @@ -65,7 +65,7 @@ class _$SnPostCopyWithImpl<$Res> | ||||
|  | ||||
| /// Create a copy of SnPost | ||||
| /// with the given fields replaced by the non-null parameter values. | ||||
| @pragma('vm:prefer-inline') @override $Res call({Object? id = null,Object? title = freezed,Object? description = freezed,Object? language = freezed,Object? editedAt = freezed,Object? publishedAt = freezed,Object? visibility = null,Object? content = freezed,Object? type = null,Object? meta = freezed,Object? viewsUnique = null,Object? viewsTotal = null,Object? upvotes = null,Object? downvotes = null,Object? repliesCount = null,Object? threadedPostId = freezed,Object? threadedPost = freezed,Object? repliedPostId = freezed,Object? repliedPost = freezed,Object? forwardedPostId = freezed,Object? forwardedPost = freezed,Object? attachments = null,Object? publisher = null,Object? reactionsCount = null,Object? reactions = null,Object? tags = null,Object? categories = null,Object? collections = null,Object? createdAt = freezed,Object? updatedAt = freezed,Object? deletedAt = freezed,Object? isTruncated = null,}) { | ||||
| @pragma('vm:prefer-inline') @override $Res call({Object? id = null,Object? title = freezed,Object? description = freezed,Object? language = freezed,Object? editedAt = freezed,Object? publishedAt = freezed,Object? visibility = null,Object? content = freezed,Object? type = null,Object? meta = freezed,Object? viewsUnique = null,Object? viewsTotal = null,Object? upvotes = null,Object? downvotes = null,Object? repliesCount = null,Object? threadedPostId = freezed,Object? threadedPost = freezed,Object? repliedPostId = freezed,Object? repliedPost = freezed,Object? forwardedPostId = freezed,Object? forwardedPost = freezed,Object? attachments = null,Object? publisher = null,Object? reactionsCount = null,Object? reactionsMade = null,Object? reactions = null,Object? tags = null,Object? categories = null,Object? collections = null,Object? createdAt = freezed,Object? updatedAt = freezed,Object? deletedAt = freezed,Object? isTruncated = null,}) { | ||||
|   return _then(_self.copyWith( | ||||
| id: null == id ? _self.id : id // ignore: cast_nullable_to_non_nullable | ||||
| as String,title: freezed == title ? _self.title : title // ignore: cast_nullable_to_non_nullable | ||||
| @@ -91,7 +91,8 @@ as String?,forwardedPost: freezed == forwardedPost ? _self.forwardedPost : forwa | ||||
| as SnPost?,attachments: null == attachments ? _self.attachments : attachments // ignore: cast_nullable_to_non_nullable | ||||
| as List<SnCloudFile>,publisher: null == publisher ? _self.publisher : publisher // ignore: cast_nullable_to_non_nullable | ||||
| as SnPublisher,reactionsCount: null == reactionsCount ? _self.reactionsCount : reactionsCount // ignore: cast_nullable_to_non_nullable | ||||
| as Map<String, int>,reactions: null == reactions ? _self.reactions : reactions // ignore: cast_nullable_to_non_nullable | ||||
| as Map<String, int>,reactionsMade: null == reactionsMade ? _self.reactionsMade : reactionsMade // ignore: cast_nullable_to_non_nullable | ||||
| as Map<String, bool>,reactions: null == reactions ? _self.reactions : reactions // ignore: cast_nullable_to_non_nullable | ||||
| as List<dynamic>,tags: null == tags ? _self.tags : tags // ignore: cast_nullable_to_non_nullable | ||||
| as List<PostTag>,categories: null == categories ? _self.categories : categories // ignore: cast_nullable_to_non_nullable | ||||
| as List<PostCategory>,collections: null == collections ? _self.collections : collections // ignore: cast_nullable_to_non_nullable | ||||
| @@ -226,10 +227,10 @@ return $default(_that);case _: | ||||
| /// } | ||||
| /// ``` | ||||
|  | ||||
| @optionalTypeArgs TResult maybeWhen<TResult extends Object?>(TResult Function( String id,  String? title,  String? description,  String? language,  DateTime? editedAt,  DateTime? publishedAt,  int visibility,  String? content,  int type,  Map<String, dynamic>? meta,  int viewsUnique,  int viewsTotal,  int upvotes,  int downvotes,  int repliesCount,  String? threadedPostId,  SnPost? threadedPost,  String? repliedPostId,  SnPost? repliedPost,  String? forwardedPostId,  SnPost? forwardedPost,  List<SnCloudFile> attachments,  SnPublisher publisher,  Map<String, int> reactionsCount,  List<dynamic> reactions,  List<PostTag> tags,  List<PostCategory> categories,  List<dynamic> collections,  DateTime? createdAt,  DateTime? updatedAt,  DateTime? deletedAt,  bool isTruncated)?  $default,{required TResult orElse(),}) {final _that = this; | ||||
| @optionalTypeArgs TResult maybeWhen<TResult extends Object?>(TResult Function( String id,  String? title,  String? description,  String? language,  DateTime? editedAt,  DateTime? publishedAt,  int visibility,  String? content,  int type,  Map<String, dynamic>? meta,  int viewsUnique,  int viewsTotal,  int upvotes,  int downvotes,  int repliesCount,  String? threadedPostId,  SnPost? threadedPost,  String? repliedPostId,  SnPost? repliedPost,  String? forwardedPostId,  SnPost? forwardedPost,  List<SnCloudFile> attachments,  SnPublisher publisher,  Map<String, int> reactionsCount,  Map<String, bool> reactionsMade,  List<dynamic> reactions,  List<PostTag> tags,  List<PostCategory> categories,  List<dynamic> collections,  DateTime? createdAt,  DateTime? updatedAt,  DateTime? deletedAt,  bool isTruncated)?  $default,{required TResult orElse(),}) {final _that = this; | ||||
| switch (_that) { | ||||
| case _SnPost() when $default != null: | ||||
| return $default(_that.id,_that.title,_that.description,_that.language,_that.editedAt,_that.publishedAt,_that.visibility,_that.content,_that.type,_that.meta,_that.viewsUnique,_that.viewsTotal,_that.upvotes,_that.downvotes,_that.repliesCount,_that.threadedPostId,_that.threadedPost,_that.repliedPostId,_that.repliedPost,_that.forwardedPostId,_that.forwardedPost,_that.attachments,_that.publisher,_that.reactionsCount,_that.reactions,_that.tags,_that.categories,_that.collections,_that.createdAt,_that.updatedAt,_that.deletedAt,_that.isTruncated);case _: | ||||
| return $default(_that.id,_that.title,_that.description,_that.language,_that.editedAt,_that.publishedAt,_that.visibility,_that.content,_that.type,_that.meta,_that.viewsUnique,_that.viewsTotal,_that.upvotes,_that.downvotes,_that.repliesCount,_that.threadedPostId,_that.threadedPost,_that.repliedPostId,_that.repliedPost,_that.forwardedPostId,_that.forwardedPost,_that.attachments,_that.publisher,_that.reactionsCount,_that.reactionsMade,_that.reactions,_that.tags,_that.categories,_that.collections,_that.createdAt,_that.updatedAt,_that.deletedAt,_that.isTruncated);case _: | ||||
|   return orElse(); | ||||
|  | ||||
| } | ||||
| @@ -247,10 +248,10 @@ return $default(_that.id,_that.title,_that.description,_that.language,_that.edit | ||||
| /// } | ||||
| /// ``` | ||||
|  | ||||
| @optionalTypeArgs TResult when<TResult extends Object?>(TResult Function( String id,  String? title,  String? description,  String? language,  DateTime? editedAt,  DateTime? publishedAt,  int visibility,  String? content,  int type,  Map<String, dynamic>? meta,  int viewsUnique,  int viewsTotal,  int upvotes,  int downvotes,  int repliesCount,  String? threadedPostId,  SnPost? threadedPost,  String? repliedPostId,  SnPost? repliedPost,  String? forwardedPostId,  SnPost? forwardedPost,  List<SnCloudFile> attachments,  SnPublisher publisher,  Map<String, int> reactionsCount,  List<dynamic> reactions,  List<PostTag> tags,  List<PostCategory> categories,  List<dynamic> collections,  DateTime? createdAt,  DateTime? updatedAt,  DateTime? deletedAt,  bool isTruncated)  $default,) {final _that = this; | ||||
| @optionalTypeArgs TResult when<TResult extends Object?>(TResult Function( String id,  String? title,  String? description,  String? language,  DateTime? editedAt,  DateTime? publishedAt,  int visibility,  String? content,  int type,  Map<String, dynamic>? meta,  int viewsUnique,  int viewsTotal,  int upvotes,  int downvotes,  int repliesCount,  String? threadedPostId,  SnPost? threadedPost,  String? repliedPostId,  SnPost? repliedPost,  String? forwardedPostId,  SnPost? forwardedPost,  List<SnCloudFile> attachments,  SnPublisher publisher,  Map<String, int> reactionsCount,  Map<String, bool> reactionsMade,  List<dynamic> reactions,  List<PostTag> tags,  List<PostCategory> categories,  List<dynamic> collections,  DateTime? createdAt,  DateTime? updatedAt,  DateTime? deletedAt,  bool isTruncated)  $default,) {final _that = this; | ||||
| switch (_that) { | ||||
| case _SnPost(): | ||||
| return $default(_that.id,_that.title,_that.description,_that.language,_that.editedAt,_that.publishedAt,_that.visibility,_that.content,_that.type,_that.meta,_that.viewsUnique,_that.viewsTotal,_that.upvotes,_that.downvotes,_that.repliesCount,_that.threadedPostId,_that.threadedPost,_that.repliedPostId,_that.repliedPost,_that.forwardedPostId,_that.forwardedPost,_that.attachments,_that.publisher,_that.reactionsCount,_that.reactions,_that.tags,_that.categories,_that.collections,_that.createdAt,_that.updatedAt,_that.deletedAt,_that.isTruncated);} | ||||
| return $default(_that.id,_that.title,_that.description,_that.language,_that.editedAt,_that.publishedAt,_that.visibility,_that.content,_that.type,_that.meta,_that.viewsUnique,_that.viewsTotal,_that.upvotes,_that.downvotes,_that.repliesCount,_that.threadedPostId,_that.threadedPost,_that.repliedPostId,_that.repliedPost,_that.forwardedPostId,_that.forwardedPost,_that.attachments,_that.publisher,_that.reactionsCount,_that.reactionsMade,_that.reactions,_that.tags,_that.categories,_that.collections,_that.createdAt,_that.updatedAt,_that.deletedAt,_that.isTruncated);} | ||||
| } | ||||
| /// A variant of `when` that fallback to returning `null` | ||||
| /// | ||||
| @@ -264,10 +265,10 @@ return $default(_that.id,_that.title,_that.description,_that.language,_that.edit | ||||
| /// } | ||||
| /// ``` | ||||
|  | ||||
| @optionalTypeArgs TResult? whenOrNull<TResult extends Object?>(TResult? Function( String id,  String? title,  String? description,  String? language,  DateTime? editedAt,  DateTime? publishedAt,  int visibility,  String? content,  int type,  Map<String, dynamic>? meta,  int viewsUnique,  int viewsTotal,  int upvotes,  int downvotes,  int repliesCount,  String? threadedPostId,  SnPost? threadedPost,  String? repliedPostId,  SnPost? repliedPost,  String? forwardedPostId,  SnPost? forwardedPost,  List<SnCloudFile> attachments,  SnPublisher publisher,  Map<String, int> reactionsCount,  List<dynamic> reactions,  List<PostTag> tags,  List<PostCategory> categories,  List<dynamic> collections,  DateTime? createdAt,  DateTime? updatedAt,  DateTime? deletedAt,  bool isTruncated)?  $default,) {final _that = this; | ||||
| @optionalTypeArgs TResult? whenOrNull<TResult extends Object?>(TResult? Function( String id,  String? title,  String? description,  String? language,  DateTime? editedAt,  DateTime? publishedAt,  int visibility,  String? content,  int type,  Map<String, dynamic>? meta,  int viewsUnique,  int viewsTotal,  int upvotes,  int downvotes,  int repliesCount,  String? threadedPostId,  SnPost? threadedPost,  String? repliedPostId,  SnPost? repliedPost,  String? forwardedPostId,  SnPost? forwardedPost,  List<SnCloudFile> attachments,  SnPublisher publisher,  Map<String, int> reactionsCount,  Map<String, bool> reactionsMade,  List<dynamic> reactions,  List<PostTag> tags,  List<PostCategory> categories,  List<dynamic> collections,  DateTime? createdAt,  DateTime? updatedAt,  DateTime? deletedAt,  bool isTruncated)?  $default,) {final _that = this; | ||||
| switch (_that) { | ||||
| case _SnPost() when $default != null: | ||||
| return $default(_that.id,_that.title,_that.description,_that.language,_that.editedAt,_that.publishedAt,_that.visibility,_that.content,_that.type,_that.meta,_that.viewsUnique,_that.viewsTotal,_that.upvotes,_that.downvotes,_that.repliesCount,_that.threadedPostId,_that.threadedPost,_that.repliedPostId,_that.repliedPost,_that.forwardedPostId,_that.forwardedPost,_that.attachments,_that.publisher,_that.reactionsCount,_that.reactions,_that.tags,_that.categories,_that.collections,_that.createdAt,_that.updatedAt,_that.deletedAt,_that.isTruncated);case _: | ||||
| return $default(_that.id,_that.title,_that.description,_that.language,_that.editedAt,_that.publishedAt,_that.visibility,_that.content,_that.type,_that.meta,_that.viewsUnique,_that.viewsTotal,_that.upvotes,_that.downvotes,_that.repliesCount,_that.threadedPostId,_that.threadedPost,_that.repliedPostId,_that.repliedPost,_that.forwardedPostId,_that.forwardedPost,_that.attachments,_that.publisher,_that.reactionsCount,_that.reactionsMade,_that.reactions,_that.tags,_that.categories,_that.collections,_that.createdAt,_that.updatedAt,_that.deletedAt,_that.isTruncated);case _: | ||||
|   return null; | ||||
|  | ||||
| } | ||||
| @@ -279,7 +280,7 @@ return $default(_that.id,_that.title,_that.description,_that.language,_that.edit | ||||
| @JsonSerializable() | ||||
|  | ||||
| class _SnPost implements SnPost { | ||||
|   const _SnPost({required this.id, this.title, this.description, this.language, this.editedAt, this.publishedAt = null, this.visibility = 0, this.content, this.type = 0, final  Map<String, dynamic>? meta, this.viewsUnique = 0, this.viewsTotal = 0, this.upvotes = 0, this.downvotes = 0, this.repliesCount = 0, this.threadedPostId, this.threadedPost, this.repliedPostId, this.repliedPost, this.forwardedPostId, this.forwardedPost, final  List<SnCloudFile> attachments = const [], required this.publisher, final  Map<String, int> reactionsCount = const {}, final  List<dynamic> reactions = const [], final  List<PostTag> tags = const [], final  List<PostCategory> categories = const [], final  List<dynamic> collections = const [], this.createdAt = null, this.updatedAt = null, this.deletedAt, this.isTruncated = false}): _meta = meta,_attachments = attachments,_reactionsCount = reactionsCount,_reactions = reactions,_tags = tags,_categories = categories,_collections = collections; | ||||
|   const _SnPost({required this.id, this.title, this.description, this.language, this.editedAt, this.publishedAt = null, this.visibility = 0, this.content, this.type = 0, final  Map<String, dynamic>? meta, this.viewsUnique = 0, this.viewsTotal = 0, this.upvotes = 0, this.downvotes = 0, this.repliesCount = 0, this.threadedPostId, this.threadedPost, this.repliedPostId, this.repliedPost, this.forwardedPostId, this.forwardedPost, final  List<SnCloudFile> attachments = const [], required this.publisher, final  Map<String, int> reactionsCount = const {}, final  Map<String, bool> reactionsMade = const {}, final  List<dynamic> reactions = const [], final  List<PostTag> tags = const [], final  List<PostCategory> categories = const [], final  List<dynamic> collections = const [], this.createdAt = null, this.updatedAt = null, this.deletedAt, this.isTruncated = false}): _meta = meta,_attachments = attachments,_reactionsCount = reactionsCount,_reactionsMade = reactionsMade,_reactions = reactions,_tags = tags,_categories = categories,_collections = collections; | ||||
|   factory _SnPost.fromJson(Map<String, dynamic> json) => _$SnPostFromJson(json); | ||||
|  | ||||
| @override final  String id; | ||||
| @@ -326,6 +327,13 @@ class _SnPost implements SnPost { | ||||
|   return EqualUnmodifiableMapView(_reactionsCount); | ||||
| } | ||||
|  | ||||
|  final  Map<String, bool> _reactionsMade; | ||||
| @override@JsonKey() Map<String, bool> get reactionsMade { | ||||
|   if (_reactionsMade is EqualUnmodifiableMapView) return _reactionsMade; | ||||
|   // ignore: implicit_dynamic_type | ||||
|   return EqualUnmodifiableMapView(_reactionsMade); | ||||
| } | ||||
|  | ||||
|  final  List<dynamic> _reactions; | ||||
| @override@JsonKey() List<dynamic> get reactions { | ||||
|   if (_reactions is EqualUnmodifiableListView) return _reactions; | ||||
| @@ -372,16 +380,16 @@ Map<String, dynamic> toJson() { | ||||
|  | ||||
| @override | ||||
| bool operator ==(Object other) { | ||||
|   return identical(this, other) || (other.runtimeType == runtimeType&&other is _SnPost&&(identical(other.id, id) || other.id == id)&&(identical(other.title, title) || other.title == title)&&(identical(other.description, description) || other.description == description)&&(identical(other.language, language) || other.language == language)&&(identical(other.editedAt, editedAt) || other.editedAt == editedAt)&&(identical(other.publishedAt, publishedAt) || other.publishedAt == publishedAt)&&(identical(other.visibility, visibility) || other.visibility == visibility)&&(identical(other.content, content) || other.content == content)&&(identical(other.type, type) || other.type == type)&&const DeepCollectionEquality().equals(other._meta, _meta)&&(identical(other.viewsUnique, viewsUnique) || other.viewsUnique == viewsUnique)&&(identical(other.viewsTotal, viewsTotal) || other.viewsTotal == viewsTotal)&&(identical(other.upvotes, upvotes) || other.upvotes == upvotes)&&(identical(other.downvotes, downvotes) || other.downvotes == downvotes)&&(identical(other.repliesCount, repliesCount) || other.repliesCount == repliesCount)&&(identical(other.threadedPostId, threadedPostId) || other.threadedPostId == threadedPostId)&&(identical(other.threadedPost, threadedPost) || other.threadedPost == threadedPost)&&(identical(other.repliedPostId, repliedPostId) || other.repliedPostId == repliedPostId)&&(identical(other.repliedPost, repliedPost) || other.repliedPost == repliedPost)&&(identical(other.forwardedPostId, forwardedPostId) || other.forwardedPostId == forwardedPostId)&&(identical(other.forwardedPost, forwardedPost) || other.forwardedPost == forwardedPost)&&const DeepCollectionEquality().equals(other._attachments, _attachments)&&(identical(other.publisher, publisher) || other.publisher == publisher)&&const DeepCollectionEquality().equals(other._reactionsCount, _reactionsCount)&&const DeepCollectionEquality().equals(other._reactions, _reactions)&&const DeepCollectionEquality().equals(other._tags, _tags)&&const DeepCollectionEquality().equals(other._categories, _categories)&&const DeepCollectionEquality().equals(other._collections, _collections)&&(identical(other.createdAt, createdAt) || other.createdAt == createdAt)&&(identical(other.updatedAt, updatedAt) || other.updatedAt == updatedAt)&&(identical(other.deletedAt, deletedAt) || other.deletedAt == deletedAt)&&(identical(other.isTruncated, isTruncated) || other.isTruncated == isTruncated)); | ||||
|   return identical(this, other) || (other.runtimeType == runtimeType&&other is _SnPost&&(identical(other.id, id) || other.id == id)&&(identical(other.title, title) || other.title == title)&&(identical(other.description, description) || other.description == description)&&(identical(other.language, language) || other.language == language)&&(identical(other.editedAt, editedAt) || other.editedAt == editedAt)&&(identical(other.publishedAt, publishedAt) || other.publishedAt == publishedAt)&&(identical(other.visibility, visibility) || other.visibility == visibility)&&(identical(other.content, content) || other.content == content)&&(identical(other.type, type) || other.type == type)&&const DeepCollectionEquality().equals(other._meta, _meta)&&(identical(other.viewsUnique, viewsUnique) || other.viewsUnique == viewsUnique)&&(identical(other.viewsTotal, viewsTotal) || other.viewsTotal == viewsTotal)&&(identical(other.upvotes, upvotes) || other.upvotes == upvotes)&&(identical(other.downvotes, downvotes) || other.downvotes == downvotes)&&(identical(other.repliesCount, repliesCount) || other.repliesCount == repliesCount)&&(identical(other.threadedPostId, threadedPostId) || other.threadedPostId == threadedPostId)&&(identical(other.threadedPost, threadedPost) || other.threadedPost == threadedPost)&&(identical(other.repliedPostId, repliedPostId) || other.repliedPostId == repliedPostId)&&(identical(other.repliedPost, repliedPost) || other.repliedPost == repliedPost)&&(identical(other.forwardedPostId, forwardedPostId) || other.forwardedPostId == forwardedPostId)&&(identical(other.forwardedPost, forwardedPost) || other.forwardedPost == forwardedPost)&&const DeepCollectionEquality().equals(other._attachments, _attachments)&&(identical(other.publisher, publisher) || other.publisher == publisher)&&const DeepCollectionEquality().equals(other._reactionsCount, _reactionsCount)&&const DeepCollectionEquality().equals(other._reactionsMade, _reactionsMade)&&const DeepCollectionEquality().equals(other._reactions, _reactions)&&const DeepCollectionEquality().equals(other._tags, _tags)&&const DeepCollectionEquality().equals(other._categories, _categories)&&const DeepCollectionEquality().equals(other._collections, _collections)&&(identical(other.createdAt, createdAt) || other.createdAt == createdAt)&&(identical(other.updatedAt, updatedAt) || other.updatedAt == updatedAt)&&(identical(other.deletedAt, deletedAt) || other.deletedAt == deletedAt)&&(identical(other.isTruncated, isTruncated) || other.isTruncated == isTruncated)); | ||||
| } | ||||
|  | ||||
| @JsonKey(includeFromJson: false, includeToJson: false) | ||||
| @override | ||||
| int get hashCode => Object.hashAll([runtimeType,id,title,description,language,editedAt,publishedAt,visibility,content,type,const DeepCollectionEquality().hash(_meta),viewsUnique,viewsTotal,upvotes,downvotes,repliesCount,threadedPostId,threadedPost,repliedPostId,repliedPost,forwardedPostId,forwardedPost,const DeepCollectionEquality().hash(_attachments),publisher,const DeepCollectionEquality().hash(_reactionsCount),const DeepCollectionEquality().hash(_reactions),const DeepCollectionEquality().hash(_tags),const DeepCollectionEquality().hash(_categories),const DeepCollectionEquality().hash(_collections),createdAt,updatedAt,deletedAt,isTruncated]); | ||||
| int get hashCode => Object.hashAll([runtimeType,id,title,description,language,editedAt,publishedAt,visibility,content,type,const DeepCollectionEquality().hash(_meta),viewsUnique,viewsTotal,upvotes,downvotes,repliesCount,threadedPostId,threadedPost,repliedPostId,repliedPost,forwardedPostId,forwardedPost,const DeepCollectionEquality().hash(_attachments),publisher,const DeepCollectionEquality().hash(_reactionsCount),const DeepCollectionEquality().hash(_reactionsMade),const DeepCollectionEquality().hash(_reactions),const DeepCollectionEquality().hash(_tags),const DeepCollectionEquality().hash(_categories),const DeepCollectionEquality().hash(_collections),createdAt,updatedAt,deletedAt,isTruncated]); | ||||
|  | ||||
| @override | ||||
| String toString() { | ||||
|   return 'SnPost(id: $id, title: $title, description: $description, language: $language, editedAt: $editedAt, publishedAt: $publishedAt, visibility: $visibility, content: $content, type: $type, meta: $meta, viewsUnique: $viewsUnique, viewsTotal: $viewsTotal, upvotes: $upvotes, downvotes: $downvotes, repliesCount: $repliesCount, threadedPostId: $threadedPostId, threadedPost: $threadedPost, repliedPostId: $repliedPostId, repliedPost: $repliedPost, forwardedPostId: $forwardedPostId, forwardedPost: $forwardedPost, attachments: $attachments, publisher: $publisher, reactionsCount: $reactionsCount, reactions: $reactions, tags: $tags, categories: $categories, collections: $collections, createdAt: $createdAt, updatedAt: $updatedAt, deletedAt: $deletedAt, isTruncated: $isTruncated)'; | ||||
|   return 'SnPost(id: $id, title: $title, description: $description, language: $language, editedAt: $editedAt, publishedAt: $publishedAt, visibility: $visibility, content: $content, type: $type, meta: $meta, viewsUnique: $viewsUnique, viewsTotal: $viewsTotal, upvotes: $upvotes, downvotes: $downvotes, repliesCount: $repliesCount, threadedPostId: $threadedPostId, threadedPost: $threadedPost, repliedPostId: $repliedPostId, repliedPost: $repliedPost, forwardedPostId: $forwardedPostId, forwardedPost: $forwardedPost, attachments: $attachments, publisher: $publisher, reactionsCount: $reactionsCount, reactionsMade: $reactionsMade, reactions: $reactions, tags: $tags, categories: $categories, collections: $collections, createdAt: $createdAt, updatedAt: $updatedAt, deletedAt: $deletedAt, isTruncated: $isTruncated)'; | ||||
| } | ||||
|  | ||||
|  | ||||
| @@ -392,7 +400,7 @@ abstract mixin class _$SnPostCopyWith<$Res> implements $SnPostCopyWith<$Res> { | ||||
|   factory _$SnPostCopyWith(_SnPost value, $Res Function(_SnPost) _then) = __$SnPostCopyWithImpl; | ||||
| @override @useResult | ||||
| $Res call({ | ||||
|  String id, String? title, String? description, String? language, DateTime? editedAt, DateTime? publishedAt, int visibility, String? content, int type, Map<String, dynamic>? meta, int viewsUnique, int viewsTotal, int upvotes, int downvotes, int repliesCount, String? threadedPostId, SnPost? threadedPost, String? repliedPostId, SnPost? repliedPost, String? forwardedPostId, SnPost? forwardedPost, List<SnCloudFile> attachments, SnPublisher publisher, Map<String, int> reactionsCount, List<dynamic> reactions, List<PostTag> tags, List<PostCategory> categories, List<dynamic> collections, DateTime? createdAt, DateTime? updatedAt, DateTime? deletedAt, bool isTruncated | ||||
|  String id, String? title, String? description, String? language, DateTime? editedAt, DateTime? publishedAt, int visibility, String? content, int type, Map<String, dynamic>? meta, int viewsUnique, int viewsTotal, int upvotes, int downvotes, int repliesCount, String? threadedPostId, SnPost? threadedPost, String? repliedPostId, SnPost? repliedPost, String? forwardedPostId, SnPost? forwardedPost, List<SnCloudFile> attachments, SnPublisher publisher, Map<String, int> reactionsCount, Map<String, bool> reactionsMade, List<dynamic> reactions, List<PostTag> tags, List<PostCategory> categories, List<dynamic> collections, DateTime? createdAt, DateTime? updatedAt, DateTime? deletedAt, bool isTruncated | ||||
| }); | ||||
|  | ||||
|  | ||||
| @@ -409,7 +417,7 @@ class __$SnPostCopyWithImpl<$Res> | ||||
|  | ||||
| /// Create a copy of SnPost | ||||
| /// with the given fields replaced by the non-null parameter values. | ||||
| @override @pragma('vm:prefer-inline') $Res call({Object? id = null,Object? title = freezed,Object? description = freezed,Object? language = freezed,Object? editedAt = freezed,Object? publishedAt = freezed,Object? visibility = null,Object? content = freezed,Object? type = null,Object? meta = freezed,Object? viewsUnique = null,Object? viewsTotal = null,Object? upvotes = null,Object? downvotes = null,Object? repliesCount = null,Object? threadedPostId = freezed,Object? threadedPost = freezed,Object? repliedPostId = freezed,Object? repliedPost = freezed,Object? forwardedPostId = freezed,Object? forwardedPost = freezed,Object? attachments = null,Object? publisher = null,Object? reactionsCount = null,Object? reactions = null,Object? tags = null,Object? categories = null,Object? collections = null,Object? createdAt = freezed,Object? updatedAt = freezed,Object? deletedAt = freezed,Object? isTruncated = null,}) { | ||||
| @override @pragma('vm:prefer-inline') $Res call({Object? id = null,Object? title = freezed,Object? description = freezed,Object? language = freezed,Object? editedAt = freezed,Object? publishedAt = freezed,Object? visibility = null,Object? content = freezed,Object? type = null,Object? meta = freezed,Object? viewsUnique = null,Object? viewsTotal = null,Object? upvotes = null,Object? downvotes = null,Object? repliesCount = null,Object? threadedPostId = freezed,Object? threadedPost = freezed,Object? repliedPostId = freezed,Object? repliedPost = freezed,Object? forwardedPostId = freezed,Object? forwardedPost = freezed,Object? attachments = null,Object? publisher = null,Object? reactionsCount = null,Object? reactionsMade = null,Object? reactions = null,Object? tags = null,Object? categories = null,Object? collections = null,Object? createdAt = freezed,Object? updatedAt = freezed,Object? deletedAt = freezed,Object? isTruncated = null,}) { | ||||
|   return _then(_SnPost( | ||||
| id: null == id ? _self.id : id // ignore: cast_nullable_to_non_nullable | ||||
| as String,title: freezed == title ? _self.title : title // ignore: cast_nullable_to_non_nullable | ||||
| @@ -435,7 +443,8 @@ as String?,forwardedPost: freezed == forwardedPost ? _self.forwardedPost : forwa | ||||
| as SnPost?,attachments: null == attachments ? _self._attachments : attachments // ignore: cast_nullable_to_non_nullable | ||||
| as List<SnCloudFile>,publisher: null == publisher ? _self.publisher : publisher // ignore: cast_nullable_to_non_nullable | ||||
| as SnPublisher,reactionsCount: null == reactionsCount ? _self._reactionsCount : reactionsCount // ignore: cast_nullable_to_non_nullable | ||||
| as Map<String, int>,reactions: null == reactions ? _self._reactions : reactions // ignore: cast_nullable_to_non_nullable | ||||
| as Map<String, int>,reactionsMade: null == reactionsMade ? _self._reactionsMade : reactionsMade // ignore: cast_nullable_to_non_nullable | ||||
| as Map<String, bool>,reactions: null == reactions ? _self._reactions : reactions // ignore: cast_nullable_to_non_nullable | ||||
| as List<dynamic>,tags: null == tags ? _self._tags : tags // ignore: cast_nullable_to_non_nullable | ||||
| as List<PostTag>,categories: null == categories ? _self._categories : categories // ignore: cast_nullable_to_non_nullable | ||||
| as List<PostCategory>,collections: null == collections ? _self._collections : collections // ignore: cast_nullable_to_non_nullable | ||||
|   | ||||
| @@ -54,6 +54,11 @@ _SnPost _$SnPostFromJson(Map<String, dynamic> json) => _SnPost( | ||||
|         (k, e) => MapEntry(k, (e as num).toInt()), | ||||
|       ) ?? | ||||
|       const {}, | ||||
|   reactionsMade: | ||||
|       (json['reactions_made'] as Map<String, dynamic>?)?.map( | ||||
|         (k, e) => MapEntry(k, e as bool), | ||||
|       ) ?? | ||||
|       const {}, | ||||
|   reactions: json['reactions'] as List<dynamic>? ?? const [], | ||||
|   tags: | ||||
|       (json['tags'] as List<dynamic>?) | ||||
| @@ -106,6 +111,7 @@ Map<String, dynamic> _$SnPostToJson(_SnPost instance) => <String, dynamic>{ | ||||
|   'attachments': instance.attachments.map((e) => e.toJson()).toList(), | ||||
|   'publisher': instance.publisher.toJson(), | ||||
|   'reactions_count': instance.reactionsCount, | ||||
|   'reactions_made': instance.reactionsMade, | ||||
|   'reactions': instance.reactions, | ||||
|   'tags': instance.tags.map((e) => e.toJson()).toList(), | ||||
|   'categories': instance.categories.map((e) => e.toJson()).toList(), | ||||
|   | ||||
| @@ -14,6 +14,7 @@ sealed class SnAccount with _$SnAccount { | ||||
|     required String language, | ||||
|     required bool isSuperuser, | ||||
|     required SnAccountProfile profile, | ||||
|     required SnWalletSubscriptionRef? perkSubscription, | ||||
|     @Default([]) List<SnAccountBadge> badges, | ||||
|     required DateTime createdAt, | ||||
|     required DateTime updatedAt, | ||||
| @@ -45,7 +46,6 @@ sealed class SnAccountProfile with _$SnAccountProfile { | ||||
|     required SnCloudFile? picture, | ||||
|     required SnCloudFile? background, | ||||
|     required SnVerificationMark? verification, | ||||
|     required SnWalletSubscriptionRef? stellarMembership, | ||||
|     required DateTime createdAt, | ||||
|     required DateTime updatedAt, | ||||
|     required DateTime? deletedAt, | ||||
|   | ||||
| @@ -15,7 +15,7 @@ T _$identity<T>(T value) => value; | ||||
| /// @nodoc | ||||
| mixin _$SnAccount { | ||||
|  | ||||
|  String get id; String get name; String get nick; String get language; bool get isSuperuser; SnAccountProfile get profile; List<SnAccountBadge> get badges; DateTime get createdAt; DateTime get updatedAt; DateTime? get deletedAt; | ||||
|  String get id; String get name; String get nick; String get language; bool get isSuperuser; SnAccountProfile get profile; SnWalletSubscriptionRef? get perkSubscription; List<SnAccountBadge> get badges; DateTime get createdAt; DateTime get updatedAt; DateTime? get deletedAt; | ||||
| /// Create a copy of SnAccount | ||||
| /// with the given fields replaced by the non-null parameter values. | ||||
| @JsonKey(includeFromJson: false, includeToJson: false) | ||||
| @@ -28,16 +28,16 @@ $SnAccountCopyWith<SnAccount> get copyWith => _$SnAccountCopyWithImpl<SnAccount> | ||||
|  | ||||
| @override | ||||
| bool operator ==(Object other) { | ||||
|   return identical(this, other) || (other.runtimeType == runtimeType&&other is SnAccount&&(identical(other.id, id) || other.id == id)&&(identical(other.name, name) || other.name == name)&&(identical(other.nick, nick) || other.nick == nick)&&(identical(other.language, language) || other.language == language)&&(identical(other.isSuperuser, isSuperuser) || other.isSuperuser == isSuperuser)&&(identical(other.profile, profile) || other.profile == profile)&&const DeepCollectionEquality().equals(other.badges, badges)&&(identical(other.createdAt, createdAt) || other.createdAt == createdAt)&&(identical(other.updatedAt, updatedAt) || other.updatedAt == updatedAt)&&(identical(other.deletedAt, deletedAt) || other.deletedAt == deletedAt)); | ||||
|   return identical(this, other) || (other.runtimeType == runtimeType&&other is SnAccount&&(identical(other.id, id) || other.id == id)&&(identical(other.name, name) || other.name == name)&&(identical(other.nick, nick) || other.nick == nick)&&(identical(other.language, language) || other.language == language)&&(identical(other.isSuperuser, isSuperuser) || other.isSuperuser == isSuperuser)&&(identical(other.profile, profile) || other.profile == profile)&&(identical(other.perkSubscription, perkSubscription) || other.perkSubscription == perkSubscription)&&const DeepCollectionEquality().equals(other.badges, badges)&&(identical(other.createdAt, createdAt) || other.createdAt == createdAt)&&(identical(other.updatedAt, updatedAt) || other.updatedAt == updatedAt)&&(identical(other.deletedAt, deletedAt) || other.deletedAt == deletedAt)); | ||||
| } | ||||
|  | ||||
| @JsonKey(includeFromJson: false, includeToJson: false) | ||||
| @override | ||||
| int get hashCode => Object.hash(runtimeType,id,name,nick,language,isSuperuser,profile,const DeepCollectionEquality().hash(badges),createdAt,updatedAt,deletedAt); | ||||
| int get hashCode => Object.hash(runtimeType,id,name,nick,language,isSuperuser,profile,perkSubscription,const DeepCollectionEquality().hash(badges),createdAt,updatedAt,deletedAt); | ||||
|  | ||||
| @override | ||||
| String toString() { | ||||
|   return 'SnAccount(id: $id, name: $name, nick: $nick, language: $language, isSuperuser: $isSuperuser, profile: $profile, badges: $badges, createdAt: $createdAt, updatedAt: $updatedAt, deletedAt: $deletedAt)'; | ||||
|   return 'SnAccount(id: $id, name: $name, nick: $nick, language: $language, isSuperuser: $isSuperuser, profile: $profile, perkSubscription: $perkSubscription, badges: $badges, createdAt: $createdAt, updatedAt: $updatedAt, deletedAt: $deletedAt)'; | ||||
| } | ||||
|  | ||||
|  | ||||
| @@ -48,11 +48,11 @@ abstract mixin class $SnAccountCopyWith<$Res>  { | ||||
|   factory $SnAccountCopyWith(SnAccount value, $Res Function(SnAccount) _then) = _$SnAccountCopyWithImpl; | ||||
| @useResult | ||||
| $Res call({ | ||||
|  String id, String name, String nick, String language, bool isSuperuser, SnAccountProfile profile, List<SnAccountBadge> badges, DateTime createdAt, DateTime updatedAt, DateTime? deletedAt | ||||
|  String id, String name, String nick, String language, bool isSuperuser, SnAccountProfile profile, SnWalletSubscriptionRef? perkSubscription, List<SnAccountBadge> badges, DateTime createdAt, DateTime updatedAt, DateTime? deletedAt | ||||
| }); | ||||
|  | ||||
|  | ||||
| $SnAccountProfileCopyWith<$Res> get profile; | ||||
| $SnAccountProfileCopyWith<$Res> get profile;$SnWalletSubscriptionRefCopyWith<$Res>? get perkSubscription; | ||||
|  | ||||
| } | ||||
| /// @nodoc | ||||
| @@ -65,7 +65,7 @@ class _$SnAccountCopyWithImpl<$Res> | ||||
|  | ||||
| /// Create a copy of SnAccount | ||||
| /// with the given fields replaced by the non-null parameter values. | ||||
| @pragma('vm:prefer-inline') @override $Res call({Object? id = null,Object? name = null,Object? nick = null,Object? language = null,Object? isSuperuser = null,Object? profile = null,Object? badges = null,Object? createdAt = null,Object? updatedAt = null,Object? deletedAt = freezed,}) { | ||||
| @pragma('vm:prefer-inline') @override $Res call({Object? id = null,Object? name = null,Object? nick = null,Object? language = null,Object? isSuperuser = null,Object? profile = null,Object? perkSubscription = freezed,Object? badges = null,Object? createdAt = null,Object? updatedAt = null,Object? deletedAt = freezed,}) { | ||||
|   return _then(_self.copyWith( | ||||
| id: null == id ? _self.id : id // ignore: cast_nullable_to_non_nullable | ||||
| as String,name: null == name ? _self.name : name // ignore: cast_nullable_to_non_nullable | ||||
| @@ -73,7 +73,8 @@ as String,nick: null == nick ? _self.nick : nick // ignore: cast_nullable_to_non | ||||
| as String,language: null == language ? _self.language : language // ignore: cast_nullable_to_non_nullable | ||||
| as String,isSuperuser: null == isSuperuser ? _self.isSuperuser : isSuperuser // ignore: cast_nullable_to_non_nullable | ||||
| as bool,profile: null == profile ? _self.profile : profile // ignore: cast_nullable_to_non_nullable | ||||
| as SnAccountProfile,badges: null == badges ? _self.badges : badges // ignore: cast_nullable_to_non_nullable | ||||
| as SnAccountProfile,perkSubscription: freezed == perkSubscription ? _self.perkSubscription : perkSubscription // ignore: cast_nullable_to_non_nullable | ||||
| as SnWalletSubscriptionRef?,badges: null == badges ? _self.badges : badges // ignore: cast_nullable_to_non_nullable | ||||
| as List<SnAccountBadge>,createdAt: null == createdAt ? _self.createdAt : createdAt // ignore: cast_nullable_to_non_nullable | ||||
| as DateTime,updatedAt: null == updatedAt ? _self.updatedAt : updatedAt // ignore: cast_nullable_to_non_nullable | ||||
| as DateTime,deletedAt: freezed == deletedAt ? _self.deletedAt : deletedAt // ignore: cast_nullable_to_non_nullable | ||||
| @@ -89,6 +90,18 @@ $SnAccountProfileCopyWith<$Res> get profile { | ||||
|   return $SnAccountProfileCopyWith<$Res>(_self.profile, (value) { | ||||
|     return _then(_self.copyWith(profile: value)); | ||||
|   }); | ||||
| }/// Create a copy of SnAccount | ||||
| /// with the given fields replaced by the non-null parameter values. | ||||
| @override | ||||
| @pragma('vm:prefer-inline') | ||||
| $SnWalletSubscriptionRefCopyWith<$Res>? get perkSubscription { | ||||
|     if (_self.perkSubscription == null) { | ||||
|     return null; | ||||
|   } | ||||
|  | ||||
|   return $SnWalletSubscriptionRefCopyWith<$Res>(_self.perkSubscription!, (value) { | ||||
|     return _then(_self.copyWith(perkSubscription: value)); | ||||
|   }); | ||||
| } | ||||
| } | ||||
|  | ||||
| @@ -168,10 +181,10 @@ return $default(_that);case _: | ||||
| /// } | ||||
| /// ``` | ||||
|  | ||||
| @optionalTypeArgs TResult maybeWhen<TResult extends Object?>(TResult Function( String id,  String name,  String nick,  String language,  bool isSuperuser,  SnAccountProfile profile,  List<SnAccountBadge> badges,  DateTime createdAt,  DateTime updatedAt,  DateTime? deletedAt)?  $default,{required TResult orElse(),}) {final _that = this; | ||||
| @optionalTypeArgs TResult maybeWhen<TResult extends Object?>(TResult Function( String id,  String name,  String nick,  String language,  bool isSuperuser,  SnAccountProfile profile,  SnWalletSubscriptionRef? perkSubscription,  List<SnAccountBadge> badges,  DateTime createdAt,  DateTime updatedAt,  DateTime? deletedAt)?  $default,{required TResult orElse(),}) {final _that = this; | ||||
| switch (_that) { | ||||
| case _SnAccount() when $default != null: | ||||
| return $default(_that.id,_that.name,_that.nick,_that.language,_that.isSuperuser,_that.profile,_that.badges,_that.createdAt,_that.updatedAt,_that.deletedAt);case _: | ||||
| return $default(_that.id,_that.name,_that.nick,_that.language,_that.isSuperuser,_that.profile,_that.perkSubscription,_that.badges,_that.createdAt,_that.updatedAt,_that.deletedAt);case _: | ||||
|   return orElse(); | ||||
|  | ||||
| } | ||||
| @@ -189,10 +202,10 @@ return $default(_that.id,_that.name,_that.nick,_that.language,_that.isSuperuser, | ||||
| /// } | ||||
| /// ``` | ||||
|  | ||||
| @optionalTypeArgs TResult when<TResult extends Object?>(TResult Function( String id,  String name,  String nick,  String language,  bool isSuperuser,  SnAccountProfile profile,  List<SnAccountBadge> badges,  DateTime createdAt,  DateTime updatedAt,  DateTime? deletedAt)  $default,) {final _that = this; | ||||
| @optionalTypeArgs TResult when<TResult extends Object?>(TResult Function( String id,  String name,  String nick,  String language,  bool isSuperuser,  SnAccountProfile profile,  SnWalletSubscriptionRef? perkSubscription,  List<SnAccountBadge> badges,  DateTime createdAt,  DateTime updatedAt,  DateTime? deletedAt)  $default,) {final _that = this; | ||||
| switch (_that) { | ||||
| case _SnAccount(): | ||||
| return $default(_that.id,_that.name,_that.nick,_that.language,_that.isSuperuser,_that.profile,_that.badges,_that.createdAt,_that.updatedAt,_that.deletedAt);} | ||||
| return $default(_that.id,_that.name,_that.nick,_that.language,_that.isSuperuser,_that.profile,_that.perkSubscription,_that.badges,_that.createdAt,_that.updatedAt,_that.deletedAt);} | ||||
| } | ||||
| /// A variant of `when` that fallback to returning `null` | ||||
| /// | ||||
| @@ -206,10 +219,10 @@ return $default(_that.id,_that.name,_that.nick,_that.language,_that.isSuperuser, | ||||
| /// } | ||||
| /// ``` | ||||
|  | ||||
| @optionalTypeArgs TResult? whenOrNull<TResult extends Object?>(TResult? Function( String id,  String name,  String nick,  String language,  bool isSuperuser,  SnAccountProfile profile,  List<SnAccountBadge> badges,  DateTime createdAt,  DateTime updatedAt,  DateTime? deletedAt)?  $default,) {final _that = this; | ||||
| @optionalTypeArgs TResult? whenOrNull<TResult extends Object?>(TResult? Function( String id,  String name,  String nick,  String language,  bool isSuperuser,  SnAccountProfile profile,  SnWalletSubscriptionRef? perkSubscription,  List<SnAccountBadge> badges,  DateTime createdAt,  DateTime updatedAt,  DateTime? deletedAt)?  $default,) {final _that = this; | ||||
| switch (_that) { | ||||
| case _SnAccount() when $default != null: | ||||
| return $default(_that.id,_that.name,_that.nick,_that.language,_that.isSuperuser,_that.profile,_that.badges,_that.createdAt,_that.updatedAt,_that.deletedAt);case _: | ||||
| return $default(_that.id,_that.name,_that.nick,_that.language,_that.isSuperuser,_that.profile,_that.perkSubscription,_that.badges,_that.createdAt,_that.updatedAt,_that.deletedAt);case _: | ||||
|   return null; | ||||
|  | ||||
| } | ||||
| @@ -221,7 +234,7 @@ return $default(_that.id,_that.name,_that.nick,_that.language,_that.isSuperuser, | ||||
| @JsonSerializable() | ||||
|  | ||||
| class _SnAccount implements SnAccount { | ||||
|   const _SnAccount({required this.id, required this.name, required this.nick, required this.language, required this.isSuperuser, required this.profile, final  List<SnAccountBadge> badges = const [], required this.createdAt, required this.updatedAt, required this.deletedAt}): _badges = badges; | ||||
|   const _SnAccount({required this.id, required this.name, required this.nick, required this.language, required this.isSuperuser, required this.profile, required this.perkSubscription, final  List<SnAccountBadge> badges = const [], required this.createdAt, required this.updatedAt, required this.deletedAt}): _badges = badges; | ||||
|   factory _SnAccount.fromJson(Map<String, dynamic> json) => _$SnAccountFromJson(json); | ||||
|  | ||||
| @override final  String id; | ||||
| @@ -230,6 +243,7 @@ class _SnAccount implements SnAccount { | ||||
| @override final  String language; | ||||
| @override final  bool isSuperuser; | ||||
| @override final  SnAccountProfile profile; | ||||
| @override final  SnWalletSubscriptionRef? perkSubscription; | ||||
|  final  List<SnAccountBadge> _badges; | ||||
| @override@JsonKey() List<SnAccountBadge> get badges { | ||||
|   if (_badges is EqualUnmodifiableListView) return _badges; | ||||
| @@ -254,16 +268,16 @@ Map<String, dynamic> toJson() { | ||||
|  | ||||
| @override | ||||
| bool operator ==(Object other) { | ||||
|   return identical(this, other) || (other.runtimeType == runtimeType&&other is _SnAccount&&(identical(other.id, id) || other.id == id)&&(identical(other.name, name) || other.name == name)&&(identical(other.nick, nick) || other.nick == nick)&&(identical(other.language, language) || other.language == language)&&(identical(other.isSuperuser, isSuperuser) || other.isSuperuser == isSuperuser)&&(identical(other.profile, profile) || other.profile == profile)&&const DeepCollectionEquality().equals(other._badges, _badges)&&(identical(other.createdAt, createdAt) || other.createdAt == createdAt)&&(identical(other.updatedAt, updatedAt) || other.updatedAt == updatedAt)&&(identical(other.deletedAt, deletedAt) || other.deletedAt == deletedAt)); | ||||
|   return identical(this, other) || (other.runtimeType == runtimeType&&other is _SnAccount&&(identical(other.id, id) || other.id == id)&&(identical(other.name, name) || other.name == name)&&(identical(other.nick, nick) || other.nick == nick)&&(identical(other.language, language) || other.language == language)&&(identical(other.isSuperuser, isSuperuser) || other.isSuperuser == isSuperuser)&&(identical(other.profile, profile) || other.profile == profile)&&(identical(other.perkSubscription, perkSubscription) || other.perkSubscription == perkSubscription)&&const DeepCollectionEquality().equals(other._badges, _badges)&&(identical(other.createdAt, createdAt) || other.createdAt == createdAt)&&(identical(other.updatedAt, updatedAt) || other.updatedAt == updatedAt)&&(identical(other.deletedAt, deletedAt) || other.deletedAt == deletedAt)); | ||||
| } | ||||
|  | ||||
| @JsonKey(includeFromJson: false, includeToJson: false) | ||||
| @override | ||||
| int get hashCode => Object.hash(runtimeType,id,name,nick,language,isSuperuser,profile,const DeepCollectionEquality().hash(_badges),createdAt,updatedAt,deletedAt); | ||||
| int get hashCode => Object.hash(runtimeType,id,name,nick,language,isSuperuser,profile,perkSubscription,const DeepCollectionEquality().hash(_badges),createdAt,updatedAt,deletedAt); | ||||
|  | ||||
| @override | ||||
| String toString() { | ||||
|   return 'SnAccount(id: $id, name: $name, nick: $nick, language: $language, isSuperuser: $isSuperuser, profile: $profile, badges: $badges, createdAt: $createdAt, updatedAt: $updatedAt, deletedAt: $deletedAt)'; | ||||
|   return 'SnAccount(id: $id, name: $name, nick: $nick, language: $language, isSuperuser: $isSuperuser, profile: $profile, perkSubscription: $perkSubscription, badges: $badges, createdAt: $createdAt, updatedAt: $updatedAt, deletedAt: $deletedAt)'; | ||||
| } | ||||
|  | ||||
|  | ||||
| @@ -274,11 +288,11 @@ abstract mixin class _$SnAccountCopyWith<$Res> implements $SnAccountCopyWith<$Re | ||||
|   factory _$SnAccountCopyWith(_SnAccount value, $Res Function(_SnAccount) _then) = __$SnAccountCopyWithImpl; | ||||
| @override @useResult | ||||
| $Res call({ | ||||
|  String id, String name, String nick, String language, bool isSuperuser, SnAccountProfile profile, List<SnAccountBadge> badges, DateTime createdAt, DateTime updatedAt, DateTime? deletedAt | ||||
|  String id, String name, String nick, String language, bool isSuperuser, SnAccountProfile profile, SnWalletSubscriptionRef? perkSubscription, List<SnAccountBadge> badges, DateTime createdAt, DateTime updatedAt, DateTime? deletedAt | ||||
| }); | ||||
|  | ||||
|  | ||||
| @override $SnAccountProfileCopyWith<$Res> get profile; | ||||
| @override $SnAccountProfileCopyWith<$Res> get profile;@override $SnWalletSubscriptionRefCopyWith<$Res>? get perkSubscription; | ||||
|  | ||||
| } | ||||
| /// @nodoc | ||||
| @@ -291,7 +305,7 @@ class __$SnAccountCopyWithImpl<$Res> | ||||
|  | ||||
| /// Create a copy of SnAccount | ||||
| /// with the given fields replaced by the non-null parameter values. | ||||
| @override @pragma('vm:prefer-inline') $Res call({Object? id = null,Object? name = null,Object? nick = null,Object? language = null,Object? isSuperuser = null,Object? profile = null,Object? badges = null,Object? createdAt = null,Object? updatedAt = null,Object? deletedAt = freezed,}) { | ||||
| @override @pragma('vm:prefer-inline') $Res call({Object? id = null,Object? name = null,Object? nick = null,Object? language = null,Object? isSuperuser = null,Object? profile = null,Object? perkSubscription = freezed,Object? badges = null,Object? createdAt = null,Object? updatedAt = null,Object? deletedAt = freezed,}) { | ||||
|   return _then(_SnAccount( | ||||
| id: null == id ? _self.id : id // ignore: cast_nullable_to_non_nullable | ||||
| as String,name: null == name ? _self.name : name // ignore: cast_nullable_to_non_nullable | ||||
| @@ -299,7 +313,8 @@ as String,nick: null == nick ? _self.nick : nick // ignore: cast_nullable_to_non | ||||
| as String,language: null == language ? _self.language : language // ignore: cast_nullable_to_non_nullable | ||||
| as String,isSuperuser: null == isSuperuser ? _self.isSuperuser : isSuperuser // ignore: cast_nullable_to_non_nullable | ||||
| as bool,profile: null == profile ? _self.profile : profile // ignore: cast_nullable_to_non_nullable | ||||
| as SnAccountProfile,badges: null == badges ? _self._badges : badges // ignore: cast_nullable_to_non_nullable | ||||
| as SnAccountProfile,perkSubscription: freezed == perkSubscription ? _self.perkSubscription : perkSubscription // ignore: cast_nullable_to_non_nullable | ||||
| as SnWalletSubscriptionRef?,badges: null == badges ? _self._badges : badges // ignore: cast_nullable_to_non_nullable | ||||
| as List<SnAccountBadge>,createdAt: null == createdAt ? _self.createdAt : createdAt // ignore: cast_nullable_to_non_nullable | ||||
| as DateTime,updatedAt: null == updatedAt ? _self.updatedAt : updatedAt // ignore: cast_nullable_to_non_nullable | ||||
| as DateTime,deletedAt: freezed == deletedAt ? _self.deletedAt : deletedAt // ignore: cast_nullable_to_non_nullable | ||||
| @@ -316,6 +331,18 @@ $SnAccountProfileCopyWith<$Res> get profile { | ||||
|   return $SnAccountProfileCopyWith<$Res>(_self.profile, (value) { | ||||
|     return _then(_self.copyWith(profile: value)); | ||||
|   }); | ||||
| }/// Create a copy of SnAccount | ||||
| /// with the given fields replaced by the non-null parameter values. | ||||
| @override | ||||
| @pragma('vm:prefer-inline') | ||||
| $SnWalletSubscriptionRefCopyWith<$Res>? get perkSubscription { | ||||
|     if (_self.perkSubscription == null) { | ||||
|     return null; | ||||
|   } | ||||
|  | ||||
|   return $SnWalletSubscriptionRefCopyWith<$Res>(_self.perkSubscription!, (value) { | ||||
|     return _then(_self.copyWith(perkSubscription: value)); | ||||
|   }); | ||||
| } | ||||
| } | ||||
|  | ||||
| @@ -323,7 +350,7 @@ $SnAccountProfileCopyWith<$Res> get profile { | ||||
| /// @nodoc | ||||
| mixin _$SnAccountProfile { | ||||
|  | ||||
|  String get id; String get firstName; String get middleName; String get lastName; String get bio; String get gender; String get pronouns; String get location; String get timeZone; DateTime? get birthday; DateTime? get lastSeenAt; SnAccountBadge? get activeBadge; int get experience; int get level; double get levelingProgress; SnCloudFile? get picture; SnCloudFile? get background; SnVerificationMark? get verification; SnWalletSubscriptionRef? get stellarMembership; DateTime get createdAt; DateTime get updatedAt; DateTime? get deletedAt; | ||||
|  String get id; String get firstName; String get middleName; String get lastName; String get bio; String get gender; String get pronouns; String get location; String get timeZone; DateTime? get birthday; DateTime? get lastSeenAt; SnAccountBadge? get activeBadge; int get experience; int get level; double get levelingProgress; SnCloudFile? get picture; SnCloudFile? get background; SnVerificationMark? get verification; DateTime get createdAt; DateTime get updatedAt; DateTime? get deletedAt; | ||||
| /// Create a copy of SnAccountProfile | ||||
| /// with the given fields replaced by the non-null parameter values. | ||||
| @JsonKey(includeFromJson: false, includeToJson: false) | ||||
| @@ -336,16 +363,16 @@ $SnAccountProfileCopyWith<SnAccountProfile> get copyWith => _$SnAccountProfileCo | ||||
|  | ||||
| @override | ||||
| bool operator ==(Object other) { | ||||
|   return identical(this, other) || (other.runtimeType == runtimeType&&other is SnAccountProfile&&(identical(other.id, id) || other.id == id)&&(identical(other.firstName, firstName) || other.firstName == firstName)&&(identical(other.middleName, middleName) || other.middleName == middleName)&&(identical(other.lastName, lastName) || other.lastName == lastName)&&(identical(other.bio, bio) || other.bio == bio)&&(identical(other.gender, gender) || other.gender == gender)&&(identical(other.pronouns, pronouns) || other.pronouns == pronouns)&&(identical(other.location, location) || other.location == location)&&(identical(other.timeZone, timeZone) || other.timeZone == timeZone)&&(identical(other.birthday, birthday) || other.birthday == birthday)&&(identical(other.lastSeenAt, lastSeenAt) || other.lastSeenAt == lastSeenAt)&&(identical(other.activeBadge, activeBadge) || other.activeBadge == activeBadge)&&(identical(other.experience, experience) || other.experience == experience)&&(identical(other.level, level) || other.level == level)&&(identical(other.levelingProgress, levelingProgress) || other.levelingProgress == levelingProgress)&&(identical(other.picture, picture) || other.picture == picture)&&(identical(other.background, background) || other.background == background)&&(identical(other.verification, verification) || other.verification == verification)&&(identical(other.stellarMembership, stellarMembership) || other.stellarMembership == stellarMembership)&&(identical(other.createdAt, createdAt) || other.createdAt == createdAt)&&(identical(other.updatedAt, updatedAt) || other.updatedAt == updatedAt)&&(identical(other.deletedAt, deletedAt) || other.deletedAt == deletedAt)); | ||||
|   return identical(this, other) || (other.runtimeType == runtimeType&&other is SnAccountProfile&&(identical(other.id, id) || other.id == id)&&(identical(other.firstName, firstName) || other.firstName == firstName)&&(identical(other.middleName, middleName) || other.middleName == middleName)&&(identical(other.lastName, lastName) || other.lastName == lastName)&&(identical(other.bio, bio) || other.bio == bio)&&(identical(other.gender, gender) || other.gender == gender)&&(identical(other.pronouns, pronouns) || other.pronouns == pronouns)&&(identical(other.location, location) || other.location == location)&&(identical(other.timeZone, timeZone) || other.timeZone == timeZone)&&(identical(other.birthday, birthday) || other.birthday == birthday)&&(identical(other.lastSeenAt, lastSeenAt) || other.lastSeenAt == lastSeenAt)&&(identical(other.activeBadge, activeBadge) || other.activeBadge == activeBadge)&&(identical(other.experience, experience) || other.experience == experience)&&(identical(other.level, level) || other.level == level)&&(identical(other.levelingProgress, levelingProgress) || other.levelingProgress == levelingProgress)&&(identical(other.picture, picture) || other.picture == picture)&&(identical(other.background, background) || other.background == background)&&(identical(other.verification, verification) || other.verification == verification)&&(identical(other.createdAt, createdAt) || other.createdAt == createdAt)&&(identical(other.updatedAt, updatedAt) || other.updatedAt == updatedAt)&&(identical(other.deletedAt, deletedAt) || other.deletedAt == deletedAt)); | ||||
| } | ||||
|  | ||||
| @JsonKey(includeFromJson: false, includeToJson: false) | ||||
| @override | ||||
| int get hashCode => Object.hashAll([runtimeType,id,firstName,middleName,lastName,bio,gender,pronouns,location,timeZone,birthday,lastSeenAt,activeBadge,experience,level,levelingProgress,picture,background,verification,stellarMembership,createdAt,updatedAt,deletedAt]); | ||||
| int get hashCode => Object.hashAll([runtimeType,id,firstName,middleName,lastName,bio,gender,pronouns,location,timeZone,birthday,lastSeenAt,activeBadge,experience,level,levelingProgress,picture,background,verification,createdAt,updatedAt,deletedAt]); | ||||
|  | ||||
| @override | ||||
| String toString() { | ||||
|   return 'SnAccountProfile(id: $id, firstName: $firstName, middleName: $middleName, lastName: $lastName, bio: $bio, gender: $gender, pronouns: $pronouns, location: $location, timeZone: $timeZone, birthday: $birthday, lastSeenAt: $lastSeenAt, activeBadge: $activeBadge, experience: $experience, level: $level, levelingProgress: $levelingProgress, picture: $picture, background: $background, verification: $verification, stellarMembership: $stellarMembership, createdAt: $createdAt, updatedAt: $updatedAt, deletedAt: $deletedAt)'; | ||||
|   return 'SnAccountProfile(id: $id, firstName: $firstName, middleName: $middleName, lastName: $lastName, bio: $bio, gender: $gender, pronouns: $pronouns, location: $location, timeZone: $timeZone, birthday: $birthday, lastSeenAt: $lastSeenAt, activeBadge: $activeBadge, experience: $experience, level: $level, levelingProgress: $levelingProgress, picture: $picture, background: $background, verification: $verification, createdAt: $createdAt, updatedAt: $updatedAt, deletedAt: $deletedAt)'; | ||||
| } | ||||
|  | ||||
|  | ||||
| @@ -356,11 +383,11 @@ abstract mixin class $SnAccountProfileCopyWith<$Res>  { | ||||
|   factory $SnAccountProfileCopyWith(SnAccountProfile value, $Res Function(SnAccountProfile) _then) = _$SnAccountProfileCopyWithImpl; | ||||
| @useResult | ||||
| $Res call({ | ||||
|  String id, String firstName, String middleName, String lastName, String bio, String gender, String pronouns, String location, String timeZone, DateTime? birthday, DateTime? lastSeenAt, SnAccountBadge? activeBadge, int experience, int level, double levelingProgress, SnCloudFile? picture, SnCloudFile? background, SnVerificationMark? verification, SnWalletSubscriptionRef? stellarMembership, DateTime createdAt, DateTime updatedAt, DateTime? deletedAt | ||||
|  String id, String firstName, String middleName, String lastName, String bio, String gender, String pronouns, String location, String timeZone, DateTime? birthday, DateTime? lastSeenAt, SnAccountBadge? activeBadge, int experience, int level, double levelingProgress, SnCloudFile? picture, SnCloudFile? background, SnVerificationMark? verification, DateTime createdAt, DateTime updatedAt, DateTime? deletedAt | ||||
| }); | ||||
|  | ||||
|  | ||||
| $SnAccountBadgeCopyWith<$Res>? get activeBadge;$SnCloudFileCopyWith<$Res>? get picture;$SnCloudFileCopyWith<$Res>? get background;$SnVerificationMarkCopyWith<$Res>? get verification;$SnWalletSubscriptionRefCopyWith<$Res>? get stellarMembership; | ||||
| $SnAccountBadgeCopyWith<$Res>? get activeBadge;$SnCloudFileCopyWith<$Res>? get picture;$SnCloudFileCopyWith<$Res>? get background;$SnVerificationMarkCopyWith<$Res>? get verification; | ||||
|  | ||||
| } | ||||
| /// @nodoc | ||||
| @@ -373,7 +400,7 @@ class _$SnAccountProfileCopyWithImpl<$Res> | ||||
|  | ||||
| /// Create a copy of SnAccountProfile | ||||
| /// with the given fields replaced by the non-null parameter values. | ||||
| @pragma('vm:prefer-inline') @override $Res call({Object? id = null,Object? firstName = null,Object? middleName = null,Object? lastName = null,Object? bio = null,Object? gender = null,Object? pronouns = null,Object? location = null,Object? timeZone = null,Object? birthday = freezed,Object? lastSeenAt = freezed,Object? activeBadge = freezed,Object? experience = null,Object? level = null,Object? levelingProgress = null,Object? picture = freezed,Object? background = freezed,Object? verification = freezed,Object? stellarMembership = freezed,Object? createdAt = null,Object? updatedAt = null,Object? deletedAt = freezed,}) { | ||||
| @pragma('vm:prefer-inline') @override $Res call({Object? id = null,Object? firstName = null,Object? middleName = null,Object? lastName = null,Object? bio = null,Object? gender = null,Object? pronouns = null,Object? location = null,Object? timeZone = null,Object? birthday = freezed,Object? lastSeenAt = freezed,Object? activeBadge = freezed,Object? experience = null,Object? level = null,Object? levelingProgress = null,Object? picture = freezed,Object? background = freezed,Object? verification = freezed,Object? createdAt = null,Object? updatedAt = null,Object? deletedAt = freezed,}) { | ||||
|   return _then(_self.copyWith( | ||||
| id: null == id ? _self.id : id // ignore: cast_nullable_to_non_nullable | ||||
| as String,firstName: null == firstName ? _self.firstName : firstName // ignore: cast_nullable_to_non_nullable | ||||
| @@ -393,8 +420,7 @@ as int,levelingProgress: null == levelingProgress ? _self.levelingProgress : lev | ||||
| as double,picture: freezed == picture ? _self.picture : picture // ignore: cast_nullable_to_non_nullable | ||||
| as SnCloudFile?,background: freezed == background ? _self.background : background // ignore: cast_nullable_to_non_nullable | ||||
| as SnCloudFile?,verification: freezed == verification ? _self.verification : verification // ignore: cast_nullable_to_non_nullable | ||||
| as SnVerificationMark?,stellarMembership: freezed == stellarMembership ? _self.stellarMembership : stellarMembership // ignore: cast_nullable_to_non_nullable | ||||
| as SnWalletSubscriptionRef?,createdAt: null == createdAt ? _self.createdAt : createdAt // ignore: cast_nullable_to_non_nullable | ||||
| as SnVerificationMark?,createdAt: null == createdAt ? _self.createdAt : createdAt // ignore: cast_nullable_to_non_nullable | ||||
| as DateTime,updatedAt: null == updatedAt ? _self.updatedAt : updatedAt // ignore: cast_nullable_to_non_nullable | ||||
| as DateTime,deletedAt: freezed == deletedAt ? _self.deletedAt : deletedAt // ignore: cast_nullable_to_non_nullable | ||||
| as DateTime?, | ||||
| @@ -448,18 +474,6 @@ $SnVerificationMarkCopyWith<$Res>? get verification { | ||||
|   return $SnVerificationMarkCopyWith<$Res>(_self.verification!, (value) { | ||||
|     return _then(_self.copyWith(verification: value)); | ||||
|   }); | ||||
| }/// Create a copy of SnAccountProfile | ||||
| /// with the given fields replaced by the non-null parameter values. | ||||
| @override | ||||
| @pragma('vm:prefer-inline') | ||||
| $SnWalletSubscriptionRefCopyWith<$Res>? get stellarMembership { | ||||
|     if (_self.stellarMembership == null) { | ||||
|     return null; | ||||
|   } | ||||
|  | ||||
|   return $SnWalletSubscriptionRefCopyWith<$Res>(_self.stellarMembership!, (value) { | ||||
|     return _then(_self.copyWith(stellarMembership: value)); | ||||
|   }); | ||||
| } | ||||
| } | ||||
|  | ||||
| @@ -539,10 +553,10 @@ return $default(_that);case _: | ||||
| /// } | ||||
| /// ``` | ||||
|  | ||||
| @optionalTypeArgs TResult maybeWhen<TResult extends Object?>(TResult Function( String id,  String firstName,  String middleName,  String lastName,  String bio,  String gender,  String pronouns,  String location,  String timeZone,  DateTime? birthday,  DateTime? lastSeenAt,  SnAccountBadge? activeBadge,  int experience,  int level,  double levelingProgress,  SnCloudFile? picture,  SnCloudFile? background,  SnVerificationMark? verification,  SnWalletSubscriptionRef? stellarMembership,  DateTime createdAt,  DateTime updatedAt,  DateTime? deletedAt)?  $default,{required TResult orElse(),}) {final _that = this; | ||||
| @optionalTypeArgs TResult maybeWhen<TResult extends Object?>(TResult Function( String id,  String firstName,  String middleName,  String lastName,  String bio,  String gender,  String pronouns,  String location,  String timeZone,  DateTime? birthday,  DateTime? lastSeenAt,  SnAccountBadge? activeBadge,  int experience,  int level,  double levelingProgress,  SnCloudFile? picture,  SnCloudFile? background,  SnVerificationMark? verification,  DateTime createdAt,  DateTime updatedAt,  DateTime? deletedAt)?  $default,{required TResult orElse(),}) {final _that = this; | ||||
| switch (_that) { | ||||
| case _SnAccountProfile() when $default != null: | ||||
| return $default(_that.id,_that.firstName,_that.middleName,_that.lastName,_that.bio,_that.gender,_that.pronouns,_that.location,_that.timeZone,_that.birthday,_that.lastSeenAt,_that.activeBadge,_that.experience,_that.level,_that.levelingProgress,_that.picture,_that.background,_that.verification,_that.stellarMembership,_that.createdAt,_that.updatedAt,_that.deletedAt);case _: | ||||
| return $default(_that.id,_that.firstName,_that.middleName,_that.lastName,_that.bio,_that.gender,_that.pronouns,_that.location,_that.timeZone,_that.birthday,_that.lastSeenAt,_that.activeBadge,_that.experience,_that.level,_that.levelingProgress,_that.picture,_that.background,_that.verification,_that.createdAt,_that.updatedAt,_that.deletedAt);case _: | ||||
|   return orElse(); | ||||
|  | ||||
| } | ||||
| @@ -560,10 +574,10 @@ return $default(_that.id,_that.firstName,_that.middleName,_that.lastName,_that.b | ||||
| /// } | ||||
| /// ``` | ||||
|  | ||||
| @optionalTypeArgs TResult when<TResult extends Object?>(TResult Function( String id,  String firstName,  String middleName,  String lastName,  String bio,  String gender,  String pronouns,  String location,  String timeZone,  DateTime? birthday,  DateTime? lastSeenAt,  SnAccountBadge? activeBadge,  int experience,  int level,  double levelingProgress,  SnCloudFile? picture,  SnCloudFile? background,  SnVerificationMark? verification,  SnWalletSubscriptionRef? stellarMembership,  DateTime createdAt,  DateTime updatedAt,  DateTime? deletedAt)  $default,) {final _that = this; | ||||
| @optionalTypeArgs TResult when<TResult extends Object?>(TResult Function( String id,  String firstName,  String middleName,  String lastName,  String bio,  String gender,  String pronouns,  String location,  String timeZone,  DateTime? birthday,  DateTime? lastSeenAt,  SnAccountBadge? activeBadge,  int experience,  int level,  double levelingProgress,  SnCloudFile? picture,  SnCloudFile? background,  SnVerificationMark? verification,  DateTime createdAt,  DateTime updatedAt,  DateTime? deletedAt)  $default,) {final _that = this; | ||||
| switch (_that) { | ||||
| case _SnAccountProfile(): | ||||
| return $default(_that.id,_that.firstName,_that.middleName,_that.lastName,_that.bio,_that.gender,_that.pronouns,_that.location,_that.timeZone,_that.birthday,_that.lastSeenAt,_that.activeBadge,_that.experience,_that.level,_that.levelingProgress,_that.picture,_that.background,_that.verification,_that.stellarMembership,_that.createdAt,_that.updatedAt,_that.deletedAt);} | ||||
| return $default(_that.id,_that.firstName,_that.middleName,_that.lastName,_that.bio,_that.gender,_that.pronouns,_that.location,_that.timeZone,_that.birthday,_that.lastSeenAt,_that.activeBadge,_that.experience,_that.level,_that.levelingProgress,_that.picture,_that.background,_that.verification,_that.createdAt,_that.updatedAt,_that.deletedAt);} | ||||
| } | ||||
| /// A variant of `when` that fallback to returning `null` | ||||
| /// | ||||
| @@ -577,10 +591,10 @@ return $default(_that.id,_that.firstName,_that.middleName,_that.lastName,_that.b | ||||
| /// } | ||||
| /// ``` | ||||
|  | ||||
| @optionalTypeArgs TResult? whenOrNull<TResult extends Object?>(TResult? Function( String id,  String firstName,  String middleName,  String lastName,  String bio,  String gender,  String pronouns,  String location,  String timeZone,  DateTime? birthday,  DateTime? lastSeenAt,  SnAccountBadge? activeBadge,  int experience,  int level,  double levelingProgress,  SnCloudFile? picture,  SnCloudFile? background,  SnVerificationMark? verification,  SnWalletSubscriptionRef? stellarMembership,  DateTime createdAt,  DateTime updatedAt,  DateTime? deletedAt)?  $default,) {final _that = this; | ||||
| @optionalTypeArgs TResult? whenOrNull<TResult extends Object?>(TResult? Function( String id,  String firstName,  String middleName,  String lastName,  String bio,  String gender,  String pronouns,  String location,  String timeZone,  DateTime? birthday,  DateTime? lastSeenAt,  SnAccountBadge? activeBadge,  int experience,  int level,  double levelingProgress,  SnCloudFile? picture,  SnCloudFile? background,  SnVerificationMark? verification,  DateTime createdAt,  DateTime updatedAt,  DateTime? deletedAt)?  $default,) {final _that = this; | ||||
| switch (_that) { | ||||
| case _SnAccountProfile() when $default != null: | ||||
| return $default(_that.id,_that.firstName,_that.middleName,_that.lastName,_that.bio,_that.gender,_that.pronouns,_that.location,_that.timeZone,_that.birthday,_that.lastSeenAt,_that.activeBadge,_that.experience,_that.level,_that.levelingProgress,_that.picture,_that.background,_that.verification,_that.stellarMembership,_that.createdAt,_that.updatedAt,_that.deletedAt);case _: | ||||
| return $default(_that.id,_that.firstName,_that.middleName,_that.lastName,_that.bio,_that.gender,_that.pronouns,_that.location,_that.timeZone,_that.birthday,_that.lastSeenAt,_that.activeBadge,_that.experience,_that.level,_that.levelingProgress,_that.picture,_that.background,_that.verification,_that.createdAt,_that.updatedAt,_that.deletedAt);case _: | ||||
|   return null; | ||||
|  | ||||
| } | ||||
| @@ -592,7 +606,7 @@ return $default(_that.id,_that.firstName,_that.middleName,_that.lastName,_that.b | ||||
| @JsonSerializable() | ||||
|  | ||||
| class _SnAccountProfile implements SnAccountProfile { | ||||
|   const _SnAccountProfile({required this.id, this.firstName = '', this.middleName = '', this.lastName = '', this.bio = '', this.gender = '', this.pronouns = '', this.location = '', this.timeZone = '', this.birthday, this.lastSeenAt, this.activeBadge, required this.experience, required this.level, required this.levelingProgress, required this.picture, required this.background, required this.verification, required this.stellarMembership, required this.createdAt, required this.updatedAt, required this.deletedAt}); | ||||
|   const _SnAccountProfile({required this.id, this.firstName = '', this.middleName = '', this.lastName = '', this.bio = '', this.gender = '', this.pronouns = '', this.location = '', this.timeZone = '', this.birthday, this.lastSeenAt, this.activeBadge, required this.experience, required this.level, required this.levelingProgress, required this.picture, required this.background, required this.verification, required this.createdAt, required this.updatedAt, required this.deletedAt}); | ||||
|   factory _SnAccountProfile.fromJson(Map<String, dynamic> json) => _$SnAccountProfileFromJson(json); | ||||
|  | ||||
| @override final  String id; | ||||
| @@ -613,7 +627,6 @@ class _SnAccountProfile implements SnAccountProfile { | ||||
| @override final  SnCloudFile? picture; | ||||
| @override final  SnCloudFile? background; | ||||
| @override final  SnVerificationMark? verification; | ||||
| @override final  SnWalletSubscriptionRef? stellarMembership; | ||||
| @override final  DateTime createdAt; | ||||
| @override final  DateTime updatedAt; | ||||
| @override final  DateTime? deletedAt; | ||||
| @@ -631,16 +644,16 @@ Map<String, dynamic> toJson() { | ||||
|  | ||||
| @override | ||||
| bool operator ==(Object other) { | ||||
|   return identical(this, other) || (other.runtimeType == runtimeType&&other is _SnAccountProfile&&(identical(other.id, id) || other.id == id)&&(identical(other.firstName, firstName) || other.firstName == firstName)&&(identical(other.middleName, middleName) || other.middleName == middleName)&&(identical(other.lastName, lastName) || other.lastName == lastName)&&(identical(other.bio, bio) || other.bio == bio)&&(identical(other.gender, gender) || other.gender == gender)&&(identical(other.pronouns, pronouns) || other.pronouns == pronouns)&&(identical(other.location, location) || other.location == location)&&(identical(other.timeZone, timeZone) || other.timeZone == timeZone)&&(identical(other.birthday, birthday) || other.birthday == birthday)&&(identical(other.lastSeenAt, lastSeenAt) || other.lastSeenAt == lastSeenAt)&&(identical(other.activeBadge, activeBadge) || other.activeBadge == activeBadge)&&(identical(other.experience, experience) || other.experience == experience)&&(identical(other.level, level) || other.level == level)&&(identical(other.levelingProgress, levelingProgress) || other.levelingProgress == levelingProgress)&&(identical(other.picture, picture) || other.picture == picture)&&(identical(other.background, background) || other.background == background)&&(identical(other.verification, verification) || other.verification == verification)&&(identical(other.stellarMembership, stellarMembership) || other.stellarMembership == stellarMembership)&&(identical(other.createdAt, createdAt) || other.createdAt == createdAt)&&(identical(other.updatedAt, updatedAt) || other.updatedAt == updatedAt)&&(identical(other.deletedAt, deletedAt) || other.deletedAt == deletedAt)); | ||||
|   return identical(this, other) || (other.runtimeType == runtimeType&&other is _SnAccountProfile&&(identical(other.id, id) || other.id == id)&&(identical(other.firstName, firstName) || other.firstName == firstName)&&(identical(other.middleName, middleName) || other.middleName == middleName)&&(identical(other.lastName, lastName) || other.lastName == lastName)&&(identical(other.bio, bio) || other.bio == bio)&&(identical(other.gender, gender) || other.gender == gender)&&(identical(other.pronouns, pronouns) || other.pronouns == pronouns)&&(identical(other.location, location) || other.location == location)&&(identical(other.timeZone, timeZone) || other.timeZone == timeZone)&&(identical(other.birthday, birthday) || other.birthday == birthday)&&(identical(other.lastSeenAt, lastSeenAt) || other.lastSeenAt == lastSeenAt)&&(identical(other.activeBadge, activeBadge) || other.activeBadge == activeBadge)&&(identical(other.experience, experience) || other.experience == experience)&&(identical(other.level, level) || other.level == level)&&(identical(other.levelingProgress, levelingProgress) || other.levelingProgress == levelingProgress)&&(identical(other.picture, picture) || other.picture == picture)&&(identical(other.background, background) || other.background == background)&&(identical(other.verification, verification) || other.verification == verification)&&(identical(other.createdAt, createdAt) || other.createdAt == createdAt)&&(identical(other.updatedAt, updatedAt) || other.updatedAt == updatedAt)&&(identical(other.deletedAt, deletedAt) || other.deletedAt == deletedAt)); | ||||
| } | ||||
|  | ||||
| @JsonKey(includeFromJson: false, includeToJson: false) | ||||
| @override | ||||
| int get hashCode => Object.hashAll([runtimeType,id,firstName,middleName,lastName,bio,gender,pronouns,location,timeZone,birthday,lastSeenAt,activeBadge,experience,level,levelingProgress,picture,background,verification,stellarMembership,createdAt,updatedAt,deletedAt]); | ||||
| int get hashCode => Object.hashAll([runtimeType,id,firstName,middleName,lastName,bio,gender,pronouns,location,timeZone,birthday,lastSeenAt,activeBadge,experience,level,levelingProgress,picture,background,verification,createdAt,updatedAt,deletedAt]); | ||||
|  | ||||
| @override | ||||
| String toString() { | ||||
|   return 'SnAccountProfile(id: $id, firstName: $firstName, middleName: $middleName, lastName: $lastName, bio: $bio, gender: $gender, pronouns: $pronouns, location: $location, timeZone: $timeZone, birthday: $birthday, lastSeenAt: $lastSeenAt, activeBadge: $activeBadge, experience: $experience, level: $level, levelingProgress: $levelingProgress, picture: $picture, background: $background, verification: $verification, stellarMembership: $stellarMembership, createdAt: $createdAt, updatedAt: $updatedAt, deletedAt: $deletedAt)'; | ||||
|   return 'SnAccountProfile(id: $id, firstName: $firstName, middleName: $middleName, lastName: $lastName, bio: $bio, gender: $gender, pronouns: $pronouns, location: $location, timeZone: $timeZone, birthday: $birthday, lastSeenAt: $lastSeenAt, activeBadge: $activeBadge, experience: $experience, level: $level, levelingProgress: $levelingProgress, picture: $picture, background: $background, verification: $verification, createdAt: $createdAt, updatedAt: $updatedAt, deletedAt: $deletedAt)'; | ||||
| } | ||||
|  | ||||
|  | ||||
| @@ -651,11 +664,11 @@ abstract mixin class _$SnAccountProfileCopyWith<$Res> implements $SnAccountProfi | ||||
|   factory _$SnAccountProfileCopyWith(_SnAccountProfile value, $Res Function(_SnAccountProfile) _then) = __$SnAccountProfileCopyWithImpl; | ||||
| @override @useResult | ||||
| $Res call({ | ||||
|  String id, String firstName, String middleName, String lastName, String bio, String gender, String pronouns, String location, String timeZone, DateTime? birthday, DateTime? lastSeenAt, SnAccountBadge? activeBadge, int experience, int level, double levelingProgress, SnCloudFile? picture, SnCloudFile? background, SnVerificationMark? verification, SnWalletSubscriptionRef? stellarMembership, DateTime createdAt, DateTime updatedAt, DateTime? deletedAt | ||||
|  String id, String firstName, String middleName, String lastName, String bio, String gender, String pronouns, String location, String timeZone, DateTime? birthday, DateTime? lastSeenAt, SnAccountBadge? activeBadge, int experience, int level, double levelingProgress, SnCloudFile? picture, SnCloudFile? background, SnVerificationMark? verification, DateTime createdAt, DateTime updatedAt, DateTime? deletedAt | ||||
| }); | ||||
|  | ||||
|  | ||||
| @override $SnAccountBadgeCopyWith<$Res>? get activeBadge;@override $SnCloudFileCopyWith<$Res>? get picture;@override $SnCloudFileCopyWith<$Res>? get background;@override $SnVerificationMarkCopyWith<$Res>? get verification;@override $SnWalletSubscriptionRefCopyWith<$Res>? get stellarMembership; | ||||
| @override $SnAccountBadgeCopyWith<$Res>? get activeBadge;@override $SnCloudFileCopyWith<$Res>? get picture;@override $SnCloudFileCopyWith<$Res>? get background;@override $SnVerificationMarkCopyWith<$Res>? get verification; | ||||
|  | ||||
| } | ||||
| /// @nodoc | ||||
| @@ -668,7 +681,7 @@ class __$SnAccountProfileCopyWithImpl<$Res> | ||||
|  | ||||
| /// Create a copy of SnAccountProfile | ||||
| /// with the given fields replaced by the non-null parameter values. | ||||
| @override @pragma('vm:prefer-inline') $Res call({Object? id = null,Object? firstName = null,Object? middleName = null,Object? lastName = null,Object? bio = null,Object? gender = null,Object? pronouns = null,Object? location = null,Object? timeZone = null,Object? birthday = freezed,Object? lastSeenAt = freezed,Object? activeBadge = freezed,Object? experience = null,Object? level = null,Object? levelingProgress = null,Object? picture = freezed,Object? background = freezed,Object? verification = freezed,Object? stellarMembership = freezed,Object? createdAt = null,Object? updatedAt = null,Object? deletedAt = freezed,}) { | ||||
| @override @pragma('vm:prefer-inline') $Res call({Object? id = null,Object? firstName = null,Object? middleName = null,Object? lastName = null,Object? bio = null,Object? gender = null,Object? pronouns = null,Object? location = null,Object? timeZone = null,Object? birthday = freezed,Object? lastSeenAt = freezed,Object? activeBadge = freezed,Object? experience = null,Object? level = null,Object? levelingProgress = null,Object? picture = freezed,Object? background = freezed,Object? verification = freezed,Object? createdAt = null,Object? updatedAt = null,Object? deletedAt = freezed,}) { | ||||
|   return _then(_SnAccountProfile( | ||||
| id: null == id ? _self.id : id // ignore: cast_nullable_to_non_nullable | ||||
| as String,firstName: null == firstName ? _self.firstName : firstName // ignore: cast_nullable_to_non_nullable | ||||
| @@ -688,8 +701,7 @@ as int,levelingProgress: null == levelingProgress ? _self.levelingProgress : lev | ||||
| as double,picture: freezed == picture ? _self.picture : picture // ignore: cast_nullable_to_non_nullable | ||||
| as SnCloudFile?,background: freezed == background ? _self.background : background // ignore: cast_nullable_to_non_nullable | ||||
| as SnCloudFile?,verification: freezed == verification ? _self.verification : verification // ignore: cast_nullable_to_non_nullable | ||||
| as SnVerificationMark?,stellarMembership: freezed == stellarMembership ? _self.stellarMembership : stellarMembership // ignore: cast_nullable_to_non_nullable | ||||
| as SnWalletSubscriptionRef?,createdAt: null == createdAt ? _self.createdAt : createdAt // ignore: cast_nullable_to_non_nullable | ||||
| as SnVerificationMark?,createdAt: null == createdAt ? _self.createdAt : createdAt // ignore: cast_nullable_to_non_nullable | ||||
| as DateTime,updatedAt: null == updatedAt ? _self.updatedAt : updatedAt // ignore: cast_nullable_to_non_nullable | ||||
| as DateTime,deletedAt: freezed == deletedAt ? _self.deletedAt : deletedAt // ignore: cast_nullable_to_non_nullable | ||||
| as DateTime?, | ||||
| @@ -744,18 +756,6 @@ $SnVerificationMarkCopyWith<$Res>? get verification { | ||||
|   return $SnVerificationMarkCopyWith<$Res>(_self.verification!, (value) { | ||||
|     return _then(_self.copyWith(verification: value)); | ||||
|   }); | ||||
| }/// Create a copy of SnAccountProfile | ||||
| /// with the given fields replaced by the non-null parameter values. | ||||
| @override | ||||
| @pragma('vm:prefer-inline') | ||||
| $SnWalletSubscriptionRefCopyWith<$Res>? get stellarMembership { | ||||
|     if (_self.stellarMembership == null) { | ||||
|     return null; | ||||
|   } | ||||
|  | ||||
|   return $SnWalletSubscriptionRefCopyWith<$Res>(_self.stellarMembership!, (value) { | ||||
|     return _then(_self.copyWith(stellarMembership: value)); | ||||
|   }); | ||||
| } | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -13,6 +13,12 @@ _SnAccount _$SnAccountFromJson(Map<String, dynamic> json) => _SnAccount( | ||||
|   language: json['language'] as String, | ||||
|   isSuperuser: json['is_superuser'] as bool, | ||||
|   profile: SnAccountProfile.fromJson(json['profile'] as Map<String, dynamic>), | ||||
|   perkSubscription: | ||||
|       json['perk_subscription'] == null | ||||
|           ? null | ||||
|           : SnWalletSubscriptionRef.fromJson( | ||||
|             json['perk_subscription'] as Map<String, dynamic>, | ||||
|           ), | ||||
|   badges: | ||||
|       (json['badges'] as List<dynamic>?) | ||||
|           ?.map((e) => SnAccountBadge.fromJson(e as Map<String, dynamic>)) | ||||
| @@ -34,6 +40,7 @@ Map<String, dynamic> _$SnAccountToJson(_SnAccount instance) => | ||||
|       'language': instance.language, | ||||
|       'is_superuser': instance.isSuperuser, | ||||
|       'profile': instance.profile.toJson(), | ||||
|       'perk_subscription': instance.perkSubscription?.toJson(), | ||||
|       'badges': instance.badges.map((e) => e.toJson()).toList(), | ||||
|       'created_at': instance.createdAt.toIso8601String(), | ||||
|       'updated_at': instance.updatedAt.toIso8601String(), | ||||
| @@ -84,12 +91,6 @@ _SnAccountProfile _$SnAccountProfileFromJson(Map<String, dynamic> json) => | ||||
|               : SnVerificationMark.fromJson( | ||||
|                 json['verification'] as Map<String, dynamic>, | ||||
|               ), | ||||
|       stellarMembership: | ||||
|           json['stellar_membership'] == null | ||||
|               ? null | ||||
|               : SnWalletSubscriptionRef.fromJson( | ||||
|                 json['stellar_membership'] as Map<String, dynamic>, | ||||
|               ), | ||||
|       createdAt: DateTime.parse(json['created_at'] as String), | ||||
|       updatedAt: DateTime.parse(json['updated_at'] as String), | ||||
|       deletedAt: | ||||
| @@ -118,7 +119,6 @@ Map<String, dynamic> _$SnAccountProfileToJson(_SnAccountProfile instance) => | ||||
|       'picture': instance.picture?.toJson(), | ||||
|       'background': instance.background?.toJson(), | ||||
|       'verification': instance.verification?.toJson(), | ||||
|       'stellar_membership': instance.stellarMembership?.toJson(), | ||||
|       'created_at': instance.createdAt.toIso8601String(), | ||||
|       'updated_at': instance.updatedAt.toIso8601String(), | ||||
|       'deleted_at': instance.deletedAt?.toIso8601String(), | ||||
|   | ||||
| @@ -4,28 +4,27 @@ import 'package:island/models/webfeed.dart'; | ||||
| import 'package:island/pods/network.dart'; | ||||
|  | ||||
| /// Provider that fetches a single article by its ID | ||||
| final articleDetailProvider = FutureProvider.autoDispose.family<SnWebArticle, String>( | ||||
|   (ref, articleId) async { | ||||
|     final dio = ref.watch(apiClientProvider); | ||||
|      | ||||
|     try { | ||||
|       final response = await dio.get<Map<String, dynamic>>( | ||||
|         '/feeds/articles/$articleId', | ||||
|       ); | ||||
|        | ||||
|       if (response.statusCode == 200 && response.data != null) { | ||||
|         return SnWebArticle.fromJson(response.data!); | ||||
|       } else { | ||||
|         throw Exception('Failed to load article'); | ||||
| final articleDetailProvider = FutureProvider.autoDispose | ||||
|     .family<SnWebArticle, String>((ref, articleId) async { | ||||
|       final dio = ref.watch(apiClientProvider); | ||||
|  | ||||
|       try { | ||||
|         final response = await dio.get<Map<String, dynamic>>( | ||||
|           '/sphere/feeds/articles/$articleId', | ||||
|         ); | ||||
|  | ||||
|         if (response.statusCode == 200 && response.data != null) { | ||||
|           return SnWebArticle.fromJson(response.data!); | ||||
|         } else { | ||||
|           throw Exception('Failed to load article'); | ||||
|         } | ||||
|       } on DioException catch (e) { | ||||
|         if (e.response?.statusCode == 404) { | ||||
|           throw Exception('Article not found'); | ||||
|         } else { | ||||
|           throw Exception('Failed to load article: ${e.message}'); | ||||
|         } | ||||
|       } catch (e) { | ||||
|         throw Exception('Failed to load article: $e'); | ||||
|       } | ||||
|     } on DioException catch (e) { | ||||
|       if (e.response?.statusCode == 404) { | ||||
|         throw Exception('Article not found'); | ||||
|       } else { | ||||
|         throw Exception('Failed to load article: ${e.message}'); | ||||
|       } | ||||
|     } catch (e) { | ||||
|       throw Exception('Failed to load article: $e'); | ||||
|     } | ||||
|   }, | ||||
| ); | ||||
|     }); | ||||
|   | ||||
| @@ -1,13 +1,14 @@ | ||||
| import 'package:island/pods/userinfo.dart'; | ||||
| import 'package:island/screens/chat/chat.dart'; | ||||
| import 'dart:async'; | ||||
| import 'dart:developer'; | ||||
| import 'dart:io'; | ||||
| import 'package:flutter/foundation.dart'; | ||||
| import 'package:flutter_webrtc/flutter_webrtc.dart'; | ||||
| import 'package:island/widgets/chat/call_button.dart'; | ||||
| import 'package:livekit_client/livekit_client.dart'; | ||||
| import 'package:freezed_annotation/freezed_annotation.dart'; | ||||
| import 'dart:async'; | ||||
| import 'package:riverpod_annotation/riverpod_annotation.dart'; | ||||
| import 'package:island/pods/network.dart'; | ||||
| import 'package:island/models/chat.dart'; | ||||
| import 'package:island/pods/websocket.dart'; | ||||
|  | ||||
| part 'call.g.dart'; | ||||
| part 'call.freezed.dart'; | ||||
| @@ -27,6 +28,7 @@ sealed class CallState with _$CallState { | ||||
|     required bool isMicrophoneEnabled, | ||||
|     required bool isCameraEnabled, | ||||
|     required bool isScreenSharing, | ||||
|     required bool isSpeakerphone, | ||||
|     @Default(Duration(seconds: 0)) Duration duration, | ||||
|     String? error, | ||||
|   }) = _CallState; | ||||
| @@ -42,7 +44,8 @@ sealed class CallParticipantLive with _$CallParticipantLive { | ||||
|   }) = _CallParticipantLive; | ||||
|  | ||||
|   bool get isSpeaking => remoteParticipant.isSpeaking; | ||||
|   bool get isMuted => remoteParticipant.isMuted; | ||||
|   bool get isMuted => | ||||
|       remoteParticipant.isMuted || !remoteParticipant.isMicrophoneEnabled(); | ||||
|   bool get isScreenSharing => remoteParticipant.isScreenShareEnabled(); | ||||
|   bool get isScreenSharingWithAudio => | ||||
|       remoteParticipant.isScreenShareAudioEnabled(); | ||||
| @@ -57,13 +60,14 @@ class CallNotifier extends _$CallNotifier { | ||||
|   LocalParticipant? _localParticipant; | ||||
|   List<CallParticipantLive> _participants = []; | ||||
|   final Map<String, CallParticipant> _participantInfoByIdentity = {}; | ||||
|   StreamSubscription? _wsSubscription; | ||||
|   EventsListener? _roomListener; | ||||
|  | ||||
|   List<CallParticipantLive> get participants => | ||||
|       List.unmodifiable(_participants); | ||||
|   LocalParticipant? get localParticipant => _localParticipant; | ||||
|  | ||||
|   Map<String, double> participantsVolumes = {}; | ||||
|  | ||||
|   Timer? _durationTimer; | ||||
|  | ||||
|   Room? get room => _room; | ||||
| @@ -71,36 +75,15 @@ class CallNotifier extends _$CallNotifier { | ||||
|   @override | ||||
|   CallState build() { | ||||
|     // Subscribe to websocket updates | ||||
|     _subscribeToParticipantsUpdate(); | ||||
|     return const CallState( | ||||
|       isConnected: false, | ||||
|       isMicrophoneEnabled: true, | ||||
|       isCameraEnabled: false, | ||||
|       isScreenSharing: false, | ||||
|       isSpeakerphone: true, | ||||
|     ); | ||||
|   } | ||||
|  | ||||
|   void _subscribeToParticipantsUpdate() { | ||||
|     // Only subscribe once | ||||
|     if (_wsSubscription != null) return; | ||||
|     final ws = ref.read(websocketProvider); | ||||
|     _wsSubscription = ws.dataStream.listen((packet) { | ||||
|       if (packet.type == 'call.participants.update' && packet.data != null) { | ||||
|         final participantsData = packet.data!["participants"]; | ||||
|         if (participantsData is List) { | ||||
|           final parsed = | ||||
|               participantsData | ||||
|                   .map( | ||||
|                     (e) => | ||||
|                         CallParticipant.fromJson(Map<String, dynamic>.from(e)), | ||||
|                   ) | ||||
|                   .toList(); | ||||
|           _updateLiveParticipants(parsed); | ||||
|         } | ||||
|       } | ||||
|     }); | ||||
|   } | ||||
|  | ||||
|   void _initRoomListeners() { | ||||
|     if (_room == null) return; | ||||
|     _roomListener?.dispose(); | ||||
| @@ -143,8 +126,6 @@ class CallNotifier extends _$CallNotifier { | ||||
|               identity: remote.identity, | ||||
|               name: remote.identity, | ||||
|               joinedAt: DateTime.now(), | ||||
|               accountId: null, | ||||
|               profile: null, | ||||
|             ); | ||||
|         return CallParticipantLive( | ||||
|           participant: match, | ||||
| @@ -169,16 +150,12 @@ class CallNotifier extends _$CallNotifier { | ||||
|       if (idx != -1) return participants[idx]; | ||||
|     } | ||||
|  | ||||
|     final userInfo = ref.read(userInfoProvider); | ||||
|     final roomIdentity = ref.read(chatroomIdentityProvider(_roomId)); | ||||
|     // Otherwise, use info from the identity map or fallback to minimal | ||||
|     return _participantInfoByIdentity[_localParticipant!.identity] ?? | ||||
|         CallParticipant( | ||||
|           identity: _localParticipant!.identity, | ||||
|           name: _localParticipant!.identity, | ||||
|           joinedAt: DateTime.now(), | ||||
|           accountId: userInfo.value?.id, | ||||
|           profile: roomIdentity.value, | ||||
|         ); | ||||
|   } | ||||
|  | ||||
| @@ -205,6 +182,7 @@ class CallNotifier extends _$CallNotifier { | ||||
|           remoteParticipant: _localParticipant!, | ||||
|         ), | ||||
|       ); | ||||
|       state = state.copyWith(); | ||||
|     } | ||||
|     // Add remote participants | ||||
|     _participants.addAll( | ||||
| @@ -233,7 +211,13 @@ class CallNotifier extends _$CallNotifier { | ||||
|  | ||||
|   Future<void> joinRoom(String roomId) async { | ||||
|     if (_roomId == roomId && _room != null) { | ||||
|       log('[Call] Call skipped. Already has data'); | ||||
|       return; | ||||
|     } else if (_room != null) { | ||||
|       if (!_room!.isDisposed && | ||||
|           _room!.connectionState != ConnectionState.disconnected) { | ||||
|         throw Exception('Call already connected'); | ||||
|       } | ||||
|     } | ||||
|     _roomId = roomId; | ||||
|     if (_room != null) { | ||||
| @@ -264,7 +248,8 @@ class CallNotifier extends _$CallNotifier { | ||||
|             duration: Duration( | ||||
|               milliseconds: | ||||
|                   (DateTime.now().millisecondsSinceEpoch - | ||||
|                       (ongoingCall?.createdAt.millisecondsSinceEpoch ?? 0)), | ||||
|                       (ongoingCall?.createdAt.millisecondsSinceEpoch ?? | ||||
|                           DateTime.now().millisecondsSinceEpoch)), | ||||
|             ), | ||||
|           ); | ||||
|         }); | ||||
| @@ -286,6 +271,10 @@ class CallNotifier extends _$CallNotifier { | ||||
|         _initRoomListeners(); | ||||
|         _updateLiveParticipants(participants); | ||||
|  | ||||
|         if (!kIsWeb && (Platform.isIOS || Platform.isAndroid)) { | ||||
|           Hardware.instance.setSpeakerphoneOn(true); | ||||
|         } | ||||
|  | ||||
|         // Listen for connection updates | ||||
|         _room!.addListener(() { | ||||
|           state = state.copyWith( | ||||
| @@ -318,6 +307,7 @@ class CallNotifier extends _$CallNotifier { | ||||
|           stopOnMute: autostop, | ||||
|         ); | ||||
|       } | ||||
|       state = state.copyWith(); | ||||
|     } | ||||
|   } | ||||
|  | ||||
| @@ -326,6 +316,7 @@ class CallNotifier extends _$CallNotifier { | ||||
|       final target = !_localParticipant!.isCameraEnabled(); | ||||
|       state = state.copyWith(isCameraEnabled: target); | ||||
|       await _localParticipant!.setCameraEnabled(target); | ||||
|       state = state.copyWith(); | ||||
|     } | ||||
|   } | ||||
|  | ||||
| @@ -334,9 +325,16 @@ class CallNotifier extends _$CallNotifier { | ||||
|       final target = !_localParticipant!.isScreenShareEnabled(); | ||||
|       state = state.copyWith(isScreenSharing: target); | ||||
|       await _localParticipant!.setScreenShareEnabled(target); | ||||
|       state = state.copyWith(); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   Future<void> toggleSpeakerphone() async { | ||||
|     state = state.copyWith(isSpeakerphone: !state.isSpeakerphone); | ||||
|     await Hardware.instance.setSpeakerphoneOn(state.isSpeakerphone); | ||||
|     state = state.copyWith(); | ||||
|   } | ||||
|  | ||||
|   Future<void> disconnect() async { | ||||
|     if (_room != null) { | ||||
|       await _room!.disconnect(); | ||||
| @@ -349,11 +347,39 @@ class CallNotifier extends _$CallNotifier { | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   void setParticipantVolume(CallParticipantLive live, double volume) { | ||||
|     if (participantsVolumes[live.remoteParticipant.sid] == null) { | ||||
|       participantsVolumes[live.remoteParticipant.sid] = 1; | ||||
|     } | ||||
|     Helper.setVolume( | ||||
|       volume, | ||||
|       live | ||||
|           .remoteParticipant | ||||
|           .audioTrackPublications | ||||
|           .first | ||||
|           .track! | ||||
|           .mediaStreamTrack, | ||||
|     ); | ||||
|     participantsVolumes[live.remoteParticipant.sid] = volume; | ||||
|   } | ||||
|  | ||||
|   double getParticipantVolume(CallParticipantLive live) { | ||||
|     return participantsVolumes[live.remoteParticipant.sid] ?? 1; | ||||
|   } | ||||
|  | ||||
|   void dispose() { | ||||
|     _wsSubscription?.cancel(); | ||||
|     state = state.copyWith( | ||||
|       error: null, | ||||
|       isConnected: false, | ||||
|       isMicrophoneEnabled: false, | ||||
|       isCameraEnabled: false, | ||||
|       isScreenSharing: false, | ||||
|     ); | ||||
|     _roomListener?.dispose(); | ||||
|     _room?.removeListener(_onRoomChange); | ||||
|     _room?.dispose(); | ||||
|     _durationTimer?.cancel(); | ||||
|     _roomId = null; | ||||
|     participantsVolumes = {}; | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -12,9 +12,9 @@ part of 'call.dart'; | ||||
| // dart format off | ||||
| T _$identity<T>(T value) => value; | ||||
| /// @nodoc | ||||
| mixin _$CallState { | ||||
| mixin _$CallState implements DiagnosticableTreeMixin { | ||||
|  | ||||
|  bool get isConnected; bool get isMicrophoneEnabled; bool get isCameraEnabled; bool get isScreenSharing; Duration get duration; String? get error; | ||||
|  bool get isConnected; bool get isMicrophoneEnabled; bool get isCameraEnabled; bool get isScreenSharing; bool get isSpeakerphone; Duration get duration; String? get error; | ||||
| /// Create a copy of CallState | ||||
| /// with the given fields replaced by the non-null parameter values. | ||||
| @JsonKey(includeFromJson: false, includeToJson: false) | ||||
| @@ -22,19 +22,25 @@ mixin _$CallState { | ||||
| $CallStateCopyWith<CallState> get copyWith => _$CallStateCopyWithImpl<CallState>(this as CallState, _$identity); | ||||
|  | ||||
|  | ||||
| @override | ||||
| void debugFillProperties(DiagnosticPropertiesBuilder properties) { | ||||
|   properties | ||||
|     ..add(DiagnosticsProperty('type', 'CallState')) | ||||
|     ..add(DiagnosticsProperty('isConnected', isConnected))..add(DiagnosticsProperty('isMicrophoneEnabled', isMicrophoneEnabled))..add(DiagnosticsProperty('isCameraEnabled', isCameraEnabled))..add(DiagnosticsProperty('isScreenSharing', isScreenSharing))..add(DiagnosticsProperty('isSpeakerphone', isSpeakerphone))..add(DiagnosticsProperty('duration', duration))..add(DiagnosticsProperty('error', error)); | ||||
| } | ||||
|  | ||||
| @override | ||||
| bool operator ==(Object other) { | ||||
|   return identical(this, other) || (other.runtimeType == runtimeType&&other is CallState&&(identical(other.isConnected, isConnected) || other.isConnected == isConnected)&&(identical(other.isMicrophoneEnabled, isMicrophoneEnabled) || other.isMicrophoneEnabled == isMicrophoneEnabled)&&(identical(other.isCameraEnabled, isCameraEnabled) || other.isCameraEnabled == isCameraEnabled)&&(identical(other.isScreenSharing, isScreenSharing) || other.isScreenSharing == isScreenSharing)&&(identical(other.duration, duration) || other.duration == duration)&&(identical(other.error, error) || other.error == error)); | ||||
|   return identical(this, other) || (other.runtimeType == runtimeType&&other is CallState&&(identical(other.isConnected, isConnected) || other.isConnected == isConnected)&&(identical(other.isMicrophoneEnabled, isMicrophoneEnabled) || other.isMicrophoneEnabled == isMicrophoneEnabled)&&(identical(other.isCameraEnabled, isCameraEnabled) || other.isCameraEnabled == isCameraEnabled)&&(identical(other.isScreenSharing, isScreenSharing) || other.isScreenSharing == isScreenSharing)&&(identical(other.isSpeakerphone, isSpeakerphone) || other.isSpeakerphone == isSpeakerphone)&&(identical(other.duration, duration) || other.duration == duration)&&(identical(other.error, error) || other.error == error)); | ||||
| } | ||||
|  | ||||
|  | ||||
| @override | ||||
| int get hashCode => Object.hash(runtimeType,isConnected,isMicrophoneEnabled,isCameraEnabled,isScreenSharing,duration,error); | ||||
| int get hashCode => Object.hash(runtimeType,isConnected,isMicrophoneEnabled,isCameraEnabled,isScreenSharing,isSpeakerphone,duration,error); | ||||
|  | ||||
| @override | ||||
| String toString() { | ||||
|   return 'CallState(isConnected: $isConnected, isMicrophoneEnabled: $isMicrophoneEnabled, isCameraEnabled: $isCameraEnabled, isScreenSharing: $isScreenSharing, duration: $duration, error: $error)'; | ||||
| String toString({ DiagnosticLevel minLevel = DiagnosticLevel.info }) { | ||||
|   return 'CallState(isConnected: $isConnected, isMicrophoneEnabled: $isMicrophoneEnabled, isCameraEnabled: $isCameraEnabled, isScreenSharing: $isScreenSharing, isSpeakerphone: $isSpeakerphone, duration: $duration, error: $error)'; | ||||
| } | ||||
|  | ||||
|  | ||||
| @@ -45,7 +51,7 @@ abstract mixin class $CallStateCopyWith<$Res>  { | ||||
|   factory $CallStateCopyWith(CallState value, $Res Function(CallState) _then) = _$CallStateCopyWithImpl; | ||||
| @useResult | ||||
| $Res call({ | ||||
|  bool isConnected, bool isMicrophoneEnabled, bool isCameraEnabled, bool isScreenSharing, Duration duration, String? error | ||||
|  bool isConnected, bool isMicrophoneEnabled, bool isCameraEnabled, bool isScreenSharing, bool isSpeakerphone, Duration duration, String? error | ||||
| }); | ||||
|  | ||||
|  | ||||
| @@ -62,12 +68,13 @@ class _$CallStateCopyWithImpl<$Res> | ||||
|  | ||||
| /// Create a copy of CallState | ||||
| /// with the given fields replaced by the non-null parameter values. | ||||
| @pragma('vm:prefer-inline') @override $Res call({Object? isConnected = null,Object? isMicrophoneEnabled = null,Object? isCameraEnabled = null,Object? isScreenSharing = null,Object? duration = null,Object? error = freezed,}) { | ||||
| @pragma('vm:prefer-inline') @override $Res call({Object? isConnected = null,Object? isMicrophoneEnabled = null,Object? isCameraEnabled = null,Object? isScreenSharing = null,Object? isSpeakerphone = null,Object? duration = null,Object? error = freezed,}) { | ||||
|   return _then(_self.copyWith( | ||||
| isConnected: null == isConnected ? _self.isConnected : isConnected // ignore: cast_nullable_to_non_nullable | ||||
| as bool,isMicrophoneEnabled: null == isMicrophoneEnabled ? _self.isMicrophoneEnabled : isMicrophoneEnabled // ignore: cast_nullable_to_non_nullable | ||||
| as bool,isCameraEnabled: null == isCameraEnabled ? _self.isCameraEnabled : isCameraEnabled // ignore: cast_nullable_to_non_nullable | ||||
| as bool,isScreenSharing: null == isScreenSharing ? _self.isScreenSharing : isScreenSharing // ignore: cast_nullable_to_non_nullable | ||||
| as bool,isSpeakerphone: null == isSpeakerphone ? _self.isSpeakerphone : isSpeakerphone // ignore: cast_nullable_to_non_nullable | ||||
| as bool,duration: null == duration ? _self.duration : duration // ignore: cast_nullable_to_non_nullable | ||||
| as Duration,error: freezed == error ? _self.error : error // ignore: cast_nullable_to_non_nullable | ||||
| as String?, | ||||
| @@ -152,10 +159,10 @@ return $default(_that);case _: | ||||
| /// } | ||||
| /// ``` | ||||
|  | ||||
| @optionalTypeArgs TResult maybeWhen<TResult extends Object?>(TResult Function( bool isConnected,  bool isMicrophoneEnabled,  bool isCameraEnabled,  bool isScreenSharing,  Duration duration,  String? error)?  $default,{required TResult orElse(),}) {final _that = this; | ||||
| @optionalTypeArgs TResult maybeWhen<TResult extends Object?>(TResult Function( bool isConnected,  bool isMicrophoneEnabled,  bool isCameraEnabled,  bool isScreenSharing,  bool isSpeakerphone,  Duration duration,  String? error)?  $default,{required TResult orElse(),}) {final _that = this; | ||||
| switch (_that) { | ||||
| case _CallState() when $default != null: | ||||
| return $default(_that.isConnected,_that.isMicrophoneEnabled,_that.isCameraEnabled,_that.isScreenSharing,_that.duration,_that.error);case _: | ||||
| return $default(_that.isConnected,_that.isMicrophoneEnabled,_that.isCameraEnabled,_that.isScreenSharing,_that.isSpeakerphone,_that.duration,_that.error);case _: | ||||
|   return orElse(); | ||||
|  | ||||
| } | ||||
| @@ -173,10 +180,10 @@ return $default(_that.isConnected,_that.isMicrophoneEnabled,_that.isCameraEnable | ||||
| /// } | ||||
| /// ``` | ||||
|  | ||||
| @optionalTypeArgs TResult when<TResult extends Object?>(TResult Function( bool isConnected,  bool isMicrophoneEnabled,  bool isCameraEnabled,  bool isScreenSharing,  Duration duration,  String? error)  $default,) {final _that = this; | ||||
| @optionalTypeArgs TResult when<TResult extends Object?>(TResult Function( bool isConnected,  bool isMicrophoneEnabled,  bool isCameraEnabled,  bool isScreenSharing,  bool isSpeakerphone,  Duration duration,  String? error)  $default,) {final _that = this; | ||||
| switch (_that) { | ||||
| case _CallState(): | ||||
| return $default(_that.isConnected,_that.isMicrophoneEnabled,_that.isCameraEnabled,_that.isScreenSharing,_that.duration,_that.error);} | ||||
| return $default(_that.isConnected,_that.isMicrophoneEnabled,_that.isCameraEnabled,_that.isScreenSharing,_that.isSpeakerphone,_that.duration,_that.error);} | ||||
| } | ||||
| /// A variant of `when` that fallback to returning `null` | ||||
| /// | ||||
| @@ -190,10 +197,10 @@ return $default(_that.isConnected,_that.isMicrophoneEnabled,_that.isCameraEnable | ||||
| /// } | ||||
| /// ``` | ||||
|  | ||||
| @optionalTypeArgs TResult? whenOrNull<TResult extends Object?>(TResult? Function( bool isConnected,  bool isMicrophoneEnabled,  bool isCameraEnabled,  bool isScreenSharing,  Duration duration,  String? error)?  $default,) {final _that = this; | ||||
| @optionalTypeArgs TResult? whenOrNull<TResult extends Object?>(TResult? Function( bool isConnected,  bool isMicrophoneEnabled,  bool isCameraEnabled,  bool isScreenSharing,  bool isSpeakerphone,  Duration duration,  String? error)?  $default,) {final _that = this; | ||||
| switch (_that) { | ||||
| case _CallState() when $default != null: | ||||
| return $default(_that.isConnected,_that.isMicrophoneEnabled,_that.isCameraEnabled,_that.isScreenSharing,_that.duration,_that.error);case _: | ||||
| return $default(_that.isConnected,_that.isMicrophoneEnabled,_that.isCameraEnabled,_that.isScreenSharing,_that.isSpeakerphone,_that.duration,_that.error);case _: | ||||
|   return null; | ||||
|  | ||||
| } | ||||
| @@ -204,14 +211,15 @@ return $default(_that.isConnected,_that.isMicrophoneEnabled,_that.isCameraEnable | ||||
| /// @nodoc | ||||
|  | ||||
|  | ||||
| class _CallState implements CallState { | ||||
|   const _CallState({required this.isConnected, required this.isMicrophoneEnabled, required this.isCameraEnabled, required this.isScreenSharing, this.duration = const Duration(seconds: 0), this.error}); | ||||
| class _CallState with DiagnosticableTreeMixin implements CallState { | ||||
|   const _CallState({required this.isConnected, required this.isMicrophoneEnabled, required this.isCameraEnabled, required this.isScreenSharing, required this.isSpeakerphone, this.duration = const Duration(seconds: 0), this.error}); | ||||
|    | ||||
|  | ||||
| @override final  bool isConnected; | ||||
| @override final  bool isMicrophoneEnabled; | ||||
| @override final  bool isCameraEnabled; | ||||
| @override final  bool isScreenSharing; | ||||
| @override final  bool isSpeakerphone; | ||||
| @override@JsonKey() final  Duration duration; | ||||
| @override final  String? error; | ||||
|  | ||||
| @@ -222,19 +230,25 @@ class _CallState implements CallState { | ||||
| _$CallStateCopyWith<_CallState> get copyWith => __$CallStateCopyWithImpl<_CallState>(this, _$identity); | ||||
|  | ||||
|  | ||||
| @override | ||||
| void debugFillProperties(DiagnosticPropertiesBuilder properties) { | ||||
|   properties | ||||
|     ..add(DiagnosticsProperty('type', 'CallState')) | ||||
|     ..add(DiagnosticsProperty('isConnected', isConnected))..add(DiagnosticsProperty('isMicrophoneEnabled', isMicrophoneEnabled))..add(DiagnosticsProperty('isCameraEnabled', isCameraEnabled))..add(DiagnosticsProperty('isScreenSharing', isScreenSharing))..add(DiagnosticsProperty('isSpeakerphone', isSpeakerphone))..add(DiagnosticsProperty('duration', duration))..add(DiagnosticsProperty('error', error)); | ||||
| } | ||||
|  | ||||
| @override | ||||
| bool operator ==(Object other) { | ||||
|   return identical(this, other) || (other.runtimeType == runtimeType&&other is _CallState&&(identical(other.isConnected, isConnected) || other.isConnected == isConnected)&&(identical(other.isMicrophoneEnabled, isMicrophoneEnabled) || other.isMicrophoneEnabled == isMicrophoneEnabled)&&(identical(other.isCameraEnabled, isCameraEnabled) || other.isCameraEnabled == isCameraEnabled)&&(identical(other.isScreenSharing, isScreenSharing) || other.isScreenSharing == isScreenSharing)&&(identical(other.duration, duration) || other.duration == duration)&&(identical(other.error, error) || other.error == error)); | ||||
|   return identical(this, other) || (other.runtimeType == runtimeType&&other is _CallState&&(identical(other.isConnected, isConnected) || other.isConnected == isConnected)&&(identical(other.isMicrophoneEnabled, isMicrophoneEnabled) || other.isMicrophoneEnabled == isMicrophoneEnabled)&&(identical(other.isCameraEnabled, isCameraEnabled) || other.isCameraEnabled == isCameraEnabled)&&(identical(other.isScreenSharing, isScreenSharing) || other.isScreenSharing == isScreenSharing)&&(identical(other.isSpeakerphone, isSpeakerphone) || other.isSpeakerphone == isSpeakerphone)&&(identical(other.duration, duration) || other.duration == duration)&&(identical(other.error, error) || other.error == error)); | ||||
| } | ||||
|  | ||||
|  | ||||
| @override | ||||
| int get hashCode => Object.hash(runtimeType,isConnected,isMicrophoneEnabled,isCameraEnabled,isScreenSharing,duration,error); | ||||
| int get hashCode => Object.hash(runtimeType,isConnected,isMicrophoneEnabled,isCameraEnabled,isScreenSharing,isSpeakerphone,duration,error); | ||||
|  | ||||
| @override | ||||
| String toString() { | ||||
|   return 'CallState(isConnected: $isConnected, isMicrophoneEnabled: $isMicrophoneEnabled, isCameraEnabled: $isCameraEnabled, isScreenSharing: $isScreenSharing, duration: $duration, error: $error)'; | ||||
| String toString({ DiagnosticLevel minLevel = DiagnosticLevel.info }) { | ||||
|   return 'CallState(isConnected: $isConnected, isMicrophoneEnabled: $isMicrophoneEnabled, isCameraEnabled: $isCameraEnabled, isScreenSharing: $isScreenSharing, isSpeakerphone: $isSpeakerphone, duration: $duration, error: $error)'; | ||||
| } | ||||
|  | ||||
|  | ||||
| @@ -245,7 +259,7 @@ abstract mixin class _$CallStateCopyWith<$Res> implements $CallStateCopyWith<$Re | ||||
|   factory _$CallStateCopyWith(_CallState value, $Res Function(_CallState) _then) = __$CallStateCopyWithImpl; | ||||
| @override @useResult | ||||
| $Res call({ | ||||
|  bool isConnected, bool isMicrophoneEnabled, bool isCameraEnabled, bool isScreenSharing, Duration duration, String? error | ||||
|  bool isConnected, bool isMicrophoneEnabled, bool isCameraEnabled, bool isScreenSharing, bool isSpeakerphone, Duration duration, String? error | ||||
| }); | ||||
|  | ||||
|  | ||||
| @@ -262,12 +276,13 @@ class __$CallStateCopyWithImpl<$Res> | ||||
|  | ||||
| /// Create a copy of CallState | ||||
| /// with the given fields replaced by the non-null parameter values. | ||||
| @override @pragma('vm:prefer-inline') $Res call({Object? isConnected = null,Object? isMicrophoneEnabled = null,Object? isCameraEnabled = null,Object? isScreenSharing = null,Object? duration = null,Object? error = freezed,}) { | ||||
| @override @pragma('vm:prefer-inline') $Res call({Object? isConnected = null,Object? isMicrophoneEnabled = null,Object? isCameraEnabled = null,Object? isScreenSharing = null,Object? isSpeakerphone = null,Object? duration = null,Object? error = freezed,}) { | ||||
|   return _then(_CallState( | ||||
| isConnected: null == isConnected ? _self.isConnected : isConnected // ignore: cast_nullable_to_non_nullable | ||||
| as bool,isMicrophoneEnabled: null == isMicrophoneEnabled ? _self.isMicrophoneEnabled : isMicrophoneEnabled // ignore: cast_nullable_to_non_nullable | ||||
| as bool,isCameraEnabled: null == isCameraEnabled ? _self.isCameraEnabled : isCameraEnabled // ignore: cast_nullable_to_non_nullable | ||||
| as bool,isScreenSharing: null == isScreenSharing ? _self.isScreenSharing : isScreenSharing // ignore: cast_nullable_to_non_nullable | ||||
| as bool,isSpeakerphone: null == isSpeakerphone ? _self.isSpeakerphone : isSpeakerphone // ignore: cast_nullable_to_non_nullable | ||||
| as bool,duration: null == duration ? _self.duration : duration // ignore: cast_nullable_to_non_nullable | ||||
| as Duration,error: freezed == error ? _self.error : error // ignore: cast_nullable_to_non_nullable | ||||
| as String?, | ||||
| @@ -278,7 +293,7 @@ as String?, | ||||
| } | ||||
|  | ||||
| /// @nodoc | ||||
| mixin _$CallParticipantLive { | ||||
| mixin _$CallParticipantLive implements DiagnosticableTreeMixin { | ||||
|  | ||||
|  CallParticipant get participant; Participant get remoteParticipant; | ||||
| /// Create a copy of CallParticipantLive | ||||
| @@ -288,6 +303,12 @@ mixin _$CallParticipantLive { | ||||
| $CallParticipantLiveCopyWith<CallParticipantLive> get copyWith => _$CallParticipantLiveCopyWithImpl<CallParticipantLive>(this as CallParticipantLive, _$identity); | ||||
|  | ||||
|  | ||||
| @override | ||||
| void debugFillProperties(DiagnosticPropertiesBuilder properties) { | ||||
|   properties | ||||
|     ..add(DiagnosticsProperty('type', 'CallParticipantLive')) | ||||
|     ..add(DiagnosticsProperty('participant', participant))..add(DiagnosticsProperty('remoteParticipant', remoteParticipant)); | ||||
| } | ||||
|  | ||||
| @override | ||||
| bool operator ==(Object other) { | ||||
| @@ -299,7 +320,7 @@ bool operator ==(Object other) { | ||||
| int get hashCode => Object.hash(runtimeType,participant,remoteParticipant); | ||||
|  | ||||
| @override | ||||
| String toString() { | ||||
| String toString({ DiagnosticLevel minLevel = DiagnosticLevel.info }) { | ||||
|   return 'CallParticipantLive(participant: $participant, remoteParticipant: $remoteParticipant)'; | ||||
| } | ||||
|  | ||||
| @@ -475,7 +496,7 @@ return $default(_that.participant,_that.remoteParticipant);case _: | ||||
| /// @nodoc | ||||
|  | ||||
|  | ||||
| class _CallParticipantLive extends CallParticipantLive { | ||||
| class _CallParticipantLive extends CallParticipantLive with DiagnosticableTreeMixin { | ||||
|   const _CallParticipantLive({required this.participant, required this.remoteParticipant}): super._(); | ||||
|    | ||||
|  | ||||
| @@ -489,6 +510,12 @@ class _CallParticipantLive extends CallParticipantLive { | ||||
| _$CallParticipantLiveCopyWith<_CallParticipantLive> get copyWith => __$CallParticipantLiveCopyWithImpl<_CallParticipantLive>(this, _$identity); | ||||
|  | ||||
|  | ||||
| @override | ||||
| void debugFillProperties(DiagnosticPropertiesBuilder properties) { | ||||
|   properties | ||||
|     ..add(DiagnosticsProperty('type', 'CallParticipantLive')) | ||||
|     ..add(DiagnosticsProperty('participant', participant))..add(DiagnosticsProperty('remoteParticipant', remoteParticipant)); | ||||
| } | ||||
|  | ||||
| @override | ||||
| bool operator ==(Object other) { | ||||
| @@ -500,7 +527,7 @@ bool operator ==(Object other) { | ||||
| int get hashCode => Object.hash(runtimeType,participant,remoteParticipant); | ||||
|  | ||||
| @override | ||||
| String toString() { | ||||
| String toString({ DiagnosticLevel minLevel = DiagnosticLevel.info }) { | ||||
|   return 'CallParticipantLive(participant: $participant, remoteParticipant: $remoteParticipant)'; | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -6,7 +6,7 @@ part of 'call.dart'; | ||||
| // RiverpodGenerator | ||||
| // ************************************************************************** | ||||
|  | ||||
| String _$callNotifierHash() => r'107174cd6cfab6bfafe44f8c4a72a67bcb93217b'; | ||||
| String _$callNotifierHash() => r'18fb807f067eecd3ea42631c1426c3e5f1fb4280'; | ||||
|  | ||||
| /// See also [CallNotifier]. | ||||
| @ProviderFor(CallNotifier) | ||||
|   | ||||
							
								
								
									
										38
									
								
								lib/pods/translate.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										38
									
								
								lib/pods/translate.dart
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,38 @@ | ||||
| import 'dart:convert'; | ||||
| import 'dart:developer'; | ||||
|  | ||||
| import 'package:freezed_annotation/freezed_annotation.dart'; | ||||
| import 'package:hooks_riverpod/hooks_riverpod.dart'; | ||||
| import 'package:island/pods/network.dart'; | ||||
| import 'package:riverpod_annotation/riverpod_annotation.dart'; | ||||
| import 'package:flutter_langdetect/flutter_langdetect.dart' as langdetect; | ||||
|  | ||||
| part 'translate.freezed.dart'; | ||||
| part 'translate.g.dart'; | ||||
|  | ||||
| @freezed | ||||
| sealed class TranslateQuery with _$TranslateQuery { | ||||
|   const factory TranslateQuery({required String text, required String lang}) = | ||||
|       _TranslateQuery; | ||||
| } | ||||
|  | ||||
| @riverpod | ||||
| Future<String> translateString(Ref ref, TranslateQuery query) async { | ||||
|   final client = ref.watch(apiClientProvider); | ||||
|   final response = await client.post( | ||||
|     '/sphere/translate', | ||||
|     queryParameters: {'to': query.lang}, | ||||
|     data: jsonEncode(query.text), | ||||
|   ); | ||||
|   return response.data as String; | ||||
| } | ||||
|  | ||||
| @riverpod | ||||
| String? detectStringLanguage(Ref ref, String text) { | ||||
|   try { | ||||
|     return langdetect.detectLangs(text).firstOrNull?.lang; | ||||
|   } catch (err) { | ||||
|     log('[Language] Unable to detect text\'s language: $text'); | ||||
|     return null; | ||||
|   } | ||||
| } | ||||
							
								
								
									
										268
									
								
								lib/pods/translate.freezed.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										268
									
								
								lib/pods/translate.freezed.dart
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,268 @@ | ||||
| // GENERATED CODE - DO NOT MODIFY BY HAND | ||||
| // coverage:ignore-file | ||||
| // ignore_for_file: type=lint | ||||
| // ignore_for_file: unused_element, deprecated_member_use, deprecated_member_use_from_same_package, use_function_type_syntax_for_parameters, unnecessary_const, avoid_init_to_null, invalid_override_different_default_values_named, prefer_expression_function_bodies, annotate_overrides, invalid_annotation_target, unnecessary_question_mark | ||||
|  | ||||
| part of 'translate.dart'; | ||||
|  | ||||
| // ************************************************************************** | ||||
| // FreezedGenerator | ||||
| // ************************************************************************** | ||||
|  | ||||
| // dart format off | ||||
| T _$identity<T>(T value) => value; | ||||
| /// @nodoc | ||||
| mixin _$TranslateQuery { | ||||
|  | ||||
|  String get text; String get lang; | ||||
| /// Create a copy of TranslateQuery | ||||
| /// with the given fields replaced by the non-null parameter values. | ||||
| @JsonKey(includeFromJson: false, includeToJson: false) | ||||
| @pragma('vm:prefer-inline') | ||||
| $TranslateQueryCopyWith<TranslateQuery> get copyWith => _$TranslateQueryCopyWithImpl<TranslateQuery>(this as TranslateQuery, _$identity); | ||||
|  | ||||
|  | ||||
|  | ||||
| @override | ||||
| bool operator ==(Object other) { | ||||
|   return identical(this, other) || (other.runtimeType == runtimeType&&other is TranslateQuery&&(identical(other.text, text) || other.text == text)&&(identical(other.lang, lang) || other.lang == lang)); | ||||
| } | ||||
|  | ||||
|  | ||||
| @override | ||||
| int get hashCode => Object.hash(runtimeType,text,lang); | ||||
|  | ||||
| @override | ||||
| String toString() { | ||||
|   return 'TranslateQuery(text: $text, lang: $lang)'; | ||||
| } | ||||
|  | ||||
|  | ||||
| } | ||||
|  | ||||
| /// @nodoc | ||||
| abstract mixin class $TranslateQueryCopyWith<$Res>  { | ||||
|   factory $TranslateQueryCopyWith(TranslateQuery value, $Res Function(TranslateQuery) _then) = _$TranslateQueryCopyWithImpl; | ||||
| @useResult | ||||
| $Res call({ | ||||
|  String text, String lang | ||||
| }); | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
| } | ||||
| /// @nodoc | ||||
| class _$TranslateQueryCopyWithImpl<$Res> | ||||
|     implements $TranslateQueryCopyWith<$Res> { | ||||
|   _$TranslateQueryCopyWithImpl(this._self, this._then); | ||||
|  | ||||
|   final TranslateQuery _self; | ||||
|   final $Res Function(TranslateQuery) _then; | ||||
|  | ||||
| /// Create a copy of TranslateQuery | ||||
| /// with the given fields replaced by the non-null parameter values. | ||||
| @pragma('vm:prefer-inline') @override $Res call({Object? text = null,Object? lang = null,}) { | ||||
|   return _then(_self.copyWith( | ||||
| text: null == text ? _self.text : text // ignore: cast_nullable_to_non_nullable | ||||
| as String,lang: null == lang ? _self.lang : lang // ignore: cast_nullable_to_non_nullable | ||||
| as String, | ||||
|   )); | ||||
| } | ||||
|  | ||||
| } | ||||
|  | ||||
|  | ||||
| /// Adds pattern-matching-related methods to [TranslateQuery]. | ||||
| extension TranslateQueryPatterns on TranslateQuery { | ||||
| /// A variant of `map` that fallback to returning `orElse`. | ||||
| /// | ||||
| /// It is equivalent to doing: | ||||
| /// ```dart | ||||
| /// switch (sealedClass) { | ||||
| ///   case final Subclass value: | ||||
| ///     return ...; | ||||
| ///   case _: | ||||
| ///     return orElse(); | ||||
| /// } | ||||
| /// ``` | ||||
|  | ||||
| @optionalTypeArgs TResult maybeMap<TResult extends Object?>(TResult Function( _TranslateQuery value)?  $default,{required TResult orElse(),}){ | ||||
| final _that = this; | ||||
| switch (_that) { | ||||
| case _TranslateQuery() when $default != null: | ||||
| return $default(_that);case _: | ||||
|   return orElse(); | ||||
|  | ||||
| } | ||||
| } | ||||
| /// A `switch`-like method, using callbacks. | ||||
| /// | ||||
| /// Callbacks receives the raw object, upcasted. | ||||
| /// It is equivalent to doing: | ||||
| /// ```dart | ||||
| /// switch (sealedClass) { | ||||
| ///   case final Subclass value: | ||||
| ///     return ...; | ||||
| ///   case final Subclass2 value: | ||||
| ///     return ...; | ||||
| /// } | ||||
| /// ``` | ||||
|  | ||||
| @optionalTypeArgs TResult map<TResult extends Object?>(TResult Function( _TranslateQuery value)  $default,){ | ||||
| final _that = this; | ||||
| switch (_that) { | ||||
| case _TranslateQuery(): | ||||
| return $default(_that);} | ||||
| } | ||||
| /// A variant of `map` that fallback to returning `null`. | ||||
| /// | ||||
| /// It is equivalent to doing: | ||||
| /// ```dart | ||||
| /// switch (sealedClass) { | ||||
| ///   case final Subclass value: | ||||
| ///     return ...; | ||||
| ///   case _: | ||||
| ///     return null; | ||||
| /// } | ||||
| /// ``` | ||||
|  | ||||
| @optionalTypeArgs TResult? mapOrNull<TResult extends Object?>(TResult? Function( _TranslateQuery value)?  $default,){ | ||||
| final _that = this; | ||||
| switch (_that) { | ||||
| case _TranslateQuery() when $default != null: | ||||
| return $default(_that);case _: | ||||
|   return null; | ||||
|  | ||||
| } | ||||
| } | ||||
| /// A variant of `when` that fallback to an `orElse` callback. | ||||
| /// | ||||
| /// It is equivalent to doing: | ||||
| /// ```dart | ||||
| /// switch (sealedClass) { | ||||
| ///   case Subclass(:final field): | ||||
| ///     return ...; | ||||
| ///   case _: | ||||
| ///     return orElse(); | ||||
| /// } | ||||
| /// ``` | ||||
|  | ||||
| @optionalTypeArgs TResult maybeWhen<TResult extends Object?>(TResult Function( String text,  String lang)?  $default,{required TResult orElse(),}) {final _that = this; | ||||
| switch (_that) { | ||||
| case _TranslateQuery() when $default != null: | ||||
| return $default(_that.text,_that.lang);case _: | ||||
|   return orElse(); | ||||
|  | ||||
| } | ||||
| } | ||||
| /// A `switch`-like method, using callbacks. | ||||
| /// | ||||
| /// As opposed to `map`, this offers destructuring. | ||||
| /// It is equivalent to doing: | ||||
| /// ```dart | ||||
| /// switch (sealedClass) { | ||||
| ///   case Subclass(:final field): | ||||
| ///     return ...; | ||||
| ///   case Subclass2(:final field2): | ||||
| ///     return ...; | ||||
| /// } | ||||
| /// ``` | ||||
|  | ||||
| @optionalTypeArgs TResult when<TResult extends Object?>(TResult Function( String text,  String lang)  $default,) {final _that = this; | ||||
| switch (_that) { | ||||
| case _TranslateQuery(): | ||||
| return $default(_that.text,_that.lang);} | ||||
| } | ||||
| /// A variant of `when` that fallback to returning `null` | ||||
| /// | ||||
| /// It is equivalent to doing: | ||||
| /// ```dart | ||||
| /// switch (sealedClass) { | ||||
| ///   case Subclass(:final field): | ||||
| ///     return ...; | ||||
| ///   case _: | ||||
| ///     return null; | ||||
| /// } | ||||
| /// ``` | ||||
|  | ||||
| @optionalTypeArgs TResult? whenOrNull<TResult extends Object?>(TResult? Function( String text,  String lang)?  $default,) {final _that = this; | ||||
| switch (_that) { | ||||
| case _TranslateQuery() when $default != null: | ||||
| return $default(_that.text,_that.lang);case _: | ||||
|   return null; | ||||
|  | ||||
| } | ||||
| } | ||||
|  | ||||
| } | ||||
|  | ||||
| /// @nodoc | ||||
|  | ||||
|  | ||||
| class _TranslateQuery implements TranslateQuery { | ||||
|   const _TranslateQuery({required this.text, required this.lang}); | ||||
|    | ||||
|  | ||||
| @override final  String text; | ||||
| @override final  String lang; | ||||
|  | ||||
| /// Create a copy of TranslateQuery | ||||
| /// with the given fields replaced by the non-null parameter values. | ||||
| @override @JsonKey(includeFromJson: false, includeToJson: false) | ||||
| @pragma('vm:prefer-inline') | ||||
| _$TranslateQueryCopyWith<_TranslateQuery> get copyWith => __$TranslateQueryCopyWithImpl<_TranslateQuery>(this, _$identity); | ||||
|  | ||||
|  | ||||
|  | ||||
| @override | ||||
| bool operator ==(Object other) { | ||||
|   return identical(this, other) || (other.runtimeType == runtimeType&&other is _TranslateQuery&&(identical(other.text, text) || other.text == text)&&(identical(other.lang, lang) || other.lang == lang)); | ||||
| } | ||||
|  | ||||
|  | ||||
| @override | ||||
| int get hashCode => Object.hash(runtimeType,text,lang); | ||||
|  | ||||
| @override | ||||
| String toString() { | ||||
|   return 'TranslateQuery(text: $text, lang: $lang)'; | ||||
| } | ||||
|  | ||||
|  | ||||
| } | ||||
|  | ||||
| /// @nodoc | ||||
| abstract mixin class _$TranslateQueryCopyWith<$Res> implements $TranslateQueryCopyWith<$Res> { | ||||
|   factory _$TranslateQueryCopyWith(_TranslateQuery value, $Res Function(_TranslateQuery) _then) = __$TranslateQueryCopyWithImpl; | ||||
| @override @useResult | ||||
| $Res call({ | ||||
|  String text, String lang | ||||
| }); | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
| } | ||||
| /// @nodoc | ||||
| class __$TranslateQueryCopyWithImpl<$Res> | ||||
|     implements _$TranslateQueryCopyWith<$Res> { | ||||
|   __$TranslateQueryCopyWithImpl(this._self, this._then); | ||||
|  | ||||
|   final _TranslateQuery _self; | ||||
|   final $Res Function(_TranslateQuery) _then; | ||||
|  | ||||
| /// Create a copy of TranslateQuery | ||||
| /// with the given fields replaced by the non-null parameter values. | ||||
| @override @pragma('vm:prefer-inline') $Res call({Object? text = null,Object? lang = null,}) { | ||||
|   return _then(_TranslateQuery( | ||||
| text: null == text ? _self.text : text // ignore: cast_nullable_to_non_nullable | ||||
| as String,lang: null == lang ? _self.lang : lang // ignore: cast_nullable_to_non_nullable | ||||
| as String, | ||||
|   )); | ||||
| } | ||||
|  | ||||
|  | ||||
| } | ||||
|  | ||||
| // dart format on | ||||
							
								
								
									
										274
									
								
								lib/pods/translate.g.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										274
									
								
								lib/pods/translate.g.dart
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,274 @@ | ||||
| // GENERATED CODE - DO NOT MODIFY BY HAND | ||||
|  | ||||
| part of 'translate.dart'; | ||||
|  | ||||
| // ************************************************************************** | ||||
| // RiverpodGenerator | ||||
| // ************************************************************************** | ||||
|  | ||||
| String _$translateStringHash() => r'51d638cf07cbf3ffa9469298f5bd9c667bc0ccb7'; | ||||
|  | ||||
| /// Copied from Dart SDK | ||||
| class _SystemHash { | ||||
|   _SystemHash._(); | ||||
|  | ||||
|   static int combine(int hash, int value) { | ||||
|     // ignore: parameter_assignments | ||||
|     hash = 0x1fffffff & (hash + value); | ||||
|     // ignore: parameter_assignments | ||||
|     hash = 0x1fffffff & (hash + ((0x0007ffff & hash) << 10)); | ||||
|     return hash ^ (hash >> 6); | ||||
|   } | ||||
|  | ||||
|   static int finish(int hash) { | ||||
|     // ignore: parameter_assignments | ||||
|     hash = 0x1fffffff & (hash + ((0x03ffffff & hash) << 3)); | ||||
|     // ignore: parameter_assignments | ||||
|     hash = hash ^ (hash >> 11); | ||||
|     return 0x1fffffff & (hash + ((0x00003fff & hash) << 15)); | ||||
|   } | ||||
| } | ||||
|  | ||||
| /// See also [translateString]. | ||||
| @ProviderFor(translateString) | ||||
| const translateStringProvider = TranslateStringFamily(); | ||||
|  | ||||
| /// See also [translateString]. | ||||
| class TranslateStringFamily extends Family<AsyncValue<String>> { | ||||
|   /// See also [translateString]. | ||||
|   const TranslateStringFamily(); | ||||
|  | ||||
|   /// See also [translateString]. | ||||
|   TranslateStringProvider call(TranslateQuery query) { | ||||
|     return TranslateStringProvider(query); | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   TranslateStringProvider getProviderOverride( | ||||
|     covariant TranslateStringProvider provider, | ||||
|   ) { | ||||
|     return call(provider.query); | ||||
|   } | ||||
|  | ||||
|   static const Iterable<ProviderOrFamily>? _dependencies = null; | ||||
|  | ||||
|   @override | ||||
|   Iterable<ProviderOrFamily>? get dependencies => _dependencies; | ||||
|  | ||||
|   static const Iterable<ProviderOrFamily>? _allTransitiveDependencies = null; | ||||
|  | ||||
|   @override | ||||
|   Iterable<ProviderOrFamily>? get allTransitiveDependencies => | ||||
|       _allTransitiveDependencies; | ||||
|  | ||||
|   @override | ||||
|   String? get name => r'translateStringProvider'; | ||||
| } | ||||
|  | ||||
| /// See also [translateString]. | ||||
| class TranslateStringProvider extends AutoDisposeFutureProvider<String> { | ||||
|   /// See also [translateString]. | ||||
|   TranslateStringProvider(TranslateQuery query) | ||||
|     : this._internal( | ||||
|         (ref) => translateString(ref as TranslateStringRef, query), | ||||
|         from: translateStringProvider, | ||||
|         name: r'translateStringProvider', | ||||
|         debugGetCreateSourceHash: | ||||
|             const bool.fromEnvironment('dart.vm.product') | ||||
|                 ? null | ||||
|                 : _$translateStringHash, | ||||
|         dependencies: TranslateStringFamily._dependencies, | ||||
|         allTransitiveDependencies: | ||||
|             TranslateStringFamily._allTransitiveDependencies, | ||||
|         query: query, | ||||
|       ); | ||||
|  | ||||
|   TranslateStringProvider._internal( | ||||
|     super._createNotifier, { | ||||
|     required super.name, | ||||
|     required super.dependencies, | ||||
|     required super.allTransitiveDependencies, | ||||
|     required super.debugGetCreateSourceHash, | ||||
|     required super.from, | ||||
|     required this.query, | ||||
|   }) : super.internal(); | ||||
|  | ||||
|   final TranslateQuery query; | ||||
|  | ||||
|   @override | ||||
|   Override overrideWith( | ||||
|     FutureOr<String> Function(TranslateStringRef provider) create, | ||||
|   ) { | ||||
|     return ProviderOverride( | ||||
|       origin: this, | ||||
|       override: TranslateStringProvider._internal( | ||||
|         (ref) => create(ref as TranslateStringRef), | ||||
|         from: from, | ||||
|         name: null, | ||||
|         dependencies: null, | ||||
|         allTransitiveDependencies: null, | ||||
|         debugGetCreateSourceHash: null, | ||||
|         query: query, | ||||
|       ), | ||||
|     ); | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   AutoDisposeFutureProviderElement<String> createElement() { | ||||
|     return _TranslateStringProviderElement(this); | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   bool operator ==(Object other) { | ||||
|     return other is TranslateStringProvider && other.query == query; | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   int get hashCode { | ||||
|     var hash = _SystemHash.combine(0, runtimeType.hashCode); | ||||
|     hash = _SystemHash.combine(hash, query.hashCode); | ||||
|  | ||||
|     return _SystemHash.finish(hash); | ||||
|   } | ||||
| } | ||||
|  | ||||
| @Deprecated('Will be removed in 3.0. Use Ref instead') | ||||
| // ignore: unused_element | ||||
| mixin TranslateStringRef on AutoDisposeFutureProviderRef<String> { | ||||
|   /// The parameter `query` of this provider. | ||||
|   TranslateQuery get query; | ||||
| } | ||||
|  | ||||
| class _TranslateStringProviderElement | ||||
|     extends AutoDisposeFutureProviderElement<String> | ||||
|     with TranslateStringRef { | ||||
|   _TranslateStringProviderElement(super.provider); | ||||
|  | ||||
|   @override | ||||
|   TranslateQuery get query => (origin as TranslateStringProvider).query; | ||||
| } | ||||
|  | ||||
| String _$detectStringLanguageHash() => | ||||
|     r'697b68464b3d00927cc43ccc1ba8ba93f2a470ed'; | ||||
|  | ||||
| /// See also [detectStringLanguage]. | ||||
| @ProviderFor(detectStringLanguage) | ||||
| const detectStringLanguageProvider = DetectStringLanguageFamily(); | ||||
|  | ||||
| /// See also [detectStringLanguage]. | ||||
| class DetectStringLanguageFamily extends Family<String?> { | ||||
|   /// See also [detectStringLanguage]. | ||||
|   const DetectStringLanguageFamily(); | ||||
|  | ||||
|   /// See also [detectStringLanguage]. | ||||
|   DetectStringLanguageProvider call(String text) { | ||||
|     return DetectStringLanguageProvider(text); | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   DetectStringLanguageProvider getProviderOverride( | ||||
|     covariant DetectStringLanguageProvider provider, | ||||
|   ) { | ||||
|     return call(provider.text); | ||||
|   } | ||||
|  | ||||
|   static const Iterable<ProviderOrFamily>? _dependencies = null; | ||||
|  | ||||
|   @override | ||||
|   Iterable<ProviderOrFamily>? get dependencies => _dependencies; | ||||
|  | ||||
|   static const Iterable<ProviderOrFamily>? _allTransitiveDependencies = null; | ||||
|  | ||||
|   @override | ||||
|   Iterable<ProviderOrFamily>? get allTransitiveDependencies => | ||||
|       _allTransitiveDependencies; | ||||
|  | ||||
|   @override | ||||
|   String? get name => r'detectStringLanguageProvider'; | ||||
| } | ||||
|  | ||||
| /// See also [detectStringLanguage]. | ||||
| class DetectStringLanguageProvider extends AutoDisposeProvider<String?> { | ||||
|   /// See also [detectStringLanguage]. | ||||
|   DetectStringLanguageProvider(String text) | ||||
|     : this._internal( | ||||
|         (ref) => detectStringLanguage(ref as DetectStringLanguageRef, text), | ||||
|         from: detectStringLanguageProvider, | ||||
|         name: r'detectStringLanguageProvider', | ||||
|         debugGetCreateSourceHash: | ||||
|             const bool.fromEnvironment('dart.vm.product') | ||||
|                 ? null | ||||
|                 : _$detectStringLanguageHash, | ||||
|         dependencies: DetectStringLanguageFamily._dependencies, | ||||
|         allTransitiveDependencies: | ||||
|             DetectStringLanguageFamily._allTransitiveDependencies, | ||||
|         text: text, | ||||
|       ); | ||||
|  | ||||
|   DetectStringLanguageProvider._internal( | ||||
|     super._createNotifier, { | ||||
|     required super.name, | ||||
|     required super.dependencies, | ||||
|     required super.allTransitiveDependencies, | ||||
|     required super.debugGetCreateSourceHash, | ||||
|     required super.from, | ||||
|     required this.text, | ||||
|   }) : super.internal(); | ||||
|  | ||||
|   final String text; | ||||
|  | ||||
|   @override | ||||
|   Override overrideWith( | ||||
|     String? Function(DetectStringLanguageRef provider) create, | ||||
|   ) { | ||||
|     return ProviderOverride( | ||||
|       origin: this, | ||||
|       override: DetectStringLanguageProvider._internal( | ||||
|         (ref) => create(ref as DetectStringLanguageRef), | ||||
|         from: from, | ||||
|         name: null, | ||||
|         dependencies: null, | ||||
|         allTransitiveDependencies: null, | ||||
|         debugGetCreateSourceHash: null, | ||||
|         text: text, | ||||
|       ), | ||||
|     ); | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   AutoDisposeProviderElement<String?> createElement() { | ||||
|     return _DetectStringLanguageProviderElement(this); | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   bool operator ==(Object other) { | ||||
|     return other is DetectStringLanguageProvider && other.text == text; | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   int get hashCode { | ||||
|     var hash = _SystemHash.combine(0, runtimeType.hashCode); | ||||
|     hash = _SystemHash.combine(hash, text.hashCode); | ||||
|  | ||||
|     return _SystemHash.finish(hash); | ||||
|   } | ||||
| } | ||||
|  | ||||
| @Deprecated('Will be removed in 3.0. Use Ref instead') | ||||
| // ignore: unused_element | ||||
| mixin DetectStringLanguageRef on AutoDisposeProviderRef<String?> { | ||||
|   /// The parameter `text` of this provider. | ||||
|   String get text; | ||||
| } | ||||
|  | ||||
| class _DetectStringLanguageProviderElement | ||||
|     extends AutoDisposeProviderElement<String?> | ||||
|     with DetectStringLanguageRef { | ||||
|   _DetectStringLanguageProviderElement(super.provider); | ||||
|  | ||||
|   @override | ||||
|   String get text => (origin as DetectStringLanguageProvider).text; | ||||
| } | ||||
|  | ||||
| // ignore_for_file: type=lint | ||||
| // ignore_for_file: subtype_of_sealed_class, invalid_use_of_internal_member, invalid_use_of_visible_for_testing_member, deprecated_member_use_from_same_package | ||||
| @@ -46,6 +46,10 @@ class WebSocketService { | ||||
|   final StreamController<WebSocketState> _statusStreamController = | ||||
|       StreamController<WebSocketState>.broadcast(); | ||||
|   Timer? _reconnectTimer; | ||||
|   Timer? _heartbeatTimer; | ||||
|  | ||||
|   DateTime? _heartbeatAt; | ||||
|   Duration? _heartbeatDelay; | ||||
|  | ||||
|   Stream<WebSocketPacket> get dataStream => _streamController.stream; | ||||
|   Stream<WebSocketState> get statusStream => _statusStreamController.stream; | ||||
| @@ -71,6 +75,7 @@ class WebSocketService { | ||||
|       } | ||||
|       await _channel!.ready; | ||||
|       _statusStreamController.sink.add(WebSocketState.connected()); | ||||
|       _scheduleHeartbeat(); | ||||
|       _channel!.stream.listen( | ||||
|         (data) { | ||||
|           final dataStr = | ||||
| @@ -80,6 +85,13 @@ class WebSocketService { | ||||
|           log( | ||||
|             "[WebSocket] Received packet: ${packet.type} ${packet.errorMessage}", | ||||
|           ); | ||||
|           if (packet.type == 'pong' && _heartbeatAt != null) { | ||||
|             var now = DateTime.now(); | ||||
|             _heartbeatDelay = now.difference(_heartbeatAt!); | ||||
|             log( | ||||
|               "[WebSocket] Server respond last heartbeat for ${_heartbeatDelay!.inMilliseconds} ms", | ||||
|             ); | ||||
|           } | ||||
|         }, | ||||
|         onDone: () { | ||||
|           log('[WebSocket] Connection closed, attempting to reconnect...'); | ||||
| @@ -108,6 +120,19 @@ class WebSocketService { | ||||
|     }); | ||||
|   } | ||||
|  | ||||
|   void _scheduleHeartbeat() { | ||||
|     _heartbeatTimer?.cancel(); | ||||
|     _heartbeatTimer = Timer.periodic(const Duration(seconds: 60), (_) { | ||||
|       _beatTheHeart(); | ||||
|     }); | ||||
|   } | ||||
|  | ||||
|   void _beatTheHeart() { | ||||
|     _heartbeatAt = DateTime.now(); | ||||
|     log('[WebSocket] We\'re beating the heart! $_heartbeatAt'); | ||||
|     sendMessage(jsonEncode(WebSocketPacket(type: 'ping', data: null))); | ||||
|   } | ||||
|  | ||||
|   WebSocketChannel? get ws => _channel; | ||||
|  | ||||
|   void sendMessage(String message) { | ||||
|   | ||||
							
								
								
									
										127
									
								
								lib/route.dart
									
									
									
									
									
								
							
							
						
						
									
										127
									
								
								lib/route.dart
									
									
									
									
									
								
							| @@ -28,9 +28,11 @@ import 'package:island/screens/creators/hub.dart'; | ||||
| import 'package:island/screens/creators/posts/post_manage_list.dart'; | ||||
| import 'package:island/screens/creators/stickers/stickers.dart'; | ||||
| import 'package:island/screens/creators/stickers/pack_detail.dart'; | ||||
| import 'package:island/screens/creators/poll/poll_list.dart'; | ||||
| import 'package:island/screens/creators/publishers.dart'; | ||||
| import 'package:island/screens/creators/webfeed/webfeed_list.dart'; | ||||
| import 'package:island/screens/creators/webfeed/webfeed_edit.dart'; | ||||
| import 'package:island/screens/poll/poll_editor.dart'; | ||||
| import 'package:island/screens/posts/compose.dart'; | ||||
| import 'package:island/screens/posts/post_detail.dart'; | ||||
| import 'package:island/screens/posts/pub_profile.dart'; | ||||
| @@ -144,6 +146,37 @@ final routerProvider = Provider<GoRouter>((ref) { | ||||
|                   return CreatorPostListScreen(pubName: name); | ||||
|                 }, | ||||
|               ), | ||||
|               // Poll list route | ||||
|               GoRoute( | ||||
|                 name: 'creatorPolls', | ||||
|                 path: '/creators/:name/polls', | ||||
|                 builder: (context, state) { | ||||
|                   final name = state.pathParameters['name']!; | ||||
|                   return CreatorPollListScreen(pubName: name); | ||||
|                 }, | ||||
|               ), | ||||
|               // Poll routes | ||||
|               GoRoute( | ||||
|                 name: 'creatorPollNew', | ||||
|                 path: '/creators/:name/polls/new', | ||||
|                 builder: (context, state) { | ||||
|                   final name = state.pathParameters['name']!; | ||||
|                   // initialPollId left null for create; initialPublisher prefilled | ||||
|                   return PollEditorScreen(initialPublisher: name); | ||||
|                 }, | ||||
|               ), | ||||
|               GoRoute( | ||||
|                 name: 'creatorPollEdit', | ||||
|                 path: '/creators/:name/polls/:id/edit', | ||||
|                 builder: (context, state) { | ||||
|                   final name = state.pathParameters['name']!; | ||||
|                   final id = state.pathParameters['id']!; | ||||
|                   return PollEditorScreen( | ||||
|                     initialPollId: id, | ||||
|                     initialPublisher: name, | ||||
|                   ); | ||||
|                 }, | ||||
|               ), | ||||
|               GoRoute( | ||||
|                 name: 'creatorStickers', | ||||
|                 path: '/creators/:name/stickers', | ||||
| @@ -287,12 +320,6 @@ final routerProvider = Provider<GoRouter>((ref) { | ||||
|             builder: (context, state) => const AboutScreen(), | ||||
|           ), | ||||
|  | ||||
|           GoRoute( | ||||
|             name: 'reportList', | ||||
|             path: '/safety/reports/me', | ||||
|             builder: (context, state) => const AbuseReportListScreen(), | ||||
|           ), | ||||
|  | ||||
|           GoRoute( | ||||
|             name: 'reportDetail', | ||||
|             path: '/safety/reports/me/:id', | ||||
| @@ -310,42 +337,36 @@ final routerProvider = Provider<GoRouter>((ref) { | ||||
|             }, | ||||
|             routes: [ | ||||
|               // Explore tab | ||||
|               ShellRoute( | ||||
|                 builder: | ||||
|                     (context, state, child) => ExploreShellScreen(child: child), | ||||
|                 routes: [ | ||||
|                   GoRoute( | ||||
|                     name: 'explore', | ||||
|                     path: '/', | ||||
|                     builder: (context, state) => const ExploreScreen(), | ||||
|                   ), | ||||
|                   GoRoute( | ||||
|                     name: 'postSearch', | ||||
|                     path: '/posts/search', | ||||
|                     builder: (context, state) => const PostSearchScreen(), | ||||
|                   ), | ||||
|                   GoRoute( | ||||
|                     name: 'postDetail', | ||||
|                     path: '/posts/:id', | ||||
|                     builder: (context, state) { | ||||
|                       final id = state.pathParameters['id']!; | ||||
|                       return PostDetailScreen(id: id); | ||||
|                     }, | ||||
|                   ), | ||||
|                   GoRoute( | ||||
|                     name: 'publisherProfile', | ||||
|                     path: '/publishers/:name', | ||||
|                     builder: (context, state) { | ||||
|                       final name = state.pathParameters['name']!; | ||||
|                       return PublisherProfileScreen(name: name); | ||||
|                     }, | ||||
|                   ), | ||||
|                   GoRoute( | ||||
|                     name: 'discoveryRealms', | ||||
|                     path: '/discovery/realms', | ||||
|                     builder: (context, state) => const DiscoveryRealmsScreen(), | ||||
|                   ), | ||||
|                 ], | ||||
|               GoRoute( | ||||
|                 name: 'explore', | ||||
|                 path: '/', | ||||
|                 builder: (context, state) => const ExploreScreen(), | ||||
|               ), | ||||
|               GoRoute( | ||||
|                 name: 'postSearch', | ||||
|                 path: '/posts/search', | ||||
|                 builder: (context, state) => const PostSearchScreen(), | ||||
|               ), | ||||
|               GoRoute( | ||||
|                 name: 'postDetail', | ||||
|                 path: '/posts/:id', | ||||
|                 builder: (context, state) { | ||||
|                   final id = state.pathParameters['id']!; | ||||
|                   return PostDetailScreen(id: id); | ||||
|                 }, | ||||
|               ), | ||||
|               GoRoute( | ||||
|                 name: 'publisherProfile', | ||||
|                 path: '/publishers/:name', | ||||
|                 builder: (context, state) { | ||||
|                   final name = state.pathParameters['name']!; | ||||
|                   return PublisherProfileScreen(name: name); | ||||
|                 }, | ||||
|               ), | ||||
|               GoRoute( | ||||
|                 name: 'discoveryRealms', | ||||
|                 path: '/discovery/realms', | ||||
|                 builder: (context, state) => const DiscoveryRealmsScreen(), | ||||
|               ), | ||||
|  | ||||
|               // Chat tab | ||||
| @@ -445,14 +466,6 @@ final routerProvider = Provider<GoRouter>((ref) { | ||||
|                     path: '/account/relationships', | ||||
|                     builder: (context, state) => const RelationshipScreen(), | ||||
|                   ), | ||||
|                   GoRoute( | ||||
|                     name: 'accountProfile', | ||||
|                     path: '/account/:name', | ||||
|                     builder: (context, state) { | ||||
|                       final name = state.pathParameters['name']!; | ||||
|                       return AccountProfileScreen(name: name); | ||||
|                     }, | ||||
|                   ), | ||||
|                   GoRoute( | ||||
|                     name: 'profileUpdate', | ||||
|                     path: '/account/me/update', | ||||
| @@ -468,8 +481,22 @@ final routerProvider = Provider<GoRouter>((ref) { | ||||
|                     path: '/account/me/settings', | ||||
|                     builder: (context, state) => const AccountSettingsScreen(), | ||||
|                   ), | ||||
|                   GoRoute( | ||||
|                     name: 'reportList', | ||||
|                     path: '/safety/reports/me', | ||||
|                     builder: (context, state) => const AbuseReportListScreen(), | ||||
|                   ), | ||||
|                 ], | ||||
|               ), | ||||
|  | ||||
|               GoRoute( | ||||
|                 name: 'accountProfile', | ||||
|                 path: '/account/:name', | ||||
|                 builder: (context, state) { | ||||
|                   final name = state.pathParameters['name']!; | ||||
|                   return AccountProfileScreen(name: name); | ||||
|                 }, | ||||
|               ), | ||||
|             ], | ||||
|           ), | ||||
|         ], | ||||
|   | ||||
| @@ -93,6 +93,7 @@ class _AboutScreenState extends ConsumerState<AboutScreen> { | ||||
|     final theme = Theme.of(context); | ||||
|  | ||||
|     return AppScaffold( | ||||
|       isNoBackground: false, | ||||
|       appBar: AppBar(title: Text('about'.tr()), elevation: 0), | ||||
|       body: | ||||
|           _isLoading | ||||
| @@ -204,7 +205,7 @@ class _AboutScreenState extends ConsumerState<AboutScreen> { | ||||
|                           title: 'aboutScreenTermsOfServiceTitle'.tr(), | ||||
|                           onTap: | ||||
|                               () => _launchURL( | ||||
|                                 'https://solsynth.dev/terms/basic-law', | ||||
|                                 'https://solsynth.dev/terms/user-agreement', | ||||
|                               ), | ||||
|                         ), | ||||
|                         _buildListTile( | ||||
|   | ||||
| @@ -64,7 +64,7 @@ class AccountScreen extends HookConsumerWidget { | ||||
|     } | ||||
|  | ||||
|     return AppScaffold( | ||||
|       noBackground: isWide, | ||||
|       isNoBackground: isWide, | ||||
|       appBar: AppBar(backgroundColor: Colors.transparent, toolbarHeight: 0), | ||||
|       body: SingleChildScrollView( | ||||
|         padding: getTabbedPadding(context), | ||||
| @@ -231,7 +231,7 @@ class AccountScreen extends HookConsumerWidget { | ||||
|             ListTile( | ||||
|               minTileHeight: 48, | ||||
|               title: Text('abuseReports').tr(), | ||||
|               contentPadding: const EdgeInsets.only(left: 24, right: 17), | ||||
|               contentPadding: const EdgeInsets.symmetric(horizontal: 24), | ||||
|               leading: const Icon(Symbols.gavel), | ||||
|               trailing: const Icon(Symbols.chevron_right), | ||||
|               onTap: () => context.pushNamed('reportList'), | ||||
|   | ||||
| @@ -46,7 +46,7 @@ class EventCalanderScreen extends HookConsumerWidget { | ||||
|     } | ||||
|  | ||||
|     return AppScaffold( | ||||
|       noBackground: false, | ||||
|       isNoBackground: false, | ||||
|       appBar: AppBar( | ||||
|         leading: const PageBackButton(), | ||||
|         title: Text('eventCalander').tr(), | ||||
|   | ||||
| @@ -27,7 +27,7 @@ part 'leveling.g.dart'; | ||||
| Future<SnWalletSubscription?> accountStellarSubscription(Ref ref) async { | ||||
|   try { | ||||
|     final client = ref.watch(apiClientProvider); | ||||
|     final resp = await client.get('/subscriptions/fuzzy/solian.stellar'); | ||||
|     final resp = await client.get('/id/subscriptions/fuzzy/solian.stellar'); | ||||
|     return SnWalletSubscription.fromJson(resp.data); | ||||
|   } catch (err) { | ||||
|     if (err is DioException && err.response?.statusCode == 404) return null; | ||||
| @@ -280,7 +280,7 @@ class LevelingScreen extends HookConsumerWidget { | ||||
|       try { | ||||
|         showLoadingModal(context); | ||||
|         final client = ref.watch(apiClientProvider); | ||||
|         await client.post('/subscriptions/${membership.identifier}/cancel'); | ||||
|         await client.post('/id/subscriptions/${membership.identifier}/cancel'); | ||||
|         ref.invalidate(accountStellarSubscriptionProvider); | ||||
|         ref.read(userInfoProvider.notifier).fetchUser(); | ||||
|         if (context.mounted) { | ||||
| @@ -432,7 +432,7 @@ class LevelingScreen extends HookConsumerWidget { | ||||
|         'id': 'solian.stellar.nova', | ||||
|         'name': 'membershipTierNova'.tr(), | ||||
|         'price': 'membershipPriceNova'.tr(), | ||||
|         'color': Colors.indigo, | ||||
|         'color': Color.fromRGBO(57, 197, 187, 1), | ||||
|       }, | ||||
|       { | ||||
|         'id': 'solian.stellar.supernova', | ||||
| @@ -603,7 +603,7 @@ class LevelingScreen extends HookConsumerWidget { | ||||
|     try { | ||||
|       showLoadingModal(context); | ||||
|       final resp = await client.post( | ||||
|         '/subscriptions', | ||||
|         '/id/subscriptions', | ||||
|         data: { | ||||
|           'identifier': tierId, | ||||
|           'payment_method': 'solian.wallet', | ||||
| @@ -615,7 +615,7 @@ class LevelingScreen extends HookConsumerWidget { | ||||
|       final subscription = SnWalletSubscription.fromJson(resp.data); | ||||
|       if (subscription.status == 1) return; | ||||
|       final orderResp = await client.post( | ||||
|         '/subscriptions/${subscription.identifier}/order', | ||||
|         '/id/subscriptions/${subscription.identifier}/order', | ||||
|       ); | ||||
|       final order = SnWalletOrder.fromJson(orderResp.data); | ||||
|  | ||||
| @@ -633,7 +633,7 @@ class LevelingScreen extends HookConsumerWidget { | ||||
|  | ||||
|       if (paidOrder != null) { | ||||
|         await client.post( | ||||
|           '/subscriptions/order/handle', | ||||
|           '/id/subscriptions/order/handle', | ||||
|           data: {'order_id': paidOrder.id}, | ||||
|         ); | ||||
|  | ||||
|   | ||||
| @@ -7,7 +7,7 @@ part of 'leveling.dart'; | ||||
| // ************************************************************************** | ||||
|  | ||||
| String _$accountStellarSubscriptionHash() => | ||||
|     r'37fb821460e3ac50b5cf777c933b6779f732daee'; | ||||
|     r'80abcdefb3868775fd8fe3c980215713efff5948'; | ||||
|  | ||||
| /// See also [accountStellarSubscription]. | ||||
| @ProviderFor(accountStellarSubscription) | ||||
|   | ||||
| @@ -94,7 +94,7 @@ class AccountSettingsScreen extends HookConsumerWidget { | ||||
|         final userInfo = ref.read(userInfoProvider); | ||||
|         final client = ref.read(apiClientProvider); | ||||
|         await client.post( | ||||
|           '/accounts/recovery/password', | ||||
|           '/id/accounts/recovery/password', | ||||
|           data: {'account': userInfo.value!.name, 'captcha_token': captchaTk}, | ||||
|         ); | ||||
|         if (context.mounted) { | ||||
|   | ||||
| @@ -12,6 +12,7 @@ import 'package:island/pods/event_calendar.dart'; | ||||
| import 'package:island/pods/network.dart'; | ||||
| import 'package:island/pods/userinfo.dart'; | ||||
| import 'package:island/services/color.dart'; | ||||
| import 'package:island/services/responsive.dart'; | ||||
| import 'package:island/services/time.dart'; | ||||
| import 'package:island/services/timezone/native.dart'; | ||||
| import 'package:island/widgets/account/account_name.dart'; | ||||
| @@ -22,10 +23,12 @@ import 'package:island/widgets/account/status.dart'; | ||||
| import 'package:island/widgets/alert.dart'; | ||||
| import 'package:island/widgets/app_scaffold.dart'; | ||||
| import 'package:island/widgets/content/cloud_files.dart'; | ||||
| import 'package:island/widgets/content/markdown.dart'; | ||||
| import 'package:island/widgets/safety/abuse_report_helper.dart'; | ||||
| import 'package:material_symbols_icons/symbols.dart'; | ||||
| import 'package:palette_generator/palette_generator.dart'; | ||||
| import 'package:riverpod_annotation/riverpod_annotation.dart'; | ||||
| import 'package:share_plus/share_plus.dart'; | ||||
| import 'package:styled_widget/styled_widget.dart'; | ||||
|  | ||||
| part 'profile.g.dart'; | ||||
| @@ -248,294 +251,402 @@ class AccountProfileScreen extends HookConsumerWidget { | ||||
|  | ||||
|     final user = ref.watch(userInfoProvider); | ||||
|  | ||||
|     Widget accountBasicInfo(SnAccount data) => Padding( | ||||
|       padding: const EdgeInsets.fromLTRB(24, 24, 24, 8), | ||||
|       child: Row( | ||||
|         crossAxisAlignment: CrossAxisAlignment.start, | ||||
|         children: [ | ||||
|           ProfilePictureWidget(file: data.profile.picture, radius: 32), | ||||
|           const Gap(20), | ||||
|           Expanded( | ||||
|             child: Column( | ||||
|               crossAxisAlignment: CrossAxisAlignment.stretch, | ||||
|               children: [ | ||||
|                 Row( | ||||
|                   children: [ | ||||
|                     AccountName(account: data, style: TextStyle(fontSize: 20)), | ||||
|                     const Gap(6), | ||||
|                     Flexible( | ||||
|                       child: Text( | ||||
|                         '@${data.name}', | ||||
|                         maxLines: 1, | ||||
|                         overflow: TextOverflow.ellipsis, | ||||
|                       ).fontSize(14).opacity(0.85), | ||||
|                     ), | ||||
|                   ], | ||||
|                 ), | ||||
|                 AccountStatusWidget(uname: name, padding: EdgeInsets.zero), | ||||
|               ], | ||||
|             ), | ||||
|           ), | ||||
|           IconButton( | ||||
|             onPressed: () { | ||||
|               SharePlus.instance.share( | ||||
|                 ShareParams( | ||||
|                   uri: Uri.parse('https://id.solian.app/@${data.name}'), | ||||
|                 ), | ||||
|               ); | ||||
|             }, | ||||
|             icon: const Icon(Symbols.share), | ||||
|           ), | ||||
|         ], | ||||
|       ), | ||||
|     ); | ||||
|  | ||||
|     Widget accountProfileBio(SnAccount data) => Card( | ||||
|       child: Column( | ||||
|         crossAxisAlignment: CrossAxisAlignment.start, | ||||
|         children: [ | ||||
|           Text('bio').tr().bold().fontSize(15).padding(bottom: 8), | ||||
|           if (data.profile.bio.isEmpty) | ||||
|             Text('descriptionNone').tr().italic() | ||||
|           else | ||||
|             MarkdownTextContent( | ||||
|               content: data.profile.bio, | ||||
|               linesMargin: EdgeInsets.zero, | ||||
|             ), | ||||
|         ], | ||||
|       ).padding(horizontal: 24, vertical: 20), | ||||
|     ); | ||||
|  | ||||
|     Widget accountProfileDetail(SnAccount data) => Card( | ||||
|       child: Column( | ||||
|         crossAxisAlignment: CrossAxisAlignment.stretch, | ||||
|         spacing: 24, | ||||
|         children: [ | ||||
|           if (buildSubcolumn(data).isNotEmpty) | ||||
|             Column( | ||||
|               crossAxisAlignment: CrossAxisAlignment.start, | ||||
|               spacing: 2, | ||||
|               children: buildSubcolumn(data), | ||||
|             ), | ||||
|           if (data.profile.timeZone.isNotEmpty) | ||||
|             Column( | ||||
|               crossAxisAlignment: CrossAxisAlignment.start, | ||||
|               children: [ | ||||
|                 Text('timeZone').tr().bold(), | ||||
|                 Row( | ||||
|                   crossAxisAlignment: CrossAxisAlignment.baseline, | ||||
|                   textBaseline: TextBaseline.alphabetic, | ||||
|                   spacing: 6, | ||||
|                   children: [ | ||||
|                     Text(data.profile.timeZone), | ||||
|                     Text( | ||||
|                       getTzInfo( | ||||
|                         data.profile.timeZone, | ||||
|                       ).$2.formatCustomGlobal('HH:mm'), | ||||
|                     ), | ||||
|                     Text( | ||||
|                       getTzInfo(data.profile.timeZone).$1.formatOffsetLocal(), | ||||
|                     ).fontSize(11), | ||||
|                     Text( | ||||
|                       'UTC${getTzInfo(data.profile.timeZone).$1.formatOffset()}', | ||||
|                     ).fontSize(11).opacity(0.75), | ||||
|                   ], | ||||
|                 ), | ||||
|               ], | ||||
|             ), | ||||
|         ], | ||||
|       ).padding(horizontal: 24, vertical: 16), | ||||
|     ); | ||||
|  | ||||
|     Widget accountAction(SnAccount data) => Card( | ||||
|       child: Column( | ||||
|         children: [ | ||||
|           Row( | ||||
|             spacing: 8, | ||||
|             children: [ | ||||
|               if (accountRelationship.value == null || | ||||
|                   accountRelationship.value!.status > -100) | ||||
|                 Expanded( | ||||
|                   child: FilledButton.icon( | ||||
|                     style: ButtonStyle( | ||||
|                       backgroundColor: WidgetStatePropertyAll( | ||||
|                         accountRelationship.value == null | ||||
|                             ? null | ||||
|                             : Theme.of(context).colorScheme.secondary, | ||||
|                       ), | ||||
|                       foregroundColor: WidgetStatePropertyAll( | ||||
|                         accountRelationship.value == null | ||||
|                             ? null | ||||
|                             : Theme.of(context).colorScheme.onSecondary, | ||||
|                       ), | ||||
|                     ), | ||||
|                     onPressed: relationshipAction, | ||||
|                     label: | ||||
|                         Text( | ||||
|                           accountRelationship.value == null | ||||
|                               ? 'addFriendShort' | ||||
|                               : 'added', | ||||
|                         ).tr(), | ||||
|                     icon: | ||||
|                         accountRelationship.value == null | ||||
|                             ? const Icon(Symbols.person_add) | ||||
|                             : const Icon(Symbols.person_check), | ||||
|                   ), | ||||
|                 ), | ||||
|               if (accountRelationship.value == null || | ||||
|                   accountRelationship.value!.status <= -100) | ||||
|                 Expanded( | ||||
|                   child: FilledButton.icon( | ||||
|                     style: ButtonStyle( | ||||
|                       backgroundColor: WidgetStatePropertyAll( | ||||
|                         accountRelationship.value == null | ||||
|                             ? null | ||||
|                             : Theme.of(context).colorScheme.secondary, | ||||
|                       ), | ||||
|                       foregroundColor: WidgetStatePropertyAll( | ||||
|                         accountRelationship.value == null | ||||
|                             ? null | ||||
|                             : Theme.of(context).colorScheme.onSecondary, | ||||
|                       ), | ||||
|                     ), | ||||
|                     onPressed: blockAction, | ||||
|                     label: | ||||
|                         Text( | ||||
|                           accountRelationship.value == null | ||||
|                               ? 'blockUser' | ||||
|                               : 'unblockUser', | ||||
|                         ).tr(), | ||||
|                     icon: | ||||
|                         accountRelationship.value == null | ||||
|                             ? const Icon(Symbols.block) | ||||
|                             : const Icon(Symbols.person_cancel), | ||||
|                   ), | ||||
|                 ), | ||||
|             ], | ||||
|           ), | ||||
|           Row( | ||||
|             spacing: 8, | ||||
|             children: [ | ||||
|               Expanded( | ||||
|                 child: FilledButton.icon( | ||||
|                   onPressed: directMessageAction, | ||||
|                   icon: const Icon(Symbols.message), | ||||
|                   label: | ||||
|                       Text( | ||||
|                         accountChat.value == null | ||||
|                             ? 'createDirectMessage' | ||||
|                             : 'gotoDirectMessage', | ||||
|                         maxLines: 1, | ||||
|                       ).tr(), | ||||
|                 ), | ||||
|               ), | ||||
|               IconButton.filled( | ||||
|                 onPressed: () { | ||||
|                   showAbuseReportSheet( | ||||
|                     context, | ||||
|                     resourceIdentifier: 'account/${data.id}', | ||||
|                   ); | ||||
|                 }, | ||||
|                 icon: Icon( | ||||
|                   Symbols.flag, | ||||
|                   color: Theme.of(context).colorScheme.onError, | ||||
|                 ), | ||||
|                 style: ButtonStyle( | ||||
|                   backgroundColor: WidgetStatePropertyAll( | ||||
|                     Theme.of(context).colorScheme.error, | ||||
|                   ), | ||||
|                 ), | ||||
|               ), | ||||
|             ], | ||||
|           ), | ||||
|         ], | ||||
|       ).padding(horizontal: 16, vertical: 8), | ||||
|     ); | ||||
|  | ||||
|     return account.when( | ||||
|       data: | ||||
|           (data) => AppScaffold( | ||||
|             body: CustomScrollView( | ||||
|               slivers: [ | ||||
|                 SliverAppBar( | ||||
|                   foregroundColor: appbarColor.value, | ||||
|                   expandedHeight: 180, | ||||
|                   pinned: true, | ||||
|                   leading: PageBackButton( | ||||
|                     color: appbarColor.value, | ||||
|                     shadows: [appbarShadow], | ||||
|                   ), | ||||
|                   flexibleSpace: Stack( | ||||
|                     children: [ | ||||
|                       Positioned.fill( | ||||
|                         child: | ||||
|                             data.profile.background?.id != null | ||||
|                                 ? CloudImageWidget( | ||||
|                                   file: data.profile.background, | ||||
|                                 ) | ||||
|                                 : Container( | ||||
|                                   color: | ||||
|                                       Theme.of( | ||||
|                                         context, | ||||
|                                       ).appBarTheme.backgroundColor, | ||||
|                                 ), | ||||
|             isNoBackground: false, | ||||
|             appBar: | ||||
|                 isWideScreen(context) | ||||
|                     ? AppBar( | ||||
|                       foregroundColor: appbarColor.value, | ||||
|                       leading: PageBackButton( | ||||
|                         color: appbarColor.value, | ||||
|                         shadows: [appbarShadow], | ||||
|                       ), | ||||
|                       FlexibleSpaceBar( | ||||
|                         title: Text( | ||||
|                           data.nick, | ||||
|                           style: TextStyle( | ||||
|                             color: | ||||
|                                 appbarColor.value ?? | ||||
|                                 Theme.of(context).appBarTheme.foregroundColor, | ||||
|                             shadows: [appbarShadow], | ||||
|                       flexibleSpace: Stack( | ||||
|                         children: [ | ||||
|                           Positioned.fill( | ||||
|                             child: | ||||
|                                 data.profile.background?.id != null | ||||
|                                     ? CloudImageWidget( | ||||
|                                       file: data.profile.background, | ||||
|                                     ) | ||||
|                                     : Container( | ||||
|                                       color: | ||||
|                                           Theme.of( | ||||
|                                             context, | ||||
|                                           ).appBarTheme.backgroundColor, | ||||
|                                     ), | ||||
|                           ), | ||||
|                           FlexibleSpaceBar( | ||||
|                             title: Text( | ||||
|                               data.nick, | ||||
|                               style: TextStyle( | ||||
|                                 color: | ||||
|                                     appbarColor.value ?? | ||||
|                                     Theme.of( | ||||
|                                       context, | ||||
|                                     ).appBarTheme.foregroundColor, | ||||
|                                 shadows: [appbarShadow], | ||||
|                               ), | ||||
|                             ), | ||||
|                           ), | ||||
|                         ], | ||||
|                       ), | ||||
|                     ) | ||||
|                     : null, | ||||
|             body: | ||||
|                 isWideScreen(context) | ||||
|                     ? Row( | ||||
|                       children: [ | ||||
|                         Flexible( | ||||
|                           child: CustomScrollView( | ||||
|                             slivers: [ | ||||
|                               SliverToBoxAdapter(child: accountBasicInfo(data)), | ||||
|                               if (data.badges.isNotEmpty) | ||||
|                                 SliverToBoxAdapter( | ||||
|                                   child: BadgeList( | ||||
|                                     badges: data.badges, | ||||
|                                   ).padding(horizontal: 24, bottom: 24), | ||||
|                                 ), | ||||
|                               SliverToBoxAdapter( | ||||
|                                 child: Column( | ||||
|                                   spacing: 12, | ||||
|                                   children: [ | ||||
|                                     LevelingProgressCard( | ||||
|                                       level: data.profile.level, | ||||
|                                       experience: data.profile.experience, | ||||
|                                       progress: data.profile.levelingProgress, | ||||
|                                     ), | ||||
|                                     if (data.profile.verification != null) | ||||
|                                       Card( | ||||
|                                         child: VerificationStatusCard( | ||||
|                                           mark: data.profile.verification!, | ||||
|                                         ), | ||||
|                                       ), | ||||
|                                   ], | ||||
|                                 ).padding(horizontal: 4, top: 8), | ||||
|                               ), | ||||
|                               SliverToBoxAdapter( | ||||
|                                 child: accountProfileBio(data).padding(top: 4), | ||||
|                               ), | ||||
|                               SliverToBoxAdapter( | ||||
|                                 child: accountProfileDetail(data), | ||||
|                               ), | ||||
|                             ], | ||||
|                           ), | ||||
|                         ), | ||||
|                       ), | ||||
|                     ], | ||||
|                   ), | ||||
|                 ), | ||||
|                 SliverToBoxAdapter( | ||||
|                   child: Padding( | ||||
|                     padding: const EdgeInsets.fromLTRB(24, 24, 24, 8), | ||||
|                     child: Row( | ||||
|                       crossAxisAlignment: CrossAxisAlignment.start, | ||||
|                       children: [ | ||||
|                         ProfilePictureWidget( | ||||
|                           file: data.profile.picture, | ||||
|                           radius: 32, | ||||
|                         ), | ||||
|                         const Gap(20), | ||||
|                         Expanded( | ||||
|                           child: Column( | ||||
|                             crossAxisAlignment: CrossAxisAlignment.stretch, | ||||
|                             children: [ | ||||
|                               Row( | ||||
|                                 children: [ | ||||
|                                   AccountName( | ||||
|                                     account: data, | ||||
|                                     style: TextStyle(fontSize: 20), | ||||
|                         Flexible( | ||||
|                           child: CustomScrollView( | ||||
|                             slivers: [ | ||||
|                               SliverGap(24), | ||||
|                               if (user.value != null) | ||||
|                                 SliverToBoxAdapter(child: accountAction(data)), | ||||
|                               SliverToBoxAdapter( | ||||
|                                 child: Card( | ||||
|                                   child: FortuneGraphWidget( | ||||
|                                     events: accountEvents, | ||||
|                                     eventCalanderUser: data.name, | ||||
|                                     margin: EdgeInsets.zero, | ||||
|                                   ), | ||||
|                                   const Gap(6), | ||||
|                                   Text( | ||||
|                                     '@${data.name}', | ||||
|                                   ).fontSize(14).opacity(0.85), | ||||
|                                 ], | ||||
|                               ), | ||||
|                               AccountStatusWidget( | ||||
|                                 uname: name, | ||||
|                                 padding: EdgeInsets.zero, | ||||
|                                 ), | ||||
|                               ), | ||||
|                             ], | ||||
|                           ), | ||||
|                         ), | ||||
|                       ], | ||||
|                     ), | ||||
|                   ), | ||||
|                 ), | ||||
|                 if (data.badges.isNotEmpty) | ||||
|                   SliverToBoxAdapter( | ||||
|                     child: BadgeList( | ||||
|                       badges: data.badges, | ||||
|                     ).padding(horizontal: 24, bottom: 24), | ||||
|                   ), | ||||
|                 SliverToBoxAdapter( | ||||
|                   child: Column( | ||||
|                     spacing: 12, | ||||
|                     children: [ | ||||
|                       LevelingProgressCard( | ||||
|                         level: data.profile.level, | ||||
|                         experience: data.profile.experience, | ||||
|                         progress: data.profile.levelingProgress, | ||||
|                       ), | ||||
|                       if (data.profile.verification != null) | ||||
|                         VerificationStatusCard( | ||||
|                           mark: data.profile.verification!, | ||||
|                         ), | ||||
|                     ], | ||||
|                   ).padding(horizontal: 20), | ||||
|                 ), | ||||
|  | ||||
|                 SliverToBoxAdapter( | ||||
|                   child: const Divider(height: 1).padding(vertical: 24), | ||||
|                 ), | ||||
|                 SliverToBoxAdapter( | ||||
|                   child: Column( | ||||
|                     crossAxisAlignment: CrossAxisAlignment.stretch, | ||||
|                     spacing: 24, | ||||
|                     children: [ | ||||
|                       if (buildSubcolumn(data).isNotEmpty) | ||||
|                         Column( | ||||
|                           crossAxisAlignment: CrossAxisAlignment.start, | ||||
|                           spacing: 2, | ||||
|                           children: buildSubcolumn(data), | ||||
|                         ), | ||||
|                       Column( | ||||
|                         crossAxisAlignment: CrossAxisAlignment.start, | ||||
|                         children: [ | ||||
|                           Text('bio').tr().bold(), | ||||
|                           Text( | ||||
|                             data.profile.bio.isEmpty | ||||
|                                 ? 'descriptionNone'.tr() | ||||
|                                 : data.profile.bio, | ||||
|                     ).padding(horizontal: 24) | ||||
|                     : CustomScrollView( | ||||
|                       slivers: [ | ||||
|                         SliverAppBar( | ||||
|                           foregroundColor: appbarColor.value, | ||||
|                           expandedHeight: 180, | ||||
|                           pinned: true, | ||||
|                           leading: PageBackButton( | ||||
|                             color: appbarColor.value, | ||||
|                             shadows: [appbarShadow], | ||||
|                           ), | ||||
|                         ], | ||||
|                       ), | ||||
|                       if (data.profile.timeZone.isNotEmpty) | ||||
|                         Column( | ||||
|                           crossAxisAlignment: CrossAxisAlignment.start, | ||||
|                           children: [ | ||||
|                             Text('timeZone').tr().bold(), | ||||
|                             Row( | ||||
|                               crossAxisAlignment: CrossAxisAlignment.baseline, | ||||
|                               textBaseline: TextBaseline.alphabetic, | ||||
|                               spacing: 6, | ||||
|                               children: [ | ||||
|                                 Text(data.profile.timeZone), | ||||
|                                 Text( | ||||
|                                   getTzInfo( | ||||
|                                     data.profile.timeZone, | ||||
|                                   ).$2.formatCustomGlobal('HH:mm'), | ||||
|                                 ), | ||||
|                                 Text( | ||||
|                                   getTzInfo( | ||||
|                                     data.profile.timeZone, | ||||
|                                   ).$1.formatOffsetLocal(), | ||||
|                                 ).fontSize(11), | ||||
|                                 Text( | ||||
|                                   'UTC${getTzInfo(data.profile.timeZone).$1.formatOffset()}', | ||||
|                                 ).fontSize(11).opacity(0.75), | ||||
|                               ], | ||||
|                             ), | ||||
|                           ], | ||||
|                         ), | ||||
|                     ], | ||||
|                   ).padding(horizontal: 24), | ||||
|                 ), | ||||
|  | ||||
|                 if (user.value != null) | ||||
|                   SliverToBoxAdapter( | ||||
|                     child: const Divider( | ||||
|                       height: 1, | ||||
|                     ).padding(top: 24, bottom: 12), | ||||
|                   ), | ||||
|                 if (user.value != null) | ||||
|                   SliverToBoxAdapter( | ||||
|                     child: Row( | ||||
|                       spacing: 8, | ||||
|                       children: [ | ||||
|                         if (accountRelationship.value == null || | ||||
|                             accountRelationship.value!.status > -100) | ||||
|                           Expanded( | ||||
|                             child: FilledButton.icon( | ||||
|                               style: ButtonStyle( | ||||
|                                 backgroundColor: WidgetStatePropertyAll( | ||||
|                                   accountRelationship.value == null | ||||
|                                       ? null | ||||
|                                       : Theme.of(context).colorScheme.secondary, | ||||
|                                 ), | ||||
|                                 foregroundColor: WidgetStatePropertyAll( | ||||
|                                   accountRelationship.value == null | ||||
|                                       ? null | ||||
|                                       : Theme.of( | ||||
|                                         context, | ||||
|                                       ).colorScheme.onSecondary, | ||||
|                           flexibleSpace: Stack( | ||||
|                             children: [ | ||||
|                               Positioned.fill( | ||||
|                                 child: | ||||
|                                     data.profile.background?.id != null | ||||
|                                         ? CloudImageWidget( | ||||
|                                           file: data.profile.background, | ||||
|                                         ) | ||||
|                                         : Container( | ||||
|                                           color: | ||||
|                                               Theme.of( | ||||
|                                                 context, | ||||
|                                               ).appBarTheme.backgroundColor, | ||||
|                                         ), | ||||
|                               ), | ||||
|                               FlexibleSpaceBar( | ||||
|                                 title: Text( | ||||
|                                   data.nick, | ||||
|                                   style: TextStyle( | ||||
|                                     color: | ||||
|                                         appbarColor.value ?? | ||||
|                                         Theme.of( | ||||
|                                           context, | ||||
|                                         ).appBarTheme.foregroundColor, | ||||
|                                     shadows: [appbarShadow], | ||||
|                                   ), | ||||
|                                 ), | ||||
|                               ), | ||||
|                               onPressed: relationshipAction, | ||||
|                               label: | ||||
|                                   Text( | ||||
|                                     accountRelationship.value == null | ||||
|                                         ? 'addFriendShort' | ||||
|                                         : 'added', | ||||
|                                   ).tr(), | ||||
|                               icon: | ||||
|                                   accountRelationship.value == null | ||||
|                                       ? const Icon(Symbols.person_add) | ||||
|                                       : const Icon(Symbols.person_check), | ||||
|                             ), | ||||
|                             ], | ||||
|                           ), | ||||
|                         if (accountRelationship.value == null || | ||||
|                             accountRelationship.value!.status <= -100) | ||||
|                           Expanded( | ||||
|                             child: FilledButton.icon( | ||||
|                               style: ButtonStyle( | ||||
|                                 backgroundColor: WidgetStatePropertyAll( | ||||
|                                   accountRelationship.value == null | ||||
|                                       ? null | ||||
|                                       : Theme.of(context).colorScheme.secondary, | ||||
|                                 ), | ||||
|                                 foregroundColor: WidgetStatePropertyAll( | ||||
|                                   accountRelationship.value == null | ||||
|                                       ? null | ||||
|                                       : Theme.of( | ||||
|                                         context, | ||||
|                                       ).colorScheme.onSecondary, | ||||
|                                 ), | ||||
|                               ), | ||||
|                               onPressed: blockAction, | ||||
|                               label: | ||||
|                                   Text( | ||||
|                                     accountRelationship.value == null | ||||
|                                         ? 'blockUser' | ||||
|                                         : 'unblockUser', | ||||
|                                   ).tr(), | ||||
|                               icon: | ||||
|                                   accountRelationship.value == null | ||||
|                                       ? const Icon(Symbols.block) | ||||
|                                       : const Icon(Symbols.person_cancel), | ||||
|                             ), | ||||
|                         ), | ||||
|                         SliverToBoxAdapter(child: accountBasicInfo(data)), | ||||
|                         if (data.badges.isNotEmpty) | ||||
|                           SliverToBoxAdapter( | ||||
|                             child: BadgeList( | ||||
|                               badges: data.badges, | ||||
|                             ).padding(horizontal: 24, bottom: 24), | ||||
|                           ), | ||||
|                         SliverToBoxAdapter( | ||||
|                           child: Column( | ||||
|                             children: [ | ||||
|                               LevelingProgressCard( | ||||
|                                 level: data.profile.level, | ||||
|                                 experience: data.profile.experience, | ||||
|                                 progress: data.profile.levelingProgress, | ||||
|                               ).padding(top: 8, horizontal: 8, bottom: 4), | ||||
|                               if (data.profile.verification != null) | ||||
|                                 Card( | ||||
|                                   child: VerificationStatusCard( | ||||
|                                     mark: data.profile.verification!, | ||||
|                                   ), | ||||
|                                 ).padding(horizontal: 4), | ||||
|                             ], | ||||
|                           ), | ||||
|                         ), | ||||
|                         SliverToBoxAdapter( | ||||
|                           child: accountProfileBio(data).padding(horizontal: 4), | ||||
|                         ), | ||||
|                         SliverToBoxAdapter( | ||||
|                           child: accountProfileDetail( | ||||
|                             data, | ||||
|                           ).padding(horizontal: 4), | ||||
|                         ), | ||||
|                         if (user.value != null) | ||||
|                           SliverToBoxAdapter( | ||||
|                             child: accountAction(data).padding(horizontal: 4), | ||||
|                           ), | ||||
|                         SliverToBoxAdapter( | ||||
|                           child: Card( | ||||
|                             child: FortuneGraphWidget( | ||||
|                               events: accountEvents, | ||||
|                               eventCalanderUser: data.name, | ||||
|                             ), | ||||
|                           ).padding(horizontal: 4), | ||||
|                         ), | ||||
|                       ], | ||||
|                     ).padding(horizontal: 16), | ||||
|                   ), | ||||
|                 SliverToBoxAdapter( | ||||
|                   child: Row( | ||||
|                     spacing: 8, | ||||
|                     children: [ | ||||
|                       Expanded( | ||||
|                         child: FilledButton.icon( | ||||
|                           onPressed: directMessageAction, | ||||
|                           icon: const Icon(Symbols.message), | ||||
|                           label: | ||||
|                               Text( | ||||
|                                 accountChat.value == null | ||||
|                                     ? 'createDirectMessage' | ||||
|                                     : 'gotoDirectMessage', | ||||
|                                 maxLines: 1, | ||||
|                               ).tr(), | ||||
|                         ), | ||||
|                       ), | ||||
|                       IconButton.filled( | ||||
|                         onPressed: () { | ||||
|                           showAbuseReportSheet( | ||||
|                             context, | ||||
|                             resourceIdentifier: 'account/${data.id}', | ||||
|                           ); | ||||
|                         }, | ||||
|                         icon: Icon( | ||||
|                           Symbols.flag, | ||||
|                           color: Theme.of(context).colorScheme.onError, | ||||
|                         ), | ||||
|                         style: ButtonStyle( | ||||
|                           backgroundColor: WidgetStatePropertyAll( | ||||
|                             Theme.of(context).colorScheme.error, | ||||
|                           ), | ||||
|                         ), | ||||
|                       ), | ||||
|                     ], | ||||
|                   ).padding(horizontal: 16, top: 4), | ||||
|                 ), | ||||
|                 SliverToBoxAdapter( | ||||
|                   child: const Divider(height: 1).padding(top: 12), | ||||
|                 ), | ||||
|                 SliverToBoxAdapter( | ||||
|                   child: Column( | ||||
|                     children: [ | ||||
|                       FortuneGraphWidget( | ||||
|                         events: accountEvents, | ||||
|                         eventCalanderUser: data.name, | ||||
|                       ), | ||||
|                     ], | ||||
|                   ).padding(all: 8), | ||||
|                 ), | ||||
|               ], | ||||
|             ), | ||||
|                     ), | ||||
|           ), | ||||
|       error: | ||||
|           (error, stackTrace) => AppScaffold( | ||||
|   | ||||
| @@ -73,7 +73,7 @@ class CreateAccountScreen extends HookConsumerWidget { | ||||
|     } | ||||
|  | ||||
|     return AppScaffold( | ||||
|       noBackground: false, | ||||
|       isNoBackground: false, | ||||
|       appBar: AppBar( | ||||
|         leading: const PageBackButton(), | ||||
|         title: Text('createAccount').tr(), | ||||
|   | ||||
| @@ -55,7 +55,7 @@ class LoginScreen extends HookConsumerWidget { | ||||
|     final factorPicked = useState<SnAuthFactor?>(null); | ||||
|  | ||||
|     return AppScaffold( | ||||
|       noBackground: false, | ||||
|       isNoBackground: false, | ||||
|       appBar: AppBar( | ||||
|         leading: const PageBackButton(), | ||||
|         title: Text('login').tr(), | ||||
|   | ||||
| @@ -80,7 +80,7 @@ class _OidcScreenState extends ConsumerState<OidcScreen> { | ||||
|                             : 'Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/114.0.0.0 Safari/537.36', | ||||
|                   ), | ||||
|                   initialUrlRequest: URLRequest( | ||||
|                     url: WebUri('$serverUrl/auth/login/${widget.provider}'), | ||||
|                     url: WebUri('$serverUrl/id/auth/login/${widget.provider}'), | ||||
|                     headers: { | ||||
|                       if (token?.token.isNotEmpty ?? false) | ||||
|                         'Authorization': 'AtField ${token!.token}', | ||||
| @@ -120,7 +120,7 @@ class _OidcScreenState extends ConsumerState<OidcScreen> { | ||||
|                       final queryParams = url.queryParameters; | ||||
|  | ||||
|                       // Check if we're on the token page | ||||
|                       if (path.endsWith('/id/auth/callback')) { | ||||
|                       if (path.endsWith('/auth/callback')) { | ||||
|                         // Extract token from URL | ||||
|                         final challenge = queryParams['challenge']; | ||||
|                         // Return the token and close the webview | ||||
| @@ -205,7 +205,7 @@ class _OidcScreenState extends ConsumerState<OidcScreen> { | ||||
|                       onPressed: () { | ||||
|                         if (currentUrl != null) { | ||||
|                           Clipboard.setData(ClipboardData(text: currentUrl!)); | ||||
|                           showSnackBar('copyToClipboard'); | ||||
|                           showSnackBar('copyToClipboard'.tr()); | ||||
|                         } | ||||
|                       }, | ||||
|                     ), | ||||
|   | ||||
| @@ -1,14 +1,16 @@ | ||||
| import 'dart:developer'; | ||||
|  | ||||
| import 'package:easy_localization/easy_localization.dart'; | ||||
| import 'package:flutter/material.dart'; | ||||
| import 'package:flutter/material.dart' hide ConnectionState; | ||||
| import 'package:flutter_hooks/flutter_hooks.dart'; | ||||
| import 'package:gap/gap.dart'; | ||||
| import 'package:hooks_riverpod/hooks_riverpod.dart'; | ||||
| import 'package:island/pods/call.dart'; | ||||
| import 'package:island/services/responsive.dart'; | ||||
| import 'package:island/widgets/app_scaffold.dart'; | ||||
| import 'package:island/widgets/chat/call_button.dart'; | ||||
| import 'package:island/widgets/chat/call_overlay.dart'; | ||||
| import 'package:island/widgets/chat/call_participant_tile.dart'; | ||||
| import 'package:island/widgets/alert.dart'; | ||||
| import 'package:livekit_client/livekit_client.dart'; | ||||
| import 'package:material_symbols_icons/symbols.dart'; | ||||
| import 'package:styled_widget/styled_widget.dart'; | ||||
| @@ -21,17 +23,39 @@ class CallScreen extends HookConsumerWidget { | ||||
|   Widget build(BuildContext context, WidgetRef ref) { | ||||
|     final ongoingCall = ref.watch(ongoingCallProvider(roomId)); | ||||
|     final callState = ref.watch(callNotifierProvider); | ||||
|     final callNotifier = ref.read(callNotifierProvider.notifier); | ||||
|     final callNotifier = ref.watch(callNotifierProvider.notifier); | ||||
|  | ||||
|     useEffect(() { | ||||
|       callNotifier.joinRoom(roomId); | ||||
|       log('[Call] Joining the call...'); | ||||
|       callNotifier.joinRoom(roomId).catchError((_) { | ||||
|         showConfirmAlert( | ||||
|           'Seems there already has a call connected, do you want override it?', | ||||
|           'Call already connected', | ||||
|         ).then((value) { | ||||
|           if (value != true) return; | ||||
|           log('[Call] Joining the call... with overrides'); | ||||
|           callNotifier.disconnect(); | ||||
|           callNotifier.dispose(); | ||||
|           callNotifier.joinRoom(roomId); | ||||
|         }); | ||||
|       }); | ||||
|       return null; | ||||
|     }, []); | ||||
|  | ||||
|     final viewMode = useState<String>('grid'); | ||||
|     final allAudioOnly = callNotifier.participants.every( | ||||
|       (p) => | ||||
|           !(p.hasVideo && | ||||
|               p.remoteParticipant.trackPublications.values.any( | ||||
|                 (pub) => | ||||
|                     pub.track != null && | ||||
|                     pub.kind == TrackType.VIDEO && | ||||
|                     !pub.muted && | ||||
|                     !pub.isDisposed, | ||||
|               )), | ||||
|     ); | ||||
|  | ||||
|     return AppScaffold( | ||||
|       noBackground: false, | ||||
|       isNoBackground: false, | ||||
|       appBar: AppBar( | ||||
|         leading: PageBackButton(), | ||||
|         title: Column( | ||||
| @@ -44,45 +68,55 @@ class CallScreen extends HookConsumerWidget { | ||||
|             Text( | ||||
|               callState.isConnected | ||||
|                   ? formatDuration(callState.duration) | ||||
|                   : 'Connecting', | ||||
|                   : (switch (callNotifier.room?.connectionState) { | ||||
|                     ConnectionState.connected => 'connected', | ||||
|                     ConnectionState.connecting => 'connecting', | ||||
|                     ConnectionState.reconnecting => 'reconnecting', | ||||
|                     _ => 'disconnected', | ||||
|                   }).tr(), | ||||
|               style: const TextStyle(fontSize: 14), | ||||
|             ), | ||||
|           ], | ||||
|         ), | ||||
|         actions: [ | ||||
|           Row( | ||||
|             mainAxisAlignment: MainAxisAlignment.end, | ||||
|             children: [ | ||||
|               IconButton( | ||||
|                 icon: Icon(Symbols.grid_view), | ||||
|                 tooltip: 'Grid View', | ||||
|                 onPressed: () => viewMode.value = 'grid', | ||||
|                 color: | ||||
|                     viewMode.value == 'grid' | ||||
|                         ? Theme.of(context).colorScheme.primary | ||||
|                         : null, | ||||
|           if (!allAudioOnly) | ||||
|             SingleChildScrollView( | ||||
|               child: Row( | ||||
|                 spacing: 4, | ||||
|                 children: [ | ||||
|                   for (final live in callNotifier.participants) | ||||
|                     SpeakingRippleAvatar(live: live, size: 30), | ||||
|                   const Gap(8), | ||||
|                 ], | ||||
|               ), | ||||
|               IconButton( | ||||
|                 icon: Icon(Symbols.view_agenda), | ||||
|                 tooltip: 'Stage View', | ||||
|                 onPressed: () => viewMode.value = 'stage', | ||||
|                 color: | ||||
|                     viewMode.value == 'stage' | ||||
|                         ? Theme.of(context).colorScheme.primary | ||||
|                         : null, | ||||
|               ), | ||||
|             ], | ||||
|           ), | ||||
|           const Gap(8), | ||||
|             ), | ||||
|         ], | ||||
|       ), | ||||
|       body: | ||||
|           callState.error != null | ||||
|               ? Center( | ||||
|                 child: Text( | ||||
|                   callState.error!, | ||||
|                   textAlign: TextAlign.center, | ||||
|                   style: const TextStyle(color: Colors.red), | ||||
|                 child: ConstrainedBox( | ||||
|                   constraints: const BoxConstraints(maxWidth: 320), | ||||
|                   child: Column( | ||||
|                     children: [ | ||||
|                       const Icon(Symbols.error_outline, size: 48), | ||||
|                       const Gap(4), | ||||
|                       Text( | ||||
|                         callState.error!, | ||||
|                         textAlign: TextAlign.center, | ||||
|                         style: const TextStyle(color: Color(0xFF757575)), | ||||
|                       ), | ||||
|                       const Gap(8), | ||||
|                       TextButton( | ||||
|                         onPressed: () { | ||||
|                           callNotifier.disconnect(); | ||||
|                           callNotifier.dispose(); | ||||
|                           callNotifier.joinRoom(roomId); | ||||
|                         }, | ||||
|                         child: Text('retry').tr(), | ||||
|                       ), | ||||
|                     ], | ||||
|                   ), | ||||
|                 ), | ||||
|               ) | ||||
|               : Column( | ||||
| @@ -100,17 +134,8 @@ class CallScreen extends HookConsumerWidget { | ||||
|                             child: Text('No participants in call'), | ||||
|                           ); | ||||
|                         } | ||||
|  | ||||
|                         final participants = callNotifier.participants; | ||||
|                         final allAudioOnly = participants.every( | ||||
|                           (p) => | ||||
|                               !(p.hasVideo && | ||||
|                                   p.remoteParticipant.trackPublications.values | ||||
|                                       .any( | ||||
|                                         (pub) => | ||||
|                                             pub.track != null && | ||||
|                                             pub.kind == TrackType.VIDEO, | ||||
|                                       )), | ||||
|                         ); | ||||
|                         if (allAudioOnly) { | ||||
|                           // Audio-only: show avatars in a compact row | ||||
|                           return Center( | ||||
| @@ -123,138 +148,41 @@ class CallScreen extends HookConsumerWidget { | ||||
|                                 runSpacing: 8, | ||||
|                                 children: [ | ||||
|                                   for (final live in participants) | ||||
|                                     Padding( | ||||
|                                       padding: const EdgeInsets.symmetric( | ||||
|                                         horizontal: 8, | ||||
|                                       ), | ||||
|                                       child: SpeakingRippleAvatar( | ||||
|                                         isSpeaking: live.isSpeaking, | ||||
|                                         audioLevel: | ||||
|                                             live.remoteParticipant.audioLevel, | ||||
|                                         pictureId: | ||||
|                                             live | ||||
|                                                 .participant | ||||
|                                                 .profile | ||||
|                                                 ?.account | ||||
|                                                 .profile | ||||
|                                                 .picture | ||||
|                                                 ?.id, | ||||
|                                         size: 72, | ||||
|                                       ), | ||||
|                                     ), | ||||
|                                     SpeakingRippleAvatar( | ||||
|                                       live: live, | ||||
|                                       size: 72, | ||||
|                                     ).padding(horizontal: 4), | ||||
|                                 ], | ||||
|                               ), | ||||
|                             ), | ||||
|                           ); | ||||
|                         } | ||||
|                         if (viewMode.value == 'stage') { | ||||
|                           // Stage view: show main speaker(s) large, others in row | ||||
|                           final mainSpeakers = | ||||
|                               participants | ||||
|                                   .where( | ||||
|                                     (p) => p | ||||
|                                         .remoteParticipant | ||||
|                                         .trackPublications | ||||
|                                         .values | ||||
|                                         .any( | ||||
|                                           (pub) => | ||||
|                                               pub.track != null && | ||||
|                                               pub.kind == TrackType.VIDEO, | ||||
|                                         ), | ||||
|                                   ) | ||||
|                                   .toList(); | ||||
|                           if (mainSpeakers.isEmpty && participants.isNotEmpty) { | ||||
|                             mainSpeakers.add(participants.first); | ||||
|                           } | ||||
|                           final others = | ||||
|                               participants | ||||
|                                   .where((p) => !mainSpeakers.contains(p)) | ||||
|                                   .toList(); | ||||
|                           return Column( | ||||
|                             children: [ | ||||
|                               Expanded( | ||||
|                                 child: Row( | ||||
|                                   mainAxisAlignment: MainAxisAlignment.center, | ||||
|                                   children: [ | ||||
|                                     for (final speaker in mainSpeakers) | ||||
|                                       Expanded( | ||||
|                                         child: | ||||
|                                             AspectRatio( | ||||
|                                               aspectRatio: 16 / 9, | ||||
|                                               child: Card( | ||||
|                                                 margin: EdgeInsets.zero, | ||||
|                                                 child: ClipRRect( | ||||
|                                                   borderRadius: | ||||
|                                                       BorderRadius.circular(8), | ||||
|                                                   child: Column( | ||||
|                                                     children: [ | ||||
|                                                       CallParticipantTile( | ||||
|                                                         live: speaker, | ||||
|                                                       ), | ||||
|                                                     ], | ||||
|                                                   ), | ||||
|                                                 ), | ||||
|                                               ), | ||||
|                                             ).center(), | ||||
|  | ||||
|                         // Stage view: show main speaker(s) large, others in row | ||||
|                         final mainSpeakers = | ||||
|                             participants | ||||
|                                 .where( | ||||
|                                   (p) => p | ||||
|                                       .remoteParticipant | ||||
|                                       .trackPublications | ||||
|                                       .values | ||||
|                                       .any( | ||||
|                                         (pub) => | ||||
|                                             pub.track != null && | ||||
|                                             pub.kind == TrackType.VIDEO, | ||||
|                                       ), | ||||
|                                   ], | ||||
|                                 ).padding(horizontal: 12), | ||||
|                               ), | ||||
|                               if (others.isNotEmpty) | ||||
|                                 SizedBox( | ||||
|                                   height: 100, | ||||
|                                   child: ListView( | ||||
|                                     scrollDirection: Axis.horizontal, | ||||
|                                     children: [ | ||||
|                                       for (final other in others) | ||||
|                                         Padding( | ||||
|                                           padding: const EdgeInsets.symmetric( | ||||
|                                             horizontal: 8, | ||||
|                                           ), | ||||
|                                           child: CallParticipantTile( | ||||
|                                             live: other, | ||||
|                                           ), | ||||
|                                         ), | ||||
|                                     ], | ||||
|                                   ), | ||||
|                                 ), | ||||
|                             ], | ||||
|                           ); | ||||
|                                 ) | ||||
|                                 .toList(); | ||||
|                         if (mainSpeakers.isEmpty && participants.isNotEmpty) { | ||||
|                           mainSpeakers.add(participants.first); | ||||
|                         } | ||||
|                         // Default: grid view | ||||
|                         return GridView.builder( | ||||
|                           padding: const EdgeInsets.symmetric( | ||||
|                             horizontal: 12, | ||||
|                             vertical: 8, | ||||
|                           ), | ||||
|                           gridDelegate: | ||||
|                               SliverGridDelegateWithFixedCrossAxisCount( | ||||
|                                 crossAxisCount: | ||||
|                                     isWidestScreen(context) | ||||
|                                         ? 4 | ||||
|                                         : isWiderScreen(context) | ||||
|                                         ? 3 | ||||
|                                         : 2, | ||||
|                                 childAspectRatio: 16 / 9, | ||||
|                                 crossAxisSpacing: 8, | ||||
|                                 mainAxisSpacing: 8, | ||||
|                         return Column( | ||||
|                           children: [ | ||||
|                             for (final speaker in mainSpeakers) | ||||
|                               Expanded( | ||||
|                                 child: CallParticipantTile(live: speaker), | ||||
|                               ), | ||||
|                           itemCount: participants.length, | ||||
|                           itemBuilder: (context, idx) { | ||||
|                             final live = participants[idx]; | ||||
|                             return AspectRatio( | ||||
|                               aspectRatio: 16 / 9, | ||||
|                               child: Card( | ||||
|                                 margin: EdgeInsets.zero, | ||||
|                                 child: ClipRRect( | ||||
|                                   borderRadius: BorderRadius.circular(8), | ||||
|                                   child: Column( | ||||
|                                     children: [CallParticipantTile(live: live)], | ||||
|                                   ), | ||||
|                                 ), | ||||
|                               ), | ||||
|                             ).center(); | ||||
|                           }, | ||||
|                           ], | ||||
|                         ); | ||||
|                       }, | ||||
|                     ), | ||||
|   | ||||
| @@ -21,7 +21,6 @@ import 'package:island/services/responsive.dart'; | ||||
| import 'package:island/widgets/account/account_picker.dart'; | ||||
| import 'package:island/widgets/alert.dart'; | ||||
| import 'package:island/widgets/app_scaffold.dart'; | ||||
| import 'package:island/widgets/chat/call_overlay.dart'; | ||||
| import 'package:island/widgets/content/cloud_files.dart'; | ||||
| import 'package:island/widgets/content/sheet.dart'; | ||||
| import 'package:island/widgets/realms/selection_dropdown.dart'; | ||||
| @@ -346,91 +345,79 @@ class ChatListScreen extends HookConsumerWidget { | ||||
|         child: const Icon(Symbols.add), | ||||
|       ), | ||||
|       floatingActionButtonLocation: TabbedFabLocation(context), | ||||
|       body: Stack( | ||||
|       body: Column( | ||||
|         children: [ | ||||
|           Column( | ||||
|             children: [ | ||||
|               Consumer( | ||||
|                 builder: (context, ref, _) { | ||||
|                   final summaryState = ref.watch(chatSummaryProvider); | ||||
|                   return summaryState.maybeWhen( | ||||
|                     loading: | ||||
|                         () => const LinearProgressIndicator( | ||||
|                           minHeight: 2, | ||||
|                           borderRadius: BorderRadius.zero, | ||||
|                         ), | ||||
|                     orElse: () => const SizedBox.shrink(), | ||||
|                   ); | ||||
|                 }, | ||||
|               ), | ||||
|               Expanded( | ||||
|                 child: chats.when( | ||||
|                   data: | ||||
|                       (items) => RefreshIndicator( | ||||
|                         onRefresh: | ||||
|                             () => Future.sync(() { | ||||
|                               ref.invalidate(chatroomsJoinedProvider); | ||||
|                             }), | ||||
|                         child: ListView.builder( | ||||
|                           padding: getTabbedPadding( | ||||
|                             context, | ||||
|                             bottom: callState.isConnected ? 96 : null, | ||||
|                           ), | ||||
|                           itemCount: | ||||
|                               items | ||||
|                                   .where( | ||||
|                                     (item) => | ||||
|                                         selectedTab.value == 0 || | ||||
|                                         (selectedTab.value == 1 && | ||||
|                                             item.type == 1) || | ||||
|                                         (selectedTab.value == 2 && | ||||
|                                             item.type != 1), | ||||
|                                   ) | ||||
|                                   .length, | ||||
|                           itemBuilder: (context, index) { | ||||
|                             final filteredItems = | ||||
|                                 items | ||||
|                                     .where( | ||||
|                                       (item) => | ||||
|                                           selectedTab.value == 0 || | ||||
|                                           (selectedTab.value == 1 && | ||||
|                                               item.type == 1) || | ||||
|                                           (selectedTab.value == 2 && | ||||
|                                               item.type != 1), | ||||
|                                     ) | ||||
|                                     .toList(); | ||||
|                             final item = filteredItems[index]; | ||||
|                             return ChatRoomListTile( | ||||
|                               room: item, | ||||
|                               isDirect: item.type == 1, | ||||
|                               onTap: () { | ||||
|                                 context.pushNamed( | ||||
|                                   'chatRoom', | ||||
|                                   pathParameters: {'id': item.id}, | ||||
|                                 ); | ||||
|                               }, | ||||
|           Consumer( | ||||
|             builder: (context, ref, _) { | ||||
|               final summaryState = ref.watch(chatSummaryProvider); | ||||
|               return summaryState.maybeWhen( | ||||
|                 loading: | ||||
|                     () => const LinearProgressIndicator( | ||||
|                       minHeight: 2, | ||||
|                       borderRadius: BorderRadius.zero, | ||||
|                     ), | ||||
|                 orElse: () => const SizedBox.shrink(), | ||||
|               ); | ||||
|             }, | ||||
|           ), | ||||
|           Expanded( | ||||
|             child: chats.when( | ||||
|               data: | ||||
|                   (items) => RefreshIndicator( | ||||
|                     onRefresh: | ||||
|                         () => Future.sync(() { | ||||
|                           ref.invalidate(chatroomsJoinedProvider); | ||||
|                         }), | ||||
|                     child: ListView.builder( | ||||
|                       padding: getTabbedPadding( | ||||
|                         context, | ||||
|                         bottom: callState.isConnected ? 96 : null, | ||||
|                       ), | ||||
|                       itemCount: | ||||
|                           items | ||||
|                               .where( | ||||
|                                 (item) => | ||||
|                                     selectedTab.value == 0 || | ||||
|                                     (selectedTab.value == 1 && | ||||
|                                         item.type == 1) || | ||||
|                                     (selectedTab.value == 2 && item.type != 1), | ||||
|                               ) | ||||
|                               .length, | ||||
|                       itemBuilder: (context, index) { | ||||
|                         final filteredItems = | ||||
|                             items | ||||
|                                 .where( | ||||
|                                   (item) => | ||||
|                                       selectedTab.value == 0 || | ||||
|                                       (selectedTab.value == 1 && | ||||
|                                           item.type == 1) || | ||||
|                                       (selectedTab.value == 2 && | ||||
|                                           item.type != 1), | ||||
|                                 ) | ||||
|                                 .toList(); | ||||
|                         final item = filteredItems[index]; | ||||
|                         return ChatRoomListTile( | ||||
|                           room: item, | ||||
|                           isDirect: item.type == 1, | ||||
|                           onTap: () { | ||||
|                             context.pushNamed( | ||||
|                               'chatRoom', | ||||
|                               pathParameters: {'id': item.id}, | ||||
|                             ); | ||||
|                           }, | ||||
|                         ), | ||||
|                       ), | ||||
|                   loading: | ||||
|                       () => const Center(child: CircularProgressIndicator()), | ||||
|                   error: | ||||
|                       (error, stack) => ResponseErrorWidget( | ||||
|                         error: error, | ||||
|                         onRetry: () { | ||||
|                           ref.invalidate(chatroomsJoinedProvider); | ||||
|                         }, | ||||
|                       ), | ||||
|                 ), | ||||
|               ), | ||||
|             ], | ||||
|           ), | ||||
|           Positioned( | ||||
|             left: 0, | ||||
|             right: 0, | ||||
|             bottom: getTabbedPadding(context).bottom + 8, | ||||
|             child: const CallOverlayBar().padding(horizontal: 16, vertical: 12), | ||||
|                         ); | ||||
|                       }, | ||||
|                     ), | ||||
|                   ), | ||||
|               loading: () => const Center(child: CircularProgressIndicator()), | ||||
|               error: | ||||
|                   (error, stack) => ResponseErrorWidget( | ||||
|                     error: error, | ||||
|                     onRetry: () { | ||||
|                       ref.invalidate(chatroomsJoinedProvider); | ||||
|                     }, | ||||
|                   ), | ||||
|             ), | ||||
|           ), | ||||
|         ], | ||||
|       ), | ||||
|   | ||||
| @@ -1070,6 +1070,10 @@ class _ChatInput extends HookConsumerWidget { | ||||
|                     item: attachments[idx], | ||||
|                     onRequestUpload: () => onUploadAttachment(idx), | ||||
|                     onDelete: () => onDeleteAttachment(idx), | ||||
|                     onUpdate: (value) { | ||||
|                       attachments[idx] = value; | ||||
|                       onAttachmentsChanged(attachments); | ||||
|                     }, | ||||
|                     onMove: (delta) => onMoveAttachment(idx, delta), | ||||
|                   ); | ||||
|                 }, | ||||
|   | ||||
| @@ -41,7 +41,7 @@ class ChatDetailScreen extends HookConsumerWidget { | ||||
|       try { | ||||
|         final client = ref.watch(apiClientProvider); | ||||
|         await client.patch( | ||||
|           '/chat/$id/members/me/notify', | ||||
|           '/sphere/chat/$id/members/me/notify', | ||||
|           data: {'notify_level': level}, | ||||
|         ); | ||||
|         ref.invalidate(chatroomIdentityProvider(id)); | ||||
| @@ -59,7 +59,7 @@ class ChatDetailScreen extends HookConsumerWidget { | ||||
|       try { | ||||
|         final client = ref.watch(apiClientProvider); | ||||
|         await client.patch( | ||||
|           '/chat/$id/members/me/notify', | ||||
|           '/sphere/chat/$id/members/me/notify', | ||||
|           data: {'break_until': until.toUtc().toIso8601String()}, | ||||
|         ); | ||||
|         ref.invalidate(chatroomProvider(id)); | ||||
| @@ -421,10 +421,10 @@ class _ChatRoomActionMenu extends HookConsumerWidget { | ||||
|                   showConfirmAlert( | ||||
|                     'deleteChatRoomHint'.tr(), | ||||
|                     'deleteChatRoom'.tr(), | ||||
|                   ).then((confirm) { | ||||
|                   ).then((confirm) async { | ||||
|                     if (confirm) { | ||||
|                       final client = ref.watch(apiClientProvider); | ||||
|                       client.delete('/sphere/chat/$id'); | ||||
|                       await client.delete('/sphere/chat/$id'); | ||||
|                       ref.invalidate(chatroomsJoinedProvider); | ||||
|                       if (context.mounted) { | ||||
|                         context.pop(); | ||||
| @@ -454,10 +454,10 @@ class _ChatRoomActionMenu extends HookConsumerWidget { | ||||
|                   showConfirmAlert( | ||||
|                     'leaveChatRoomHint'.tr(), | ||||
|                     'leaveChatRoom'.tr(), | ||||
|                   ).then((confirm) { | ||||
|                   ).then((confirm) async { | ||||
|                     if (confirm) { | ||||
|                       final client = ref.watch(apiClientProvider); | ||||
|                       client.delete('/sphere/chat/$id/members/me'); | ||||
|                       await client.delete('/sphere/chat/$id/members/me'); | ||||
|                       ref.invalidate(chatroomsJoinedProvider); | ||||
|                       if (context.mounted) { | ||||
|                         context.pop(); | ||||
|   | ||||
| @@ -7,7 +7,7 @@ part of 'room_detail.dart'; | ||||
| // ************************************************************************** | ||||
|  | ||||
| String _$chatMemberListNotifierHash() => | ||||
|     r'f2191a631ba00ae3de39ccac10e4cdd065ffee17'; | ||||
|     r'c8fbf4b95df6dae24b1ba21063e9a43351832974'; | ||||
|  | ||||
| /// Copied from Dart SDK | ||||
| class _SystemHash { | ||||
|   | ||||
| @@ -114,9 +114,9 @@ class CreatorHubShellScreen extends StatelessWidget { | ||||
|         isRoot: true, | ||||
|         child: Row( | ||||
|           children: [ | ||||
|             SizedBox(width: 360, child: const CreatorHubScreen(isAside: true)), | ||||
|             Flexible(flex: 2, child: const CreatorHubScreen(isAside: true)), | ||||
|             const VerticalDivider(width: 1), | ||||
|             Expanded(child: child), | ||||
|             Flexible(flex: 3, child: child), | ||||
|           ], | ||||
|         ), | ||||
|       ); | ||||
| @@ -380,6 +380,23 @@ class CreatorHubScreen extends HookConsumerWidget { | ||||
|                               ); | ||||
|                             }, | ||||
|                           ), | ||||
|                           ListTile( | ||||
|                             minTileHeight: 48, | ||||
|                             title: const Text('Polls'), | ||||
|                             trailing: const Icon(Symbols.chevron_right), | ||||
|                             leading: const Icon(Symbols.poll), | ||||
|                             contentPadding: const EdgeInsets.symmetric( | ||||
|                               horizontal: 24, | ||||
|                             ), | ||||
|                             onTap: () { | ||||
|                               context.pushNamed( | ||||
|                                 'creatorPolls', | ||||
|                                 pathParameters: { | ||||
|                                   'name': currentPublisher.value!.name, | ||||
|                                 }, | ||||
|                               ); | ||||
|                             }, | ||||
|                           ), | ||||
|                           ListTile( | ||||
|                             minTileHeight: 48, | ||||
|                             title: Text('publisherMembers').tr(), | ||||
|   | ||||
							
								
								
									
										175
									
								
								lib/screens/creators/poll/poll_list.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										175
									
								
								lib/screens/creators/poll/poll_list.dart
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,175 @@ | ||||
| import 'package:flutter/material.dart'; | ||||
| import 'package:gap/gap.dart'; | ||||
| import 'package:go_router/go_router.dart'; | ||||
| import 'package:hooks_riverpod/hooks_riverpod.dart'; | ||||
| import 'package:island/models/poll.dart'; | ||||
| import 'package:island/pods/network.dart'; | ||||
| import 'package:material_symbols_icons/symbols.dart'; | ||||
| import 'package:riverpod_annotation/riverpod_annotation.dart'; | ||||
| import 'package:riverpod_paging_utils/riverpod_paging_utils.dart'; | ||||
|  | ||||
| part 'poll_list.g.dart'; | ||||
|  | ||||
| @riverpod | ||||
| class PollListNotifier extends _$PollListNotifier | ||||
|     with CursorPagingNotifierMixin<SnPoll> { | ||||
|   static const int _pageSize = 20; | ||||
|  | ||||
|   @override | ||||
|   Future<CursorPagingData<SnPoll>> build(String? pubName) { | ||||
|     // immediately load first page | ||||
|     return fetch(cursor: null); | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   Future<CursorPagingData<SnPoll>> fetch({required String? cursor}) async { | ||||
|     final client = ref.read(apiClientProvider); | ||||
|     final offset = cursor == null ? 0 : int.parse(cursor); | ||||
|  | ||||
|     // read the current family argument passed to provider | ||||
|     final currentPub = pubName; | ||||
|     final queryParams = { | ||||
|       'offset': offset, | ||||
|       'take': _pageSize, | ||||
|       if (currentPub != null) 'pub': currentPub, | ||||
|     }; | ||||
|  | ||||
|     final response = await client.get( | ||||
|       '/sphere/polls/me', | ||||
|       queryParameters: queryParams, | ||||
|     ); | ||||
|     final total = int.parse(response.headers.value('X-Total') ?? '0'); | ||||
|     final List<dynamic> data = response.data; | ||||
|     final items = data.map((json) => SnPoll.fromJson(json)).toList(); | ||||
|  | ||||
|     final hasMore = offset + items.length < total; | ||||
|     final nextCursor = hasMore ? (offset + items.length).toString() : null; | ||||
|  | ||||
|     return CursorPagingData( | ||||
|       items: items, | ||||
|       hasMore: hasMore, | ||||
|       nextCursor: nextCursor, | ||||
|     ); | ||||
|   } | ||||
| } | ||||
|  | ||||
| class CreatorPollListScreen extends HookConsumerWidget { | ||||
|   const CreatorPollListScreen({super.key, required this.pubName}); | ||||
|  | ||||
|   final String pubName; | ||||
|  | ||||
|   Future<void> _createPoll(BuildContext context) async { | ||||
|     final result = await GoRouter.of( | ||||
|       context, | ||||
|     ).pushNamed('creatorPollNew', pathParameters: {'name': pubName}); | ||||
|     if (result is SnPoll && context.mounted) { | ||||
|       Navigator.of(context).maybePop(result); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context, WidgetRef ref) { | ||||
|     return Scaffold( | ||||
|       appBar: AppBar(title: const Text('Polls')), | ||||
|       floatingActionButton: FloatingActionButton( | ||||
|         onPressed: () => _createPoll(context), | ||||
|         child: const Icon(Icons.add), | ||||
|       ), | ||||
|       body: RefreshIndicator( | ||||
|         onRefresh: () => ref.refresh(pollListNotifierProvider(pubName).future), | ||||
|         child: CustomScrollView( | ||||
|           slivers: [ | ||||
|             PagingHelperSliverView( | ||||
|               provider: pollListNotifierProvider(pubName), | ||||
|               futureRefreshable: pollListNotifierProvider(pubName).future, | ||||
|               notifierRefreshable: pollListNotifierProvider(pubName).notifier, | ||||
|               contentBuilder: | ||||
|                   (data, widgetCount, endItemView) => SliverList.builder( | ||||
|                     itemCount: widgetCount, | ||||
|                     itemBuilder: (context, index) { | ||||
|                       if (index == widgetCount - 1) { | ||||
|                         return endItemView; | ||||
|                       } | ||||
|                       final poll = data.items[index]; | ||||
|                       return _CreatorPollItem(poll: poll, pubName: pubName); | ||||
|                     }, | ||||
|                   ), | ||||
|             ), | ||||
|           ], | ||||
|         ), | ||||
|       ), | ||||
|     ); | ||||
|   } | ||||
| } | ||||
|  | ||||
| class _CreatorPollItem extends StatelessWidget { | ||||
|   final String pubName; | ||||
|   const _CreatorPollItem({required this.poll, required this.pubName}); | ||||
|  | ||||
|   final SnPoll poll; | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     final theme = Theme.of(context); | ||||
|     final ended = poll.endedAt; | ||||
|     final endedText = | ||||
|         ended == null | ||||
|             ? 'No end' | ||||
|             : MaterialLocalizations.of(context).formatFullDate(ended); | ||||
|  | ||||
|     return Card( | ||||
|       margin: const EdgeInsets.symmetric(horizontal: 12, vertical: 6), | ||||
|       clipBehavior: Clip.antiAlias, | ||||
|       child: ListTile( | ||||
|         title: Text(poll.title ?? 'Untitled poll'), | ||||
|         subtitle: Column( | ||||
|           crossAxisAlignment: CrossAxisAlignment.start, | ||||
|           children: [ | ||||
|             if (poll.description != null && poll.description!.isNotEmpty) | ||||
|               Padding( | ||||
|                 padding: const EdgeInsets.only(top: 4), | ||||
|                 child: Text( | ||||
|                   poll.description!, | ||||
|                   maxLines: 2, | ||||
|                   overflow: TextOverflow.ellipsis, | ||||
|                 ), | ||||
|               ), | ||||
|             Padding( | ||||
|               padding: const EdgeInsets.only(top: 4), | ||||
|               child: Text( | ||||
|                 'Questions: ${poll.questions.length} · Ends: $endedText', | ||||
|                 style: theme.textTheme.bodySmall, | ||||
|               ), | ||||
|             ), | ||||
|           ], | ||||
|         ), | ||||
|         trailing: PopupMenuButton<String>( | ||||
|           itemBuilder: | ||||
|               (context) => [ | ||||
|                 PopupMenuItem( | ||||
|                   child: Row( | ||||
|                     children: [ | ||||
|                       const Icon(Symbols.edit), | ||||
|                       const Gap(16), | ||||
|                       Text('Edit'), | ||||
|                     ], | ||||
|                   ), | ||||
|                   onTap: () { | ||||
|                     GoRouter.of(context).pushNamed( | ||||
|                       'creatorPollEdit', | ||||
|                       pathParameters: {'name': pubName, 'id': poll.id}, | ||||
|                     ); | ||||
|                   }, | ||||
|                 ), | ||||
|               ], | ||||
|         ), | ||||
|         onTap: () { | ||||
|           // Open editor for edit | ||||
|           // Navigator push by path to keep consistency with rest of app: | ||||
|           // Note: pub name string may be required in route; when absent, route may need query or pick later. | ||||
|           // For safety, just do nothing if no publisher in list item. | ||||
|         }, | ||||
|       ), | ||||
|     ); | ||||
|   } | ||||
| } | ||||
							
								
								
									
										179
									
								
								lib/screens/creators/poll/poll_list.g.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										179
									
								
								lib/screens/creators/poll/poll_list.g.dart
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,179 @@ | ||||
| // GENERATED CODE - DO NOT MODIFY BY HAND | ||||
|  | ||||
| part of 'poll_list.dart'; | ||||
|  | ||||
| // ************************************************************************** | ||||
| // RiverpodGenerator | ||||
| // ************************************************************************** | ||||
|  | ||||
| String _$pollListNotifierHash() => r'd3da24ff6bbb8f35b06d57fc41625dc0312508e4'; | ||||
|  | ||||
| /// Copied from Dart SDK | ||||
| class _SystemHash { | ||||
|   _SystemHash._(); | ||||
|  | ||||
|   static int combine(int hash, int value) { | ||||
|     // ignore: parameter_assignments | ||||
|     hash = 0x1fffffff & (hash + value); | ||||
|     // ignore: parameter_assignments | ||||
|     hash = 0x1fffffff & (hash + ((0x0007ffff & hash) << 10)); | ||||
|     return hash ^ (hash >> 6); | ||||
|   } | ||||
|  | ||||
|   static int finish(int hash) { | ||||
|     // ignore: parameter_assignments | ||||
|     hash = 0x1fffffff & (hash + ((0x03ffffff & hash) << 3)); | ||||
|     // ignore: parameter_assignments | ||||
|     hash = hash ^ (hash >> 11); | ||||
|     return 0x1fffffff & (hash + ((0x00003fff & hash) << 15)); | ||||
|   } | ||||
| } | ||||
|  | ||||
| abstract class _$PollListNotifier | ||||
|     extends BuildlessAutoDisposeAsyncNotifier<CursorPagingData<SnPoll>> { | ||||
|   late final String? pubName; | ||||
|  | ||||
|   FutureOr<CursorPagingData<SnPoll>> build(String? pubName); | ||||
| } | ||||
|  | ||||
| /// See also [PollListNotifier]. | ||||
| @ProviderFor(PollListNotifier) | ||||
| const pollListNotifierProvider = PollListNotifierFamily(); | ||||
|  | ||||
| /// See also [PollListNotifier]. | ||||
| class PollListNotifierFamily | ||||
|     extends Family<AsyncValue<CursorPagingData<SnPoll>>> { | ||||
|   /// See also [PollListNotifier]. | ||||
|   const PollListNotifierFamily(); | ||||
|  | ||||
|   /// See also [PollListNotifier]. | ||||
|   PollListNotifierProvider call(String? pubName) { | ||||
|     return PollListNotifierProvider(pubName); | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   PollListNotifierProvider getProviderOverride( | ||||
|     covariant PollListNotifierProvider provider, | ||||
|   ) { | ||||
|     return call(provider.pubName); | ||||
|   } | ||||
|  | ||||
|   static const Iterable<ProviderOrFamily>? _dependencies = null; | ||||
|  | ||||
|   @override | ||||
|   Iterable<ProviderOrFamily>? get dependencies => _dependencies; | ||||
|  | ||||
|   static const Iterable<ProviderOrFamily>? _allTransitiveDependencies = null; | ||||
|  | ||||
|   @override | ||||
|   Iterable<ProviderOrFamily>? get allTransitiveDependencies => | ||||
|       _allTransitiveDependencies; | ||||
|  | ||||
|   @override | ||||
|   String? get name => r'pollListNotifierProvider'; | ||||
| } | ||||
|  | ||||
| /// See also [PollListNotifier]. | ||||
| class PollListNotifierProvider | ||||
|     extends | ||||
|         AutoDisposeAsyncNotifierProviderImpl< | ||||
|           PollListNotifier, | ||||
|           CursorPagingData<SnPoll> | ||||
|         > { | ||||
|   /// See also [PollListNotifier]. | ||||
|   PollListNotifierProvider(String? pubName) | ||||
|     : this._internal( | ||||
|         () => PollListNotifier()..pubName = pubName, | ||||
|         from: pollListNotifierProvider, | ||||
|         name: r'pollListNotifierProvider', | ||||
|         debugGetCreateSourceHash: | ||||
|             const bool.fromEnvironment('dart.vm.product') | ||||
|                 ? null | ||||
|                 : _$pollListNotifierHash, | ||||
|         dependencies: PollListNotifierFamily._dependencies, | ||||
|         allTransitiveDependencies: | ||||
|             PollListNotifierFamily._allTransitiveDependencies, | ||||
|         pubName: pubName, | ||||
|       ); | ||||
|  | ||||
|   PollListNotifierProvider._internal( | ||||
|     super._createNotifier, { | ||||
|     required super.name, | ||||
|     required super.dependencies, | ||||
|     required super.allTransitiveDependencies, | ||||
|     required super.debugGetCreateSourceHash, | ||||
|     required super.from, | ||||
|     required this.pubName, | ||||
|   }) : super.internal(); | ||||
|  | ||||
|   final String? pubName; | ||||
|  | ||||
|   @override | ||||
|   FutureOr<CursorPagingData<SnPoll>> runNotifierBuild( | ||||
|     covariant PollListNotifier notifier, | ||||
|   ) { | ||||
|     return notifier.build(pubName); | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   Override overrideWith(PollListNotifier Function() create) { | ||||
|     return ProviderOverride( | ||||
|       origin: this, | ||||
|       override: PollListNotifierProvider._internal( | ||||
|         () => create()..pubName = pubName, | ||||
|         from: from, | ||||
|         name: null, | ||||
|         dependencies: null, | ||||
|         allTransitiveDependencies: null, | ||||
|         debugGetCreateSourceHash: null, | ||||
|         pubName: pubName, | ||||
|       ), | ||||
|     ); | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   AutoDisposeAsyncNotifierProviderElement< | ||||
|     PollListNotifier, | ||||
|     CursorPagingData<SnPoll> | ||||
|   > | ||||
|   createElement() { | ||||
|     return _PollListNotifierProviderElement(this); | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   bool operator ==(Object other) { | ||||
|     return other is PollListNotifierProvider && other.pubName == pubName; | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   int get hashCode { | ||||
|     var hash = _SystemHash.combine(0, runtimeType.hashCode); | ||||
|     hash = _SystemHash.combine(hash, pubName.hashCode); | ||||
|  | ||||
|     return _SystemHash.finish(hash); | ||||
|   } | ||||
| } | ||||
|  | ||||
| @Deprecated('Will be removed in 3.0. Use Ref instead') | ||||
| // ignore: unused_element | ||||
| mixin PollListNotifierRef | ||||
|     on AutoDisposeAsyncNotifierProviderRef<CursorPagingData<SnPoll>> { | ||||
|   /// The parameter `pubName` of this provider. | ||||
|   String? get pubName; | ||||
| } | ||||
|  | ||||
| class _PollListNotifierProviderElement | ||||
|     extends | ||||
|         AutoDisposeAsyncNotifierProviderElement< | ||||
|           PollListNotifier, | ||||
|           CursorPagingData<SnPoll> | ||||
|         > | ||||
|     with PollListNotifierRef { | ||||
|   _PollListNotifierProviderElement(super.provider); | ||||
|  | ||||
|   @override | ||||
|   String? get pubName => (origin as PollListNotifierProvider).pubName; | ||||
| } | ||||
|  | ||||
| // ignore_for_file: type=lint | ||||
| // ignore_for_file: subtype_of_sealed_class, invalid_use_of_internal_member, invalid_use_of_visible_for_testing_member, deprecated_member_use_from_same_package | ||||
| @@ -26,7 +26,7 @@ part 'pack_detail.freezed.dart'; | ||||
| @riverpod | ||||
| Future<List<SnSticker>> stickerPackContent(Ref ref, String packId) async { | ||||
|   final apiClient = ref.watch(apiClientProvider); | ||||
|   final resp = await apiClient.get('/stickers/$packId/content'); | ||||
|   final resp = await apiClient.get('/sphere/stickers/$packId/content'); | ||||
|   return resp.data | ||||
|       .map<SnSticker>((e) => SnSticker.fromJson(e)) | ||||
|       .cast<SnSticker>() | ||||
| @@ -74,13 +74,16 @@ class StickerPackDetailScreen extends HookConsumerWidget { | ||||
|           IconButton( | ||||
|             icon: const Icon(Symbols.add_circle), | ||||
|             onPressed: () { | ||||
|               context.pushNamed('creatorStickerNew', pathParameters: {'packId': id}).then(( | ||||
|                 value, | ||||
|               ) { | ||||
|                 if (value != null) { | ||||
|                   ref.invalidate(stickerPackContentProvider(id)); | ||||
|                 } | ||||
|               }); | ||||
|               context | ||||
|                   .pushNamed( | ||||
|                     'creatorStickerNew', | ||||
|                     pathParameters: {'name': pubName, 'packId': id}, | ||||
|                   ) | ||||
|                   .then((value) { | ||||
|                     if (value != null) { | ||||
|                       ref.invalidate(stickerPackContentProvider(id)); | ||||
|                     } | ||||
|                   }); | ||||
|             }, | ||||
|           ), | ||||
|           _StickerPackActionMenu( | ||||
| @@ -173,9 +176,13 @@ class StickerPackDetailScreen extends HookConsumerWidget { | ||||
|                                         title: 'edit'.tr(), | ||||
|                                         image: MenuImage.icon(Symbols.edit), | ||||
|                                         callback: () { | ||||
|                                           context.pushNamed( | ||||
|                                           context | ||||
|                                               .pushNamed( | ||||
|                                                 'creatorStickerEdit', | ||||
|                                                 pathParameters: {'packId': id, 'id': sticker.id}, | ||||
|                                                 pathParameters: { | ||||
|                                                   'packId': id, | ||||
|                                                   'id': sticker.id, | ||||
|                                                 }, | ||||
|                                               ) | ||||
|                                               .then((value) { | ||||
|                                                 if (value != null) { | ||||
| @@ -259,9 +266,7 @@ class _StickerPackActionMenu extends HookConsumerWidget { | ||||
|           (context) => [ | ||||
|             PopupMenuItem( | ||||
|               onTap: () { | ||||
|                 context.push( | ||||
|                   '/creators/$pubName/stickers/$packId/edit', | ||||
|                 ); | ||||
|                 context.push('/creators/$pubName/stickers/$packId/edit'); | ||||
|               }, | ||||
|               child: Row( | ||||
|                 children: [ | ||||
|   | ||||
| @@ -7,7 +7,7 @@ part of 'pack_detail.dart'; | ||||
| // ************************************************************************** | ||||
|  | ||||
| String _$stickerPackContentHash() => | ||||
|     r'78de848fba1f341f217f8ae4b9eef2d8afa67964'; | ||||
|     r'42d74f51022e67e35cb601c2f30f4f02e1f2be9d'; | ||||
|  | ||||
| /// Copied from Dart SDK | ||||
| class _SystemHash { | ||||
|   | ||||
| @@ -31,7 +31,7 @@ class StickersScreen extends HookConsumerWidget { | ||||
|               context | ||||
|                   .pushNamed( | ||||
|                     'creatorStickerPackNew', | ||||
|                     queryParameters: {'pubName': pubName}, | ||||
|                     queryParameters: {'name': pubName}, | ||||
|                   ) | ||||
|                   .then((value) { | ||||
|                     if (value != null) { | ||||
| @@ -76,7 +76,7 @@ class SliverStickerPacksList extends HookConsumerWidget { | ||||
|                 onTap: () { | ||||
|                   context.pushNamed( | ||||
|                     'creatorStickerPackDetail', | ||||
|                     pathParameters: {'pubName': pubName, 'packId': sticker.id}, | ||||
|                     pathParameters: {'name': pubName, 'packId': sticker.id}, | ||||
|                   ); | ||||
|                 }, | ||||
|               ); | ||||
|   | ||||
| @@ -51,12 +51,9 @@ class DeveloperHubShellScreen extends StatelessWidget { | ||||
|         isRoot: true, | ||||
|         child: Row( | ||||
|           children: [ | ||||
|             SizedBox( | ||||
|               width: 360, | ||||
|               child: const DeveloperHubScreen(isAside: true), | ||||
|             ), | ||||
|             Flexible(flex: 2, child: const DeveloperHubScreen(isAside: true)), | ||||
|             const VerticalDivider(width: 1), | ||||
|             Expanded(child: child), | ||||
|             Flexible(flex: 3, child: child), | ||||
|           ], | ||||
|         ), | ||||
|       ); | ||||
| @@ -114,7 +111,7 @@ class DeveloperHubScreen extends HookConsumerWidget { | ||||
|     ); | ||||
|  | ||||
|     return AppScaffold( | ||||
|       noBackground: false, | ||||
|       isNoBackground: false, | ||||
|       appBar: AppBar( | ||||
|         leading: !isWide ? const PageBackButton() : null, | ||||
|         title: Text('developerHub').tr(), | ||||
|   | ||||
| @@ -7,7 +7,7 @@ part of 'articles.dart'; | ||||
| // ************************************************************************** | ||||
|  | ||||
| String _$articlesListNotifierHash() => | ||||
|     r'924f2344c3bbf0ff7b92fe69e88d3b64a534b538'; | ||||
|     r'579741af4d90c7c81f2e2697e57c4895b7a9dabc'; | ||||
|  | ||||
| /// Copied from Dart SDK | ||||
| class _SystemHash { | ||||
|   | ||||
| @@ -17,6 +17,7 @@ class DiscoveryRealmsScreen extends HookConsumerWidget { | ||||
|     final currentQuery = useState<String?>(null); | ||||
|  | ||||
|     return AppScaffold( | ||||
|       isNoBackground: false, | ||||
|       appBar: AppBar(title: Text('discoverRealms'.tr())), | ||||
|       body: Stack( | ||||
|         children: [ | ||||
|   | ||||
| @@ -9,8 +9,11 @@ import 'package:island/models/activity.dart'; | ||||
| import 'package:island/models/publisher.dart'; | ||||
| import 'package:island/models/realm.dart'; | ||||
| import 'package:island/models/webfeed.dart'; | ||||
| import 'package:island/pods/event_calendar.dart'; | ||||
| import 'package:island/pods/userinfo.dart'; | ||||
| import 'package:island/services/responsive.dart'; | ||||
| import 'package:island/widgets/account/event_calendar.dart'; | ||||
| import 'package:island/widgets/account/fortune_graph.dart'; | ||||
| import 'package:island/widgets/app_scaffold.dart'; | ||||
| import 'package:island/models/post.dart'; | ||||
| import 'package:island/widgets/check_in.dart'; | ||||
| @@ -27,42 +30,11 @@ import 'package:styled_widget/styled_widget.dart'; | ||||
|  | ||||
| part 'explore.g.dart'; | ||||
|  | ||||
| class ExploreShellScreen extends HookConsumerWidget { | ||||
|   final Widget child; | ||||
|   const ExploreShellScreen({super.key, required this.child}); | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context, WidgetRef ref) { | ||||
|     final isWide = MediaQuery.of(context).size.width > 640; | ||||
|  | ||||
|     if (isWide) { | ||||
|       return AppBackground( | ||||
|         isRoot: true, | ||||
|         child: Row( | ||||
|           children: [ | ||||
|             Flexible(flex: 2, child: ExploreScreen(isAside: true)), | ||||
|             VerticalDivider(width: 1), | ||||
|             Flexible(flex: 3, child: child), | ||||
|           ], | ||||
|         ), | ||||
|       ); | ||||
|     } | ||||
|  | ||||
|     return AppBackground(isRoot: true, child: child); | ||||
|   } | ||||
| } | ||||
|  | ||||
| class ExploreScreen extends HookConsumerWidget { | ||||
|   final bool isAside; | ||||
|   const ExploreScreen({super.key, this.isAside = false}); | ||||
|   const ExploreScreen({super.key}); | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context, WidgetRef ref) { | ||||
|     final isWide = isWideScreen(context); | ||||
|     if (isWide && !isAside) { | ||||
|       return const EmptyPageHolder(); | ||||
|     } | ||||
|  | ||||
|     final tabController = useTabController(initialLength: 3); | ||||
|     final currentFilter = useState<String?>(null); | ||||
|  | ||||
| @@ -89,8 +61,33 @@ class ExploreScreen extends HookConsumerWidget { | ||||
|       activityListNotifierProvider(currentFilter.value).notifier, | ||||
|     ); | ||||
|  | ||||
|     final now = DateTime.now(); | ||||
|  | ||||
|     final query = useState( | ||||
|       EventCalendarQuery(uname: 'me', year: now.year, month: now.month), | ||||
|     ); | ||||
|  | ||||
|     final events = ref.watch(eventCalendarProvider(query.value)); | ||||
|  | ||||
|     final selectedDay = useState(now); | ||||
|  | ||||
|     void onMonthChanged(int year, int month) { | ||||
|       query.value = EventCalendarQuery( | ||||
|         uname: query.value.uname, | ||||
|         year: year, | ||||
|         month: month, | ||||
|       ); | ||||
|     } | ||||
|  | ||||
|     // Function to handle day selection for synchronizing between widgets | ||||
|     void onDaySelected(DateTime day) { | ||||
|       selectedDay.value = day; | ||||
|     } | ||||
|  | ||||
|     final user = ref.watch(userInfoProvider); | ||||
|  | ||||
|     return AppScaffold( | ||||
|       extendBody: false, // Prevent conflicts with tabs navigation | ||||
|       isNoBackground: false, | ||||
|       appBar: AppBar( | ||||
|         toolbarHeight: 0, | ||||
|         bottom: PreferredSize( | ||||
| @@ -172,35 +169,126 @@ class ExploreScreen extends HookConsumerWidget { | ||||
|               ), | ||||
|         ), | ||||
|       ), | ||||
|       floatingActionButton: FloatingActionButton( | ||||
|         heroTag: Key("explore-page-fab"), | ||||
|         onPressed: () { | ||||
|           context.pushNamed('postCompose').then((value) { | ||||
|             if (value != null) { | ||||
|               activitiesNotifier.forceRefresh(); | ||||
|             } | ||||
|           }); | ||||
|       floatingActionButton: InkWell( | ||||
|         onLongPress: () { | ||||
|           context.pushNamed('postCompose', queryParameters: {'type': '1'}).then( | ||||
|             (value) { | ||||
|               if (value != null) { | ||||
|                 activitiesNotifier.forceRefresh(); | ||||
|               } | ||||
|             }, | ||||
|           ); | ||||
|         }, | ||||
|         child: const Icon(Symbols.edit), | ||||
|         child: FloatingActionButton( | ||||
|           heroTag: Key("explore-page-fab"), | ||||
|           onPressed: () { | ||||
|             context.pushNamed('postCompose').then((value) { | ||||
|               if (value != null) { | ||||
|                 activitiesNotifier.forceRefresh(); | ||||
|               } | ||||
|             }); | ||||
|           }, | ||||
|           child: const Icon(Symbols.edit), | ||||
|         ), | ||||
|       ), | ||||
|       floatingActionButtonLocation: TabbedFabLocation(context), | ||||
|       body: TabBarView( | ||||
|         controller: tabController, | ||||
|         physics: const NeverScrollableScrollPhysics(), | ||||
|         children: [ | ||||
|           _buildActivityList(ref, null), | ||||
|           _buildActivityList(ref, 'subscriptions'), | ||||
|           _buildActivityList(ref, 'friends'), | ||||
|         ], | ||||
|       body: Builder( | ||||
|         builder: (context) { | ||||
|           final isWider = isWiderScreen(context); | ||||
|  | ||||
|           final bodyView = _buildActivityList( | ||||
|             context, | ||||
|             ref, | ||||
|             currentFilter.value, | ||||
|           ); | ||||
|  | ||||
|           if (isWider) { | ||||
|             return Row( | ||||
|               children: [ | ||||
|                 Flexible(flex: 3, child: bodyView.padding(left: 8)), | ||||
|                 if (user.value != null) | ||||
|                   Flexible( | ||||
|                     flex: 2, | ||||
|                     child: SingleChildScrollView( | ||||
|                       child: Column( | ||||
|                         children: [ | ||||
|                           CheckInWidget( | ||||
|                             margin: EdgeInsets.only( | ||||
|                               left: 8, | ||||
|                               right: 12, | ||||
|                               top: 16, | ||||
|                             ), | ||||
|                             onChecked: () { | ||||
|                               ref.invalidate( | ||||
|                                 eventCalendarProvider(query.value), | ||||
|                               ); | ||||
|                             }, | ||||
|                           ), | ||||
|                           Card( | ||||
|                             margin: EdgeInsets.only(left: 8, right: 12, top: 8), | ||||
|                             child: Column( | ||||
|                               children: [ | ||||
|                                 // Use the reusable EventCalendarWidget | ||||
|                                 EventCalendarWidget( | ||||
|                                   events: events, | ||||
|                                   initialDate: now, | ||||
|                                   showEventDetails: true, | ||||
|                                   onMonthChanged: onMonthChanged, | ||||
|                                   onDaySelected: onDaySelected, | ||||
|                                 ), | ||||
|                               ], | ||||
|                             ), | ||||
|                           ), | ||||
|                           FortuneGraphWidget( | ||||
|                             margin: EdgeInsets.only(left: 8, right: 12, top: 8), | ||||
|                             events: events, | ||||
|                             constrainWidth: true, | ||||
|                             onPointSelected: onDaySelected, | ||||
|                           ), | ||||
|                         ], | ||||
|                       ), | ||||
|                     ), | ||||
|                   ) | ||||
|                 else | ||||
|                   Flexible( | ||||
|                     flex: 2, | ||||
|                     child: Column( | ||||
|                       mainAxisAlignment: MainAxisAlignment.center, | ||||
|                       crossAxisAlignment: CrossAxisAlignment.start, | ||||
|                       children: [ | ||||
|                         Text( | ||||
|                           'Welcome to\nthe Solar Network', | ||||
|                           style: Theme.of(context).textTheme.titleLarge, | ||||
|                         ).bold(), | ||||
|                         const Gap(2), | ||||
|                         Text( | ||||
|                           'Login to explore more!', | ||||
|                           style: Theme.of(context).textTheme.bodyLarge, | ||||
|                         ), | ||||
|                       ], | ||||
|                     ).padding(horizontal: 36, vertical: 16), | ||||
|                   ), | ||||
|               ], | ||||
|             ); | ||||
|           } | ||||
|  | ||||
|           return bodyView; | ||||
|         }, | ||||
|       ), | ||||
|     ); | ||||
|   } | ||||
|  | ||||
|   Widget _buildActivityList(WidgetRef ref, String? filter) { | ||||
|   Widget _buildActivityList( | ||||
|     BuildContext context, | ||||
|     WidgetRef ref, | ||||
|     String? filter, | ||||
|   ) { | ||||
|     final activitiesNotifier = ref.watch( | ||||
|       activityListNotifierProvider(filter).notifier, | ||||
|     ); | ||||
|  | ||||
|     final isWider = isWiderScreen(context); | ||||
|  | ||||
|     return RefreshIndicator( | ||||
|       onRefresh: () => Future.sync(activitiesNotifier.forceRefresh), | ||||
|       child: PagingHelperView( | ||||
| @@ -214,7 +302,7 @@ class ExploreScreen extends HookConsumerWidget { | ||||
|                 widgetCount: widgetCount, | ||||
|                 endItemView: endItemView, | ||||
|                 activitiesNotifier: activitiesNotifier, | ||||
|                 contentOnly: filter != null, | ||||
|                 contentOnly: isWider || filter != null, | ||||
|               ), | ||||
|             ), | ||||
|       ), | ||||
| @@ -232,56 +320,62 @@ class _DiscoveryActivityItem extends StatelessWidget { | ||||
|     final items = data['items'] as List; | ||||
|     final type = items.firstOrNull?['type'] ?? 'unknown'; | ||||
|  | ||||
|     return Column( | ||||
|       crossAxisAlignment: CrossAxisAlignment.start, | ||||
|       children: [ | ||||
|         Row( | ||||
|           crossAxisAlignment: CrossAxisAlignment.center, | ||||
|           children: [ | ||||
|             const Icon(Symbols.explore, size: 19), | ||||
|             const Gap(8), | ||||
|             Text( | ||||
|               (switch (type) { | ||||
|                 'realm' => 'discoverRealms', | ||||
|                 'publisher' => 'discoverPublishers', | ||||
|                 'article' => 'discoverWebArticles', | ||||
|                 _ => 'unknown', | ||||
|               }).tr(), | ||||
|               style: Theme.of(context).textTheme.titleMedium, | ||||
|             ).padding(top: 1), | ||||
|           ], | ||||
|         ).padding(horizontal: 20, top: 8, bottom: 4), | ||||
|         SizedBox( | ||||
|           height: 180, | ||||
|           child: ListView.builder( | ||||
|             scrollDirection: Axis.horizontal, | ||||
|             itemCount: items.length, | ||||
|             padding: const EdgeInsets.symmetric(horizontal: 8), | ||||
|             itemBuilder: (context, index) { | ||||
|               final item = items[index]; | ||||
|               switch (type) { | ||||
|                 case 'realm': | ||||
|                   return RealmCard( | ||||
|                     realm: SnRealm.fromJson(item['data']), | ||||
|                     maxWidth: 280, | ||||
|                   ); | ||||
|                 case 'publisher': | ||||
|                   return PublisherCard( | ||||
|                     publisher: SnPublisher.fromJson(item['data']), | ||||
|                     maxWidth: 280, | ||||
|                   ); | ||||
|                 case 'article': | ||||
|                   return WebArticleCard( | ||||
|                     article: SnWebArticle.fromJson(item['data']), | ||||
|                     maxWidth: 280, | ||||
|                   ); | ||||
|                 default: | ||||
|                   return Placeholder(); | ||||
|               } | ||||
|             }, | ||||
|           ), | ||||
|         ).padding(bottom: 4), | ||||
|       ], | ||||
|     return Card( | ||||
|       margin: EdgeInsets.symmetric(horizontal: 8, vertical: 4), | ||||
|       child: Column( | ||||
|         crossAxisAlignment: CrossAxisAlignment.start, | ||||
|         children: [ | ||||
|           Row( | ||||
|             crossAxisAlignment: CrossAxisAlignment.center, | ||||
|             children: [ | ||||
|               const Icon(Symbols.explore, size: 19), | ||||
|               const Gap(8), | ||||
|               Text( | ||||
|                 (switch (type) { | ||||
|                   'realm' => 'discoverRealms', | ||||
|                   'publisher' => 'discoverPublishers', | ||||
|                   'article' => 'discoverWebArticles', | ||||
|                   _ => 'unknown', | ||||
|                 }).tr(), | ||||
|                 style: Theme.of(context).textTheme.titleMedium, | ||||
|               ).padding(top: 1), | ||||
|             ], | ||||
|           ).padding(horizontal: 20, top: 8, bottom: 4), | ||||
|           SizedBox( | ||||
|             height: 180, | ||||
|             child: ConstrainedBox( | ||||
|               constraints: const BoxConstraints(maxHeight: 200), | ||||
|               child: CarouselView.weighted( | ||||
|                 flexWeights: | ||||
|                     isWideScreen(context) ? <int>[3, 2, 1] : <int>[4, 1], | ||||
|                 consumeMaxWeight: false, | ||||
|                 enableSplash: false, | ||||
|                 shape: RoundedRectangleBorder( | ||||
|                   borderRadius: BorderRadius.all(Radius.circular(8)), | ||||
|                 ), | ||||
|                 children: [ | ||||
|                   for (final item in items) | ||||
|                     switch (type) { | ||||
|                       'realm' => RealmCard( | ||||
|                         realm: SnRealm.fromJson(item['data']), | ||||
|                         maxWidth: 280, | ||||
|                       ), | ||||
|                       'publisher' => PublisherCard( | ||||
|                         publisher: SnPublisher.fromJson(item['data']), | ||||
|                         maxWidth: 280, | ||||
|                       ), | ||||
|                       'article' => WebArticleCard( | ||||
|                         article: SnWebArticle.fromJson(item['data']), | ||||
|                         maxWidth: 280, | ||||
|                       ), | ||||
|                       _ => Placeholder(), | ||||
|                     }, | ||||
|                 ], | ||||
|               ), | ||||
|             ), | ||||
|           ).padding(bottom: 8, horizontal: 8), | ||||
|         ], | ||||
|       ), | ||||
|     ); | ||||
|   } | ||||
| } | ||||
| @@ -307,8 +401,13 @@ class _ActivityListView extends HookConsumerWidget { | ||||
|  | ||||
|     return CustomScrollView( | ||||
|       slivers: [ | ||||
|         SliverGap(12), | ||||
|         if (user.value != null && !contentOnly) | ||||
|           SliverToBoxAdapter(child: CheckInWidget()), | ||||
|           SliverToBoxAdapter( | ||||
|             child: CheckInWidget( | ||||
|               margin: EdgeInsets.only(left: 8, right: 8, bottom: 4), | ||||
|             ), | ||||
|           ), | ||||
|         SliverList.builder( | ||||
|           itemCount: widgetCount, | ||||
|           itemBuilder: (context, index) { | ||||
| @@ -325,19 +424,9 @@ class _ActivityListView extends HookConsumerWidget { | ||||
|             switch (item.type) { | ||||
|               case 'posts.new': | ||||
|               case 'posts.new.replies': | ||||
|                 final isReply = item.type == 'posts.new.replies'; | ||||
|                 itemWidget = PostItem( | ||||
|                   backgroundColor: | ||||
|                       isWideScreen(context) ? Colors.transparent : null, | ||||
|                 itemWidget = PostActionableItem( | ||||
|                   borderRadius: 8, | ||||
|                   item: SnPost.fromJson(item.data!), | ||||
|                   padding: | ||||
|                       isReply | ||||
|                           ? const EdgeInsets.only( | ||||
|                             left: 16, | ||||
|                             right: 16, | ||||
|                             bottom: 16, | ||||
|                           ) | ||||
|                           : null, | ||||
|                   onRefresh: () { | ||||
|                     activitiesNotifier.forceRefresh(); | ||||
|                   }, | ||||
| @@ -348,21 +437,10 @@ class _ActivityListView extends HookConsumerWidget { | ||||
|                     ); | ||||
|                   }, | ||||
|                 ); | ||||
|                 if (isReply) { | ||||
|                   itemWidget = Column( | ||||
|                     crossAxisAlignment: CrossAxisAlignment.stretch, | ||||
|                     children: [ | ||||
|                       Row( | ||||
|                         children: [ | ||||
|                           const Icon(Symbols.reply), | ||||
|                           const Gap(8), | ||||
|                           Text('Replying your post'), | ||||
|                         ], | ||||
|                       ).padding(horizontal: 20, vertical: 8), | ||||
|                       itemWidget, | ||||
|                     ], | ||||
|                   ); | ||||
|                 } | ||||
|                 itemWidget = Card( | ||||
|                   margin: EdgeInsets.symmetric(horizontal: 8, vertical: 4), | ||||
|                   child: itemWidget, | ||||
|                 ); | ||||
|                 break; | ||||
|               case 'discovery': | ||||
|                 itemWidget = _DiscoveryActivityItem(data: item.data!); | ||||
| @@ -371,7 +449,7 @@ class _ActivityListView extends HookConsumerWidget { | ||||
|                 itemWidget = const Placeholder(); | ||||
|             } | ||||
|  | ||||
|             return Column(children: [itemWidget, const Divider(height: 1)]); | ||||
|             return itemWidget; | ||||
|           }, | ||||
|         ), | ||||
|         SliverGap(getTabbedPadding(context).bottom), | ||||
|   | ||||
							
								
								
									
										1094
									
								
								lib/screens/poll/poll_editor.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1094
									
								
								lib/screens/poll/poll_editor.dart
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -18,7 +18,8 @@ import 'package:island/widgets/post/publishers_modal.dart'; | ||||
| import 'package:island/screens/posts/post_detail.dart'; | ||||
| import 'package:island/widgets/post/compose_settings_sheet.dart'; | ||||
| import 'package:island/services/compose_storage_db.dart'; | ||||
| import 'package:island/widgets/post/draft_manager.dart'; | ||||
| // DraftManagerSheet is now imported through compose_toolbar.dart | ||||
| import 'package:island/widgets/post/compose_toolbar.dart'; | ||||
| import 'package:material_symbols_icons/symbols.dart'; | ||||
| import 'package:styled_widget/styled_widget.dart'; | ||||
|  | ||||
| @@ -52,13 +53,13 @@ class PostEditScreen extends HookConsumerWidget { | ||||
|       data: (post) => PostComposeScreen(originalPost: post), | ||||
|       loading: | ||||
|           () => AppScaffold( | ||||
|             noBackground: false, | ||||
|             isNoBackground: false, | ||||
|             appBar: AppBar(leading: const PageBackButton()), | ||||
|             body: const Center(child: CircularProgressIndicator()), | ||||
|           ), | ||||
|       error: | ||||
|           (e, _) => AppScaffold( | ||||
|             noBackground: false, | ||||
|             isNoBackground: false, | ||||
|             appBar: AppBar(leading: const PageBackButton()), | ||||
|             body: Text('Error: $e', textAlign: TextAlign.center), | ||||
|           ), | ||||
| @@ -92,7 +93,6 @@ class PostComposeScreen extends HookConsumerWidget { | ||||
|  | ||||
|     // Otherwise, continue with regular post compose | ||||
|     final theme = Theme.of(context); | ||||
|     final colorScheme = theme.colorScheme; | ||||
|  | ||||
|     // When editing, preserve the original replied/forwarded post references | ||||
|     final effectiveRepliedPost = repliedPost ?? originalPost?.repliedPost; | ||||
| @@ -207,8 +207,6 @@ class PostComposeScreen extends HookConsumerWidget { | ||||
|         isScrollControlled: true, | ||||
|         builder: | ||||
|             (context) => ComposeSettingsSheet( | ||||
|               titleController: state.titleController, | ||||
|               descriptionController: state.descriptionController, | ||||
|               visibility: state.visibility, | ||||
|               tagsController: state.tagsController, | ||||
|               categoriesController: state.categoriesController, | ||||
| @@ -238,6 +236,8 @@ class PostComposeScreen extends HookConsumerWidget { | ||||
|             onRequestUpload: | ||||
|                 () => ComposeLogic.uploadAttachment(ref, state, idx), | ||||
|             onDelete: () => ComposeLogic.deleteAttachment(ref, state, idx), | ||||
|             onUpdate: | ||||
|                 (value) => ComposeLogic.updateAttachment(state, value, idx), | ||||
|             onMove: (delta) { | ||||
|               state.attachments.value = ComposeLogic.moveAttachment( | ||||
|                 state.attachments.value, | ||||
| @@ -265,6 +265,9 @@ class PostComposeScreen extends HookConsumerWidget { | ||||
|                       () => ComposeLogic.uploadAttachment(ref, state, idx), | ||||
|                   onDelete: | ||||
|                       () => ComposeLogic.deleteAttachment(ref, state, idx), | ||||
|                   onUpdate: | ||||
|                       (value) => | ||||
|                           ComposeLogic.updateAttachment(state, value, idx), | ||||
|                   onMove: (delta) { | ||||
|                     state.attachments.value = ComposeLogic.moveAttachment( | ||||
|                       state.attachments.value, | ||||
| @@ -287,43 +290,10 @@ class PostComposeScreen extends HookConsumerWidget { | ||||
|         } | ||||
|       }, | ||||
|       child: AppScaffold( | ||||
|         noBackground: false, | ||||
|         isNoBackground: false, | ||||
|         appBar: AppBar( | ||||
|           leading: const PageBackButton(), | ||||
|           actions: [ | ||||
|             if (originalPost == null) // Only show drafts for new posts | ||||
|               IconButton( | ||||
|                 icon: const Icon(Symbols.draft), | ||||
|                 onPressed: () { | ||||
|                   showModalBottomSheet( | ||||
|                     context: context, | ||||
|                     isScrollControlled: true, | ||||
|                     builder: | ||||
|                         (context) => DraftManagerSheet( | ||||
|                           onDraftSelected: (draftId) { | ||||
|                             final draft = | ||||
|                                 ref.read( | ||||
|                                   composeStorageNotifierProvider, | ||||
|                                 )[draftId]; | ||||
|                             if (draft != null) { | ||||
|                               state.titleController.text = draft.title ?? ''; | ||||
|                               state.descriptionController.text = | ||||
|                                   draft.description ?? ''; | ||||
|                               state.contentController.text = | ||||
|                                   draft.content ?? ''; | ||||
|                               state.visibility.value = draft.visibility; | ||||
|                             } | ||||
|                           }, | ||||
|                         ), | ||||
|                   ); | ||||
|                 }, | ||||
|                 tooltip: 'drafts'.tr(), | ||||
|               ), | ||||
|             IconButton( | ||||
|               icon: const Icon(Symbols.save), | ||||
|               onPressed: () => ComposeLogic.saveDraft(ref, state), | ||||
|               tooltip: 'saveDraft'.tr(), | ||||
|             ), | ||||
|             IconButton( | ||||
|               icon: const Icon(Symbols.settings), | ||||
|               onPressed: showSettingsSheet, | ||||
| @@ -398,14 +368,52 @@ class PostComposeScreen extends HookConsumerWidget { | ||||
|                     // Post content form | ||||
|                     Expanded( | ||||
|                       child: SingleChildScrollView( | ||||
|                         padding: const EdgeInsets.symmetric(vertical: 12), | ||||
|                         padding: const EdgeInsets.symmetric(vertical: 16), | ||||
|                         child: Column( | ||||
|                           crossAxisAlignment: CrossAxisAlignment.start, | ||||
|                           children: [ | ||||
|                             TextField( | ||||
|                               controller: state.titleController, | ||||
|                               decoration: InputDecoration( | ||||
|                                 hintText: 'postTitle'.tr(), | ||||
|                                 border: InputBorder.none, | ||||
|                                 isCollapsed: true, | ||||
|                                 contentPadding: const EdgeInsets.symmetric( | ||||
|                                   vertical: 8, | ||||
|                                   horizontal: 8, | ||||
|                                 ), | ||||
|                               ), | ||||
|                               style: theme.textTheme.titleMedium, | ||||
|                               onTapOutside: | ||||
|                                   (_) => | ||||
|                                       FocusManager.instance.primaryFocus | ||||
|                                           ?.unfocus(), | ||||
|                             ), | ||||
|                             TextField( | ||||
|                               controller: state.descriptionController, | ||||
|                               decoration: InputDecoration( | ||||
|                                 hintText: 'postDescription'.tr(), | ||||
|                                 border: InputBorder.none, | ||||
|                                 isCollapsed: true, | ||||
|                                 contentPadding: const EdgeInsets.fromLTRB( | ||||
|                                   8, | ||||
|                                   4, | ||||
|                                   8, | ||||
|                                   12, | ||||
|                                 ), | ||||
|                               ), | ||||
|                               style: theme.textTheme.bodyMedium, | ||||
|                               minLines: 1, | ||||
|                               maxLines: 3, | ||||
|                               onTapOutside: | ||||
|                                   (_) => | ||||
|                                       FocusManager.instance.primaryFocus | ||||
|                                           ?.unfocus(), | ||||
|                             ), | ||||
|                             // Content field with borderless design | ||||
|                             RawKeyboardListener( | ||||
|                             KeyboardListener( | ||||
|                               focusNode: FocusNode(), | ||||
|                               onKey: | ||||
|                               onKeyEvent: | ||||
|                                   (event) => ComposeLogic.handleKeyPress( | ||||
|                                     event, | ||||
|                                     state, | ||||
| @@ -421,7 +429,11 @@ class PostComposeScreen extends HookConsumerWidget { | ||||
|                                 decoration: InputDecoration( | ||||
|                                   border: InputBorder.none, | ||||
|                                   hintText: 'postContent'.tr(), | ||||
|                                   contentPadding: const EdgeInsets.all(8), | ||||
|                                   isCollapsed: true, | ||||
|                                   contentPadding: const EdgeInsets.symmetric( | ||||
|                                     vertical: 8, | ||||
|                                     horizontal: 8, | ||||
|                                   ), | ||||
|                                 ), | ||||
|                                 maxLines: null, | ||||
|                                 onTapOutside: | ||||
| @@ -455,27 +467,7 @@ class PostComposeScreen extends HookConsumerWidget { | ||||
|             ), | ||||
|  | ||||
|             // Bottom toolbar | ||||
|             Material( | ||||
|               elevation: 4, | ||||
|               child: Row( | ||||
|                 children: [ | ||||
|                   IconButton( | ||||
|                     onPressed: () => ComposeLogic.pickPhotoMedia(ref, state), | ||||
|                     icon: const Icon(Symbols.add_a_photo), | ||||
|                     color: colorScheme.primary, | ||||
|                   ), | ||||
|                   IconButton( | ||||
|                     onPressed: () => ComposeLogic.pickVideoMedia(ref, state), | ||||
|                     icon: const Icon(Symbols.videocam), | ||||
|                     color: colorScheme.primary, | ||||
|                   ), | ||||
|                 ], | ||||
|               ).padding( | ||||
|                 bottom: MediaQuery.of(context).padding.bottom + 16, | ||||
|                 horizontal: 16, | ||||
|                 top: 8, | ||||
|               ), | ||||
|             ), | ||||
|             ComposeToolbar(state: state, originalPost: originalPost), | ||||
|           ], | ||||
|         ), | ||||
|       ), | ||||
| @@ -650,7 +642,7 @@ class PostComposeScreen extends HookConsumerWidget { | ||||
|                             child: SingleChildScrollView( | ||||
|                               controller: scrollController, | ||||
|                               padding: const EdgeInsets.all(16), | ||||
|                               child: PostItem(item: post, isOpenable: false), | ||||
|                               child: PostItem(item: post), | ||||
|                             ), | ||||
|                           ), | ||||
|                         ], | ||||
|   | ||||
| @@ -19,8 +19,8 @@ import 'package:island/widgets/content/markdown.dart'; | ||||
| import 'package:island/widgets/post/compose_shared.dart'; | ||||
| import 'package:island/widgets/post/compose_settings_sheet.dart'; | ||||
| import 'package:island/services/compose_storage_db.dart'; | ||||
| import 'package:island/widgets/post/compose_toolbar.dart'; | ||||
| import 'package:island/widgets/post/publishers_modal.dart'; | ||||
| import 'package:island/widgets/post/draft_manager.dart'; | ||||
|  | ||||
| import 'package:material_symbols_icons/symbols.dart'; | ||||
| import 'package:styled_widget/styled_widget.dart'; | ||||
| @@ -140,8 +140,6 @@ class ArticleComposeScreen extends HookConsumerWidget { | ||||
|         isScrollControlled: true, | ||||
|         builder: | ||||
|             (context) => ComposeSettingsSheet( | ||||
|               titleController: state.titleController, | ||||
|               descriptionController: state.descriptionController, | ||||
|               visibility: state.visibility, | ||||
|               tagsController: state.tagsController, | ||||
|               categoriesController: state.categoriesController, | ||||
| @@ -153,6 +151,57 @@ class ArticleComposeScreen extends HookConsumerWidget { | ||||
|     } | ||||
|  | ||||
|     Widget buildPreviewPane() { | ||||
|       final widgetItem = SingleChildScrollView( | ||||
|         padding: const EdgeInsets.symmetric(vertical: 14, horizontal: 8), | ||||
|         child: ValueListenableBuilder<TextEditingValue>( | ||||
|           valueListenable: state.titleController, | ||||
|           builder: (context, titleValue, _) { | ||||
|             return ValueListenableBuilder<TextEditingValue>( | ||||
|               valueListenable: state.descriptionController, | ||||
|               builder: (context, descriptionValue, _) { | ||||
|                 return ValueListenableBuilder<TextEditingValue>( | ||||
|                   valueListenable: state.contentController, | ||||
|                   builder: (context, contentValue, _) { | ||||
|                     return Column( | ||||
|                       crossAxisAlignment: CrossAxisAlignment.start, | ||||
|                       children: [ | ||||
|                         if (titleValue.text.isNotEmpty) ...[ | ||||
|                           Text( | ||||
|                             titleValue.text, | ||||
|                             style: theme.textTheme.headlineSmall?.copyWith( | ||||
|                               fontWeight: FontWeight.bold, | ||||
|                             ), | ||||
|                           ), | ||||
|                           const Gap(16), | ||||
|                         ], | ||||
|                         if (descriptionValue.text.isNotEmpty) ...[ | ||||
|                           Text( | ||||
|                             descriptionValue.text, | ||||
|                             style: theme.textTheme.bodyLarge?.copyWith( | ||||
|                               color: colorScheme.onSurface.withOpacity(0.7), | ||||
|                             ), | ||||
|                           ), | ||||
|                           const Gap(16), | ||||
|                         ], | ||||
|                         if (contentValue.text.isNotEmpty) | ||||
|                           MarkdownTextContent( | ||||
|                             content: contentValue.text, | ||||
|                             textStyle: theme.textTheme.bodyMedium, | ||||
|                           ), | ||||
|                       ], | ||||
|                     ); | ||||
|                   }, | ||||
|                 ); | ||||
|               }, | ||||
|             ); | ||||
|           }, | ||||
|         ), | ||||
|       ); | ||||
|  | ||||
|       if (isWideScreen(context)) { | ||||
|         return Align(alignment: Alignment.topLeft, child: widgetItem); | ||||
|       } | ||||
|  | ||||
|       return Container( | ||||
|         decoration: BoxDecoration( | ||||
|           border: Border.all(color: colorScheme.outline.withOpacity(0.3)), | ||||
| @@ -178,210 +227,155 @@ class ArticleComposeScreen extends HookConsumerWidget { | ||||
|                 ], | ||||
|               ), | ||||
|             ), | ||||
|             Expanded( | ||||
|               child: SingleChildScrollView( | ||||
|                 padding: const EdgeInsets.all(16), | ||||
|                 child: ValueListenableBuilder<TextEditingValue>( | ||||
|                   valueListenable: state.titleController, | ||||
|                   builder: (context, titleValue, _) { | ||||
|                     return ValueListenableBuilder<TextEditingValue>( | ||||
|                       valueListenable: state.descriptionController, | ||||
|                       builder: (context, descriptionValue, _) { | ||||
|                         return ValueListenableBuilder<TextEditingValue>( | ||||
|                           valueListenable: state.contentController, | ||||
|                           builder: (context, contentValue, _) { | ||||
|                             return Column( | ||||
|                               crossAxisAlignment: CrossAxisAlignment.start, | ||||
|                               children: [ | ||||
|                                 if (titleValue.text.isNotEmpty) ...[ | ||||
|                                   Text( | ||||
|                                     titleValue.text, | ||||
|                                     style: theme.textTheme.headlineSmall | ||||
|                                         ?.copyWith(fontWeight: FontWeight.bold), | ||||
|                                   ), | ||||
|                                   const Gap(16), | ||||
|                                 ], | ||||
|                                 if (descriptionValue.text.isNotEmpty) ...[ | ||||
|                                   Text( | ||||
|                                     descriptionValue.text, | ||||
|                                     style: theme.textTheme.bodyLarge?.copyWith( | ||||
|                                       color: colorScheme.onSurface.withOpacity( | ||||
|                                         0.7, | ||||
|                                       ), | ||||
|                                     ), | ||||
|                                   ), | ||||
|                                   const Gap(16), | ||||
|                                 ], | ||||
|                                 if (contentValue.text.isNotEmpty) | ||||
|                                   MarkdownTextContent( | ||||
|                                     content: contentValue.text, | ||||
|                                     textStyle: theme.textTheme.bodyMedium, | ||||
|                                   ), | ||||
|                               ], | ||||
|                             ); | ||||
|                           }, | ||||
|                         ); | ||||
|                       }, | ||||
|                     ); | ||||
|                   }, | ||||
|                 ), | ||||
|               ), | ||||
|             ), | ||||
|             Expanded(child: widgetItem), | ||||
|           ], | ||||
|         ), | ||||
|       ); | ||||
|     } | ||||
|  | ||||
|     Widget buildEditorPane() { | ||||
|       return Column( | ||||
|         crossAxisAlignment: CrossAxisAlignment.start, | ||||
|         children: [ | ||||
|           // Publisher row | ||||
|           Card( | ||||
|             margin: EdgeInsets.only(top: 8), | ||||
|             elevation: 1, | ||||
|             child: Padding( | ||||
|               padding: const EdgeInsets.all(12), | ||||
|               child: Row( | ||||
|                 children: [ | ||||
|                   GestureDetector( | ||||
|                     child: ProfilePictureWidget( | ||||
|                       fileId: state.currentPublisher.value?.picture?.id, | ||||
|                       radius: 20, | ||||
|                       fallbackIcon: | ||||
|                           state.currentPublisher.value == null | ||||
|                               ? Symbols.question_mark | ||||
|                               : null, | ||||
|                     ), | ||||
|                     onTap: () { | ||||
|                       showModalBottomSheet( | ||||
|                         isScrollControlled: true, | ||||
|                         context: context, | ||||
|                         builder: (context) => const PublisherModal(), | ||||
|                       ).then((value) { | ||||
|                         if (value != null) { | ||||
|                           state.currentPublisher.value = value; | ||||
|                         } | ||||
|                       }); | ||||
|                     }, | ||||
|                   ), | ||||
|                   const Gap(16), | ||||
|                   if (state.currentPublisher.value == null) | ||||
|                     Text( | ||||
|                       'postPublisherUnselected'.tr(), | ||||
|                       style: theme.textTheme.bodyMedium, | ||||
|                     ) | ||||
|                   else | ||||
|                     Column( | ||||
|                       crossAxisAlignment: CrossAxisAlignment.start, | ||||
|                       children: [ | ||||
|                         Text(state.currentPublisher.value!.nick).bold(), | ||||
|                         Text( | ||||
|                           '@${state.currentPublisher.value!.name}', | ||||
|                         ).fontSize(12), | ||||
|                       ], | ||||
|                     ), | ||||
|                 ], | ||||
|               ), | ||||
|             ), | ||||
|           ), | ||||
|  | ||||
|           // Content field with keyboard listener | ||||
|           Expanded( | ||||
|             child: RawKeyboardListener( | ||||
|               focusNode: FocusNode(), | ||||
|               onKey: | ||||
|                   (event) => _handleKeyPress( | ||||
|                     event, | ||||
|                     state, | ||||
|                     ref, | ||||
|                     context, | ||||
|                     originalPost: originalPost, | ||||
|                   ), | ||||
|               child: TextField( | ||||
|                 controller: state.contentController, | ||||
|                 style: theme.textTheme.bodyMedium, | ||||
|       return Center( | ||||
|         child: ConstrainedBox( | ||||
|           constraints: const BoxConstraints(maxWidth: 560), | ||||
|           child: Column( | ||||
|             crossAxisAlignment: CrossAxisAlignment.start, | ||||
|             children: [ | ||||
|               TextField( | ||||
|                 controller: state.titleController, | ||||
|                 decoration: InputDecoration( | ||||
|                   hintText: 'postTitle'.tr(), | ||||
|                   border: InputBorder.none, | ||||
|                   hintText: 'postContent'.tr(), | ||||
|                   contentPadding: const EdgeInsets.all(8), | ||||
|                   isCollapsed: true, | ||||
|                   contentPadding: const EdgeInsets.symmetric( | ||||
|                     vertical: 8, | ||||
|                     horizontal: 8, | ||||
|                   ), | ||||
|                 ), | ||||
|                 maxLines: null, | ||||
|                 expands: true, | ||||
|                 textAlignVertical: TextAlignVertical.top, | ||||
|                 style: theme.textTheme.titleMedium, | ||||
|                 onTapOutside: | ||||
|                     (_) => FocusManager.instance.primaryFocus?.unfocus(), | ||||
|               ), | ||||
|             ), | ||||
|           ), | ||||
|  | ||||
|           // Attachments preview | ||||
|           ValueListenableBuilder<List<UniversalFile>>( | ||||
|             valueListenable: state.attachments, | ||||
|             builder: (context, attachments, _) { | ||||
|               if (attachments.isEmpty) return const SizedBox.shrink(); | ||||
|               return Column( | ||||
|                 crossAxisAlignment: CrossAxisAlignment.start, | ||||
|                 children: [ | ||||
|                   const Gap(16), | ||||
|                   Text( | ||||
|                     'articleAttachmentHint'.tr(), | ||||
|                     style: Theme.of(context).textTheme.bodySmall?.copyWith( | ||||
|                       color: Theme.of(context).colorScheme.onSurfaceVariant, | ||||
|               TextField( | ||||
|                 controller: state.descriptionController, | ||||
|                 decoration: InputDecoration( | ||||
|                   hintText: 'postDescription'.tr(), | ||||
|                   border: InputBorder.none, | ||||
|                   isCollapsed: true, | ||||
|                   contentPadding: const EdgeInsets.fromLTRB(8, 4, 8, 12), | ||||
|                 ), | ||||
|                 style: theme.textTheme.bodyMedium, | ||||
|                 minLines: 1, | ||||
|                 maxLines: 3, | ||||
|                 onTapOutside: | ||||
|                     (_) => FocusManager.instance.primaryFocus?.unfocus(), | ||||
|               ), | ||||
|               Expanded( | ||||
|                 child: KeyboardListener( | ||||
|                   focusNode: FocusNode(), | ||||
|                   onKeyEvent: | ||||
|                       (event) => _handleKeyPress( | ||||
|                         event, | ||||
|                         state, | ||||
|                         ref, | ||||
|                         context, | ||||
|                         originalPost: originalPost, | ||||
|                       ), | ||||
|                   child: TextField( | ||||
|                     controller: state.contentController, | ||||
|                     style: theme.textTheme.bodyMedium, | ||||
|                     decoration: InputDecoration( | ||||
|                       border: InputBorder.none, | ||||
|                       hintText: 'postContent'.tr(), | ||||
|                       contentPadding: const EdgeInsets.symmetric( | ||||
|                         vertical: 16, | ||||
|                         horizontal: 8, | ||||
|                       ), | ||||
|                     ), | ||||
|                   ).padding(bottom: 8), | ||||
|                   ValueListenableBuilder<Map<int, double>>( | ||||
|                     valueListenable: state.attachmentProgress, | ||||
|                     builder: (context, progressMap, _) { | ||||
|                       return Wrap( | ||||
|                         spacing: 8, | ||||
|                         runSpacing: 8, | ||||
|                         children: [ | ||||
|                           for (var idx = 0; idx < attachments.length; idx++) | ||||
|                             SizedBox( | ||||
|                               width: 280, | ||||
|                               height: 280, | ||||
|                               child: AttachmentPreview( | ||||
|                                 item: attachments[idx], | ||||
|                                 progress: progressMap[idx], | ||||
|                                 onRequestUpload: | ||||
|                                     () => ComposeLogic.uploadAttachment( | ||||
|                                       ref, | ||||
|                                       state, | ||||
|                                       idx, | ||||
|                                     ), | ||||
|                                 onDelete: | ||||
|                                     () => ComposeLogic.deleteAttachment( | ||||
|                                       ref, | ||||
|                                       state, | ||||
|                                       idx, | ||||
|                                     ), | ||||
|                                 onMove: (delta) { | ||||
|                                   state | ||||
|                                       .attachments | ||||
|                                       .value = ComposeLogic.moveAttachment( | ||||
|                                     state.attachments.value, | ||||
|                                     idx, | ||||
|                                     delta, | ||||
|                                   ); | ||||
|                                 }, | ||||
|                                 onInsert: | ||||
|                                     () => ComposeLogic.insertAttachment( | ||||
|                                       ref, | ||||
|                                       state, | ||||
|                                       idx, | ||||
|                                     ), | ||||
|                               ), | ||||
|                             ), | ||||
|                         ], | ||||
|                       ); | ||||
|                     }, | ||||
|                     maxLines: null, | ||||
|                     expands: true, | ||||
|                     textAlignVertical: TextAlignVertical.top, | ||||
|                     onTapOutside: | ||||
|                         (_) => FocusManager.instance.primaryFocus?.unfocus(), | ||||
|                   ), | ||||
|                 ], | ||||
|               ); | ||||
|             }, | ||||
|                 ), | ||||
|               ), | ||||
|  | ||||
|               // Attachments preview | ||||
|               ValueListenableBuilder<List<UniversalFile>>( | ||||
|                 valueListenable: state.attachments, | ||||
|                 builder: (context, attachments, _) { | ||||
|                   if (attachments.isEmpty) return const SizedBox.shrink(); | ||||
|                   return Column( | ||||
|                     crossAxisAlignment: CrossAxisAlignment.start, | ||||
|                     children: [ | ||||
|                       const Gap(16), | ||||
|                       Text( | ||||
|                         'articleAttachmentHint'.tr(), | ||||
|                         style: Theme.of(context).textTheme.bodySmall?.copyWith( | ||||
|                           color: Theme.of(context).colorScheme.onSurfaceVariant, | ||||
|                         ), | ||||
|                       ).padding(bottom: 8), | ||||
|                       ValueListenableBuilder<Map<int, double>>( | ||||
|                         valueListenable: state.attachmentProgress, | ||||
|                         builder: (context, progressMap, _) { | ||||
|                           return Wrap( | ||||
|                             spacing: 8, | ||||
|                             runSpacing: 8, | ||||
|                             children: [ | ||||
|                               for (var idx = 0; idx < attachments.length; idx++) | ||||
|                                 SizedBox( | ||||
|                                   width: 280, | ||||
|                                   height: 280, | ||||
|                                   child: AttachmentPreview( | ||||
|                                     item: attachments[idx], | ||||
|                                     progress: progressMap[idx], | ||||
|                                     onRequestUpload: | ||||
|                                         () => ComposeLogic.uploadAttachment( | ||||
|                                           ref, | ||||
|                                           state, | ||||
|                                           idx, | ||||
|                                         ), | ||||
|                                     onUpdate: | ||||
|                                         (value) => | ||||
|                                             ComposeLogic.updateAttachment( | ||||
|                                               state, | ||||
|                                               value, | ||||
|                                               idx, | ||||
|                                             ), | ||||
|                                     onDelete: | ||||
|                                         () => ComposeLogic.deleteAttachment( | ||||
|                                           ref, | ||||
|                                           state, | ||||
|                                           idx, | ||||
|                                         ), | ||||
|                                     onMove: (delta) { | ||||
|                                       state | ||||
|                                           .attachments | ||||
|                                           .value = ComposeLogic.moveAttachment( | ||||
|                                         state.attachments.value, | ||||
|                                         idx, | ||||
|                                         delta, | ||||
|                                       ); | ||||
|                                     }, | ||||
|                                     onInsert: | ||||
|                                         () => ComposeLogic.insertAttachment( | ||||
|                                           ref, | ||||
|                                           state, | ||||
|                                           idx, | ||||
|                                         ), | ||||
|                                   ), | ||||
|                                 ), | ||||
|                             ], | ||||
|                           ); | ||||
|                         }, | ||||
|                       ), | ||||
|                     ], | ||||
|                   ); | ||||
|                 }, | ||||
|               ), | ||||
|             ], | ||||
|           ), | ||||
|         ], | ||||
|         ), | ||||
|       ); | ||||
|     } | ||||
|  | ||||
| @@ -392,7 +386,7 @@ class ArticleComposeScreen extends HookConsumerWidget { | ||||
|         } | ||||
|       }, | ||||
|       child: AppScaffold( | ||||
|         noBackground: false, | ||||
|         isNoBackground: false, | ||||
|         appBar: AppBar( | ||||
|           leading: const PageBackButton(), | ||||
|           title: ValueListenableBuilder<TextEditingValue>( | ||||
| @@ -406,38 +400,26 @@ class ArticleComposeScreen extends HookConsumerWidget { | ||||
|           actions: [ | ||||
|             // Info banner for article compose | ||||
|             const SizedBox.shrink(), | ||||
|             if (originalPost == null) // Only show drafts for new articles | ||||
|               IconButton( | ||||
|                 icon: const Icon(Symbols.draft), | ||||
|                 onPressed: () { | ||||
|                   showModalBottomSheet( | ||||
|                     context: context, | ||||
|                     isScrollControlled: true, | ||||
|                     builder: | ||||
|                         (context) => DraftManagerSheet( | ||||
|                           onDraftSelected: (draftId) { | ||||
|                             final draft = | ||||
|                                 ref.read( | ||||
|                                   composeStorageNotifierProvider, | ||||
|                                 )[draftId]; | ||||
|                             if (draft != null) { | ||||
|                               state.titleController.text = draft.title ?? ''; | ||||
|                               state.descriptionController.text = | ||||
|                                   draft.description ?? ''; | ||||
|                               state.contentController.text = | ||||
|                                   draft.content ?? ''; | ||||
|                               state.visibility.value = draft.visibility; | ||||
|                             } | ||||
|                           }, | ||||
|                         ), | ||||
|                   ); | ||||
|                 }, | ||||
|                 tooltip: 'drafts'.tr(), | ||||
|               ), | ||||
|             IconButton( | ||||
|               icon: const Icon(Symbols.save), | ||||
|               onPressed: () => ComposeLogic.saveDraft(ref, state), | ||||
|               tooltip: 'saveDraft'.tr(), | ||||
|               icon: ProfilePictureWidget( | ||||
|                 fileId: state.currentPublisher.value?.picture?.id, | ||||
|                 radius: 12, | ||||
|                 fallbackIcon: | ||||
|                     state.currentPublisher.value == null | ||||
|                         ? Symbols.question_mark | ||||
|                         : null, | ||||
|               ), | ||||
|               onPressed: () { | ||||
|                 showModalBottomSheet( | ||||
|                   isScrollControlled: true, | ||||
|                   context: context, | ||||
|                   builder: (context) => const PublisherModal(), | ||||
|                 ).then((value) { | ||||
|                   if (value != null) { | ||||
|                     state.currentPublisher.value = value; | ||||
|                   } | ||||
|                 }); | ||||
|               }, | ||||
|             ), | ||||
|             IconButton( | ||||
|               icon: const Icon(Symbols.settings), | ||||
| @@ -499,6 +481,7 @@ class ArticleComposeScreen extends HookConsumerWidget { | ||||
|                               flex: showPreview.value ? 1 : 2, | ||||
|                               child: buildEditorPane(), | ||||
|                             ), | ||||
|                             if (showPreview.value) const VerticalDivider(), | ||||
|                             if (showPreview.value) | ||||
|                               Expanded(child: buildPreviewPane()), | ||||
|                           ], | ||||
| @@ -510,27 +493,7 @@ class ArticleComposeScreen extends HookConsumerWidget { | ||||
|             ), | ||||
|  | ||||
|             // Bottom toolbar | ||||
|             Material( | ||||
|               elevation: 4, | ||||
|               child: Row( | ||||
|                 children: [ | ||||
|                   IconButton( | ||||
|                     onPressed: () => ComposeLogic.pickPhotoMedia(ref, state), | ||||
|                     icon: const Icon(Symbols.add_a_photo), | ||||
|                     color: colorScheme.primary, | ||||
|                   ), | ||||
|                   IconButton( | ||||
|                     onPressed: () => ComposeLogic.pickVideoMedia(ref, state), | ||||
|                     icon: const Icon(Symbols.videocam), | ||||
|                     color: colorScheme.primary, | ||||
|                   ), | ||||
|                 ], | ||||
|               ).padding( | ||||
|                 bottom: MediaQuery.of(context).padding.bottom + 16, | ||||
|                 horizontal: 16, | ||||
|                 top: 8, | ||||
|               ), | ||||
|             ), | ||||
|             ComposeToolbar(state: state, originalPost: originalPost), | ||||
|           ], | ||||
|         ), | ||||
|       ), | ||||
| @@ -539,7 +502,7 @@ class ArticleComposeScreen extends HookConsumerWidget { | ||||
|  | ||||
|   // Helper method to handle keyboard shortcuts | ||||
|   void _handleKeyPress( | ||||
|     RawKeyEvent event, | ||||
|     KeyEvent event, | ||||
|     ComposeState state, | ||||
|     WidgetRef ref, | ||||
|     BuildContext context, { | ||||
| @@ -549,7 +512,9 @@ class ArticleComposeScreen extends HookConsumerWidget { | ||||
|  | ||||
|     final isPaste = event.logicalKey == LogicalKeyboardKey.keyV; | ||||
|     final isSave = event.logicalKey == LogicalKeyboardKey.keyS; | ||||
|     final isModifierPressed = event.isMetaPressed || event.isControlPressed; | ||||
|     final isModifierPressed = | ||||
|         HardwareKeyboard.instance.isMetaPressed || | ||||
|         HardwareKeyboard.instance.isControlPressed; | ||||
|     final isSubmit = event.logicalKey == LogicalKeyboardKey.enter; | ||||
|  | ||||
|     if (isPaste && isModifierPressed) { | ||||
|   | ||||
| @@ -4,7 +4,6 @@ import 'package:hooks_riverpod/hooks_riverpod.dart'; | ||||
| import 'package:island/models/post.dart'; | ||||
| import 'package:island/pods/network.dart'; | ||||
| import 'package:island/pods/userinfo.dart'; | ||||
| import 'package:island/services/responsive.dart'; | ||||
| import 'package:island/widgets/app_scaffold.dart'; | ||||
| import 'package:island/widgets/post/post_item.dart'; | ||||
| import 'package:island/widgets/post/post_quick_reply.dart'; | ||||
| @@ -54,9 +53,8 @@ class PostDetailScreen extends HookConsumerWidget { | ||||
|     final postState = ref.watch(postStateProvider(id)); | ||||
|     final user = ref.watch(userInfoProvider); | ||||
|  | ||||
|     final isWide = isWideScreen(context); | ||||
|  | ||||
|     return AppScaffold( | ||||
|       isNoBackground: false, | ||||
|       appBar: AppBar(title: const Text('Post')), | ||||
|       body: postState.when( | ||||
|         data: (post) { | ||||
| @@ -66,13 +64,13 @@ class PostDetailScreen extends HookConsumerWidget { | ||||
|               CustomScrollView( | ||||
|                 slivers: [ | ||||
|                   SliverToBoxAdapter( | ||||
|                     child: Column( | ||||
|                       children: [ | ||||
|                         PostItem( | ||||
|                     child: Center( | ||||
|                       child: ConstrainedBox( | ||||
|                         constraints: BoxConstraints(maxWidth: 600), | ||||
|                         child: PostItem( | ||||
|                           item: post!, | ||||
|                           isOpenable: false, | ||||
|                           isFullPost: true, | ||||
|                           backgroundColor: isWide ? Colors.transparent : null, | ||||
|                           isEmbedReply: false, | ||||
|                           onUpdate: (newItem) { | ||||
|                             // Update the local state with the new post data | ||||
|                             ref | ||||
| @@ -80,11 +78,10 @@ class PostDetailScreen extends HookConsumerWidget { | ||||
|                                 .updatePost(newItem); | ||||
|                           }, | ||||
|                         ), | ||||
|                         const Divider(height: 1), | ||||
|                       ], | ||||
|                       ), | ||||
|                     ), | ||||
|                   ), | ||||
|                   PostRepliesList(postId: id), | ||||
|                   PostRepliesList(postId: id, maxWidth: 600), | ||||
|                   SliverGap(MediaQuery.of(context).padding.bottom + 80), | ||||
|                 ], | ||||
|               ), | ||||
|   | ||||
| @@ -5,6 +5,7 @@ import 'package:island/models/post.dart'; | ||||
| import 'package:island/pods/network.dart'; | ||||
| import 'package:island/widgets/app_scaffold.dart'; | ||||
| import 'package:island/widgets/post/post_item.dart'; | ||||
| import 'package:island/widgets/response.dart'; | ||||
| import 'package:riverpod_paging_utils/riverpod_paging_utils.dart'; | ||||
|  | ||||
| final postSearchNotifierProvider = StateNotifierProvider.autoDispose< | ||||
| @@ -55,7 +56,7 @@ class PostSearchNotifier | ||||
|           'query': _currentQuery, | ||||
|           'offset': offset, | ||||
|           'take': _pageSize, | ||||
|           'useVector': true, | ||||
|           'useVector': false, | ||||
|         }, | ||||
|       ); | ||||
|  | ||||
| @@ -109,6 +110,7 @@ class _PostSearchScreenState extends ConsumerState<PostSearchScreen> { | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     return AppScaffold( | ||||
|       isNoBackground: false, | ||||
|       appBar: AppBar( | ||||
|         title: TextField( | ||||
|           controller: _searchController, | ||||
| @@ -140,6 +142,7 @@ class _PostSearchScreenState extends ConsumerState<PostSearchScreen> { | ||||
|               } | ||||
|  | ||||
|               return ListView.builder( | ||||
|                 padding: EdgeInsets.zero, | ||||
|                 itemCount: data.items.length + (data.hasMore ? 1 : 0), | ||||
|                 itemBuilder: (context, index) { | ||||
|                   if (index >= data.items.length) { | ||||
| @@ -150,14 +153,27 @@ class _PostSearchScreenState extends ConsumerState<PostSearchScreen> { | ||||
|                   } | ||||
|  | ||||
|                   final post = data.items[index]; | ||||
|                   return Column( | ||||
|                     children: [PostItem(item: post), const Divider(height: 1)], | ||||
|                   return Center( | ||||
|                     child: ConstrainedBox( | ||||
|                       constraints: BoxConstraints(maxWidth: 600), | ||||
|                       child: Card( | ||||
|                         margin: EdgeInsets.symmetric( | ||||
|                           horizontal: 8, | ||||
|                           vertical: 4, | ||||
|                         ), | ||||
|                         child: PostActionableItem(item: post, borderRadius: 8), | ||||
|                       ), | ||||
|                     ), | ||||
|                   ); | ||||
|                 }, | ||||
|               ); | ||||
|             }, | ||||
|             loading: () => const Center(child: CircularProgressIndicator()), | ||||
|             error: (error, stack) => Center(child: Text('Error: $error')), | ||||
|             error: | ||||
|                 (error, stack) => ResponseErrorWidget( | ||||
|                   error: error, | ||||
|                   onRetry: () => ref.invalidate(postSearchNotifierProvider), | ||||
|                 ), | ||||
|           ); | ||||
|         }, | ||||
|       ), | ||||
|   | ||||
| @@ -11,12 +11,14 @@ import 'package:island/models/user.dart'; | ||||
| import 'package:island/pods/config.dart'; | ||||
| import 'package:island/pods/network.dart'; | ||||
| import 'package:island/services/color.dart'; | ||||
| import 'package:island/services/responsive.dart'; | ||||
| import 'package:island/widgets/account/account_name.dart'; | ||||
| import 'package:island/widgets/account/badge.dart'; | ||||
| import 'package:island/widgets/account/status.dart'; | ||||
| import 'package:island/widgets/alert.dart'; | ||||
| import 'package:island/widgets/app_scaffold.dart'; | ||||
| import 'package:island/widgets/content/cloud_files.dart'; | ||||
| import 'package:island/widgets/content/markdown.dart'; | ||||
| import 'package:island/widgets/post/post_list.dart'; | ||||
| import 'package:material_symbols_icons/symbols.dart'; | ||||
| import 'package:palette_generator/palette_generator.dart'; | ||||
| @@ -85,6 +87,12 @@ class PublisherProfileScreen extends HookConsumerWidget { | ||||
|       publisherAppbarForcegroundColorProvider(name), | ||||
|     ); | ||||
|  | ||||
|     final categoryTabController = useTabController(initialLength: 3); | ||||
|     final categoryTab = useState(0); | ||||
|     categoryTabController.addListener(() { | ||||
|       categoryTab.value = categoryTabController.index; | ||||
|     }); | ||||
|  | ||||
|     final subscribing = useState(false); | ||||
|  | ||||
|     Future<void> subscribe() async { | ||||
| @@ -121,209 +129,314 @@ class PublisherProfileScreen extends HookConsumerWidget { | ||||
|       offset: Offset(1.0, 1.0), | ||||
|     ); | ||||
|  | ||||
|     Widget publisherBasisWidget(SnPublisher data) => Row( | ||||
|       crossAxisAlignment: CrossAxisAlignment.start, | ||||
|       spacing: 20, | ||||
|       children: [ | ||||
|         GestureDetector( | ||||
|           child: Badge( | ||||
|             isLabelVisible: data.type == 0, | ||||
|             padding: EdgeInsets.all(4), | ||||
|             label: Icon( | ||||
|               Symbols.launch, | ||||
|               size: 16, | ||||
|               color: Theme.of(context).colorScheme.onPrimary, | ||||
|             ), | ||||
|             backgroundColor: Theme.of(context).colorScheme.primary, | ||||
|             offset: Offset(0, 48), | ||||
|             child: ProfilePictureWidget(file: data.picture, radius: 32), | ||||
|           ), | ||||
|           onTap: () { | ||||
|             Navigator.pop(context, true); | ||||
|             if (data.account?.name != null) { | ||||
|               context.pushNamed( | ||||
|                 'accountProfile', | ||||
|                 pathParameters: {'name': data.account!.name}, | ||||
|               ); | ||||
|             } | ||||
|           }, | ||||
|         ), | ||||
|         Expanded( | ||||
|           child: Column( | ||||
|             crossAxisAlignment: CrossAxisAlignment.stretch, | ||||
|             children: [ | ||||
|               Row( | ||||
|                 spacing: 6, | ||||
|                 children: [ | ||||
|                   Text(data.nick).fontSize(20), | ||||
|                   if (data.verification != null) | ||||
|                     VerificationMark(mark: data.verification!), | ||||
|                   Expanded( | ||||
|                     child: Text( | ||||
|                       '@${data.name}', | ||||
|                       maxLines: 1, | ||||
|                       overflow: TextOverflow.ellipsis, | ||||
|                     ).fontSize(14).opacity(0.85), | ||||
|                   ), | ||||
|                 ], | ||||
|               ), | ||||
|               if (data.type == 0 && data.account != null) | ||||
|                 Row( | ||||
|                   crossAxisAlignment: CrossAxisAlignment.center, | ||||
|                   spacing: 6, | ||||
|                   children: [ | ||||
|                     Icon( | ||||
|                       data.type == 0 ? Symbols.person : Symbols.workspaces, | ||||
|                       fill: 1, | ||||
|                       size: 17, | ||||
|                     ), | ||||
|                     Text( | ||||
|                       'publisherBelongsTo'.tr(args: ['@${data.account!.name}']), | ||||
|                     ).fontSize(14), | ||||
|                   ], | ||||
|                 ).opacity(0.85).padding(bottom: 6), | ||||
|               if (data.type == 0 && data.account != null) | ||||
|                 AccountStatusWidget( | ||||
|                   uname: data.account!.name, | ||||
|                   padding: EdgeInsets.zero, | ||||
|                 ), | ||||
|               subStatus | ||||
|                   .when( | ||||
|                     data: | ||||
|                         (status) => FilledButton.icon( | ||||
|                           onPressed: | ||||
|                               subscribing.value | ||||
|                                   ? null | ||||
|                                   : (status.isSubscribed | ||||
|                                       ? unsubscribe | ||||
|                                       : subscribe), | ||||
|                           icon: Icon( | ||||
|                             status.isSubscribed | ||||
|                                 ? Symbols.remove_circle | ||||
|                                 : Symbols.add_circle, | ||||
|                           ), | ||||
|                           label: | ||||
|                               Text( | ||||
|                                 status.isSubscribed | ||||
|                                     ? 'unsubscribe' | ||||
|                                     : 'subscribe', | ||||
|                               ).tr(), | ||||
|                           style: ButtonStyle( | ||||
|                             visualDensity: VisualDensity(vertical: -2), | ||||
|                           ), | ||||
|                         ), | ||||
|                     error: (_, _) => const SizedBox(), | ||||
|                     loading: | ||||
|                         () => const SizedBox( | ||||
|                           height: 36, | ||||
|                           child: Center( | ||||
|                             child: SizedBox( | ||||
|                               width: 20, | ||||
|                               height: 20, | ||||
|                               child: CircularProgressIndicator(strokeWidth: 2), | ||||
|                             ), | ||||
|                           ), | ||||
|                         ), | ||||
|                   ) | ||||
|                   .padding(top: 8), | ||||
|             ], | ||||
|           ), | ||||
|         ), | ||||
|       ], | ||||
|     ).padding(horizontal: 24, top: 24); | ||||
|  | ||||
|     Widget publisherBadgesWidget(SnPublisher data) => | ||||
|         (badges.value?.isNotEmpty ?? false) | ||||
|             ? Card( | ||||
|               child: BadgeList( | ||||
|                 badges: badges.value!, | ||||
|               ).padding(horizontal: 26, vertical: 20), | ||||
|             ).padding(horizontal: 4) | ||||
|             : const SizedBox.shrink(); | ||||
|  | ||||
|     Widget publisherVerificationWidget(SnPublisher data) => | ||||
|         (data.verification != null) | ||||
|             ? Card( | ||||
|               margin: EdgeInsets.symmetric(horizontal: 8, vertical: 4), | ||||
|               child: VerificationStatusCard(mark: data.verification!), | ||||
|             ) | ||||
|             : const SizedBox.shrink(); | ||||
|  | ||||
|     Widget publisherBioWidget(SnPublisher data) => Card( | ||||
|       margin: EdgeInsets.symmetric(horizontal: 8, vertical: 4), | ||||
|       child: Column( | ||||
|         crossAxisAlignment: CrossAxisAlignment.stretch, | ||||
|         children: [ | ||||
|           Text('bio').tr().bold().fontSize(15).padding(bottom: 8), | ||||
|           if (data.bio.isEmpty) | ||||
|             Text('descriptionNone').tr().italic() | ||||
|           else | ||||
|             MarkdownTextContent( | ||||
|               content: data.bio, | ||||
|               linesMargin: EdgeInsets.zero, | ||||
|             ), | ||||
|         ], | ||||
|       ).padding(horizontal: 20, vertical: 16), | ||||
|     ); | ||||
|  | ||||
|     Widget publisherCategoryTabWidget() => Card( | ||||
|       margin: EdgeInsets.symmetric(horizontal: 8, vertical: 4), | ||||
|       child: TabBar( | ||||
|         controller: categoryTabController, | ||||
|         dividerColor: Colors.transparent, | ||||
|         splashBorderRadius: const BorderRadius.all(Radius.circular(8)), | ||||
|         tabs: [Tab(text: 'All'), Tab(text: 'Posts'), Tab(text: 'Articles')], | ||||
|       ), | ||||
|     ); | ||||
|  | ||||
|     return publisher.when( | ||||
|       data: | ||||
|           (data) => AppScaffold( | ||||
|             body: CustomScrollView( | ||||
|               slivers: [ | ||||
|                 SliverAppBar( | ||||
|                   foregroundColor: appbarColor.value, | ||||
|                   expandedHeight: 180, | ||||
|                   pinned: true, | ||||
|                   leading: PageBackButton( | ||||
|                     color: appbarColor.value, | ||||
|                     shadows: [appbarShadow], | ||||
|                   ), | ||||
|                   flexibleSpace: Stack( | ||||
|                     children: [ | ||||
|                       Positioned.fill( | ||||
|                         child: | ||||
|                             data.background?.id != null | ||||
|                                 ? CloudImageWidget(file: data.background) | ||||
|                                 : Container( | ||||
|                                   color: | ||||
|                                       Theme.of( | ||||
|                                         context, | ||||
|                                       ).appBarTheme.backgroundColor, | ||||
|                                 ), | ||||
|             isNoBackground: false, | ||||
|             appBar: | ||||
|                 isWideScreen(context) | ||||
|                     ? AppBar( | ||||
|                       foregroundColor: appbarColor.value, | ||||
|                       leading: PageBackButton( | ||||
|                         color: appbarColor.value, | ||||
|                         shadows: [appbarShadow], | ||||
|                       ), | ||||
|                       FlexibleSpaceBar( | ||||
|                         title: Text( | ||||
|                           data.nick, | ||||
|                           style: TextStyle( | ||||
|                             color: | ||||
|                                 appbarColor.value ?? | ||||
|                                 Theme.of(context).appBarTheme.foregroundColor, | ||||
|                       flexibleSpace: Stack( | ||||
|                         children: [ | ||||
|                           Positioned.fill( | ||||
|                             child: | ||||
|                                 data.background?.id != null | ||||
|                                     ? CloudImageWidget(file: data.background) | ||||
|                                     : Container( | ||||
|                                       color: | ||||
|                                           Theme.of( | ||||
|                                             context, | ||||
|                                           ).appBarTheme.backgroundColor, | ||||
|                                     ), | ||||
|                           ), | ||||
|                           FlexibleSpaceBar( | ||||
|                             title: Text( | ||||
|                               data.nick, | ||||
|                               style: TextStyle( | ||||
|                                 color: | ||||
|                                     appbarColor.value ?? | ||||
|                                     Theme.of( | ||||
|                                       context, | ||||
|                                     ).appBarTheme.foregroundColor, | ||||
|                                 shadows: [appbarShadow], | ||||
|                               ), | ||||
|                             ), | ||||
|                             background: | ||||
|                                 Container(), // Empty container since background is handled by Stack | ||||
|                           ), | ||||
|                         ], | ||||
|                       ), | ||||
|                     ) | ||||
|                     : null, | ||||
|             body: | ||||
|                 isWideScreen(context) | ||||
|                     ? Row( | ||||
|                       children: [ | ||||
|                         Flexible( | ||||
|                           flex: 4, | ||||
|                           child: CustomScrollView( | ||||
|                             slivers: [ | ||||
|                               SliverGap(16), | ||||
|                               SliverPostList(pubName: name), | ||||
|                               SliverGap( | ||||
|                                 MediaQuery.of(context).padding.bottom + 16, | ||||
|                               ), | ||||
|                             ], | ||||
|                           ).padding(left: 8), | ||||
|                         ), | ||||
|                         Flexible( | ||||
|                           flex: 3, | ||||
|                           child: Align( | ||||
|                             alignment: Alignment.topLeft, | ||||
|                             child: SingleChildScrollView( | ||||
|                               child: Column( | ||||
|                                 crossAxisAlignment: CrossAxisAlignment.start, | ||||
|                                 children: [ | ||||
|                                   publisherBasisWidget(data), | ||||
|                                   publisherBadgesWidget(data), | ||||
|                                   publisherVerificationWidget(data), | ||||
|                                   publisherBioWidget(data), | ||||
|                                 ], | ||||
|                               ), | ||||
|                             ), | ||||
|                           ), | ||||
|                         ), | ||||
|                       ], | ||||
|                     ) | ||||
|                     : CustomScrollView( | ||||
|                       slivers: [ | ||||
|                         SliverAppBar( | ||||
|                           foregroundColor: appbarColor.value, | ||||
|                           expandedHeight: 180, | ||||
|                           pinned: true, | ||||
|                           leading: PageBackButton( | ||||
|                             color: appbarColor.value, | ||||
|                             shadows: [appbarShadow], | ||||
|                           ), | ||||
|                         ), | ||||
|                         background: | ||||
|                             Container(), // Empty container since background is handled by Stack | ||||
|                       ), | ||||
|                     ], | ||||
|                   ), | ||||
|                 ), | ||||
|                 SliverToBoxAdapter( | ||||
|                   child: Row( | ||||
|                     crossAxisAlignment: CrossAxisAlignment.start, | ||||
|                     spacing: 20, | ||||
|                     children: [ | ||||
|                       GestureDetector( | ||||
|                         child: Badge( | ||||
|                           isLabelVisible: data.type == 0, | ||||
|                           padding: EdgeInsets.all(4), | ||||
|                           label: Icon( | ||||
|                             Symbols.launch, | ||||
|                             size: 16, | ||||
|                             color: Theme.of(context).colorScheme.onPrimary, | ||||
|                           ), | ||||
|                           backgroundColor: | ||||
|                               Theme.of(context).colorScheme.primary, | ||||
|                           offset: Offset(0, 48), | ||||
|                           child: ProfilePictureWidget( | ||||
|                             file: data.picture, | ||||
|                             radius: 32, | ||||
|                           ), | ||||
|                         ), | ||||
|                         onTap: () { | ||||
|                           Navigator.pop(context, true); | ||||
|                           if (data.account?.name != null) { | ||||
|                             context.pushNamed( | ||||
|                               'accountProfile', | ||||
|                               pathParameters: {'name': data.account!.name}, | ||||
|                             ); | ||||
|                           } | ||||
|                         }, | ||||
|                       ), | ||||
|                       Expanded( | ||||
|                         child: Column( | ||||
|                           crossAxisAlignment: CrossAxisAlignment.stretch, | ||||
|                           children: [ | ||||
|                             Row( | ||||
|                               spacing: 6, | ||||
|                               children: [ | ||||
|                                 Text(data.nick).fontSize(20), | ||||
|                                 if (data.verification != null) | ||||
|                                   VerificationMark(mark: data.verification!), | ||||
|                                 Text( | ||||
|                                   '@${data.name}', | ||||
|                                 ).fontSize(14).opacity(0.85), | ||||
|                               ], | ||||
|                             ), | ||||
|                             if (data.type == 0 && data.account != null) | ||||
|                               Row( | ||||
|                                 crossAxisAlignment: CrossAxisAlignment.center, | ||||
|                                 spacing: 6, | ||||
|                                 children: [ | ||||
|                                   Icon( | ||||
|                                     data.type == 0 | ||||
|                                         ? Symbols.person | ||||
|                                         : Symbols.workspaces, | ||||
|                                     fill: 1, | ||||
|                                     size: 17, | ||||
|                                   ), | ||||
|                                   Text( | ||||
|                                     'publisherBelongsTo'.tr( | ||||
|                                       args: ['@${data.account!.name}'], | ||||
|                                     ), | ||||
|                                   ).fontSize(14), | ||||
|                                 ], | ||||
|                               ).opacity(0.85).padding(bottom: 6), | ||||
|                             if (data.type == 0 && data.account != null) | ||||
|                               AccountStatusWidget( | ||||
|                                 uname: data.account!.name, | ||||
|                                 padding: EdgeInsets.zero, | ||||
|                           flexibleSpace: Stack( | ||||
|                             children: [ | ||||
|                               Positioned.fill( | ||||
|                                 child: | ||||
|                                     data.background?.id != null | ||||
|                                         ? CloudImageWidget( | ||||
|                                           file: data.background, | ||||
|                                         ) | ||||
|                                         : Container( | ||||
|                                           color: | ||||
|                                               Theme.of( | ||||
|                                                 context, | ||||
|                                               ).appBarTheme.backgroundColor, | ||||
|                                         ), | ||||
|                               ), | ||||
|                             subStatus | ||||
|                                 .when( | ||||
|                                   data: | ||||
|                                       (status) => FilledButton.icon( | ||||
|                                         onPressed: | ||||
|                                             subscribing.value | ||||
|                                                 ? null | ||||
|                                                 : (status.isSubscribed | ||||
|                                                     ? unsubscribe | ||||
|                                                     : subscribe), | ||||
|                                         icon: Icon( | ||||
|                                           status.isSubscribed | ||||
|                                               ? Symbols.remove_circle | ||||
|                                               : Symbols.add_circle, | ||||
|                                         ), | ||||
|                                         label: | ||||
|                                             Text( | ||||
|                                               status.isSubscribed | ||||
|                                                   ? 'unsubscribe' | ||||
|                                                   : 'subscribe', | ||||
|                                             ).tr(), | ||||
|                                         style: ButtonStyle( | ||||
|                                           visualDensity: VisualDensity( | ||||
|                                             vertical: -2, | ||||
|                                           ), | ||||
|                                         ), | ||||
|                                       ), | ||||
|                                   error: (_, _) => const SizedBox(), | ||||
|                                   loading: | ||||
|                                       () => const SizedBox( | ||||
|                                         height: 36, | ||||
|                                         child: Center( | ||||
|                                           child: SizedBox( | ||||
|                                             width: 20, | ||||
|                                             height: 20, | ||||
|                                             child: CircularProgressIndicator( | ||||
|                                               strokeWidth: 2, | ||||
|                                             ), | ||||
|                                           ), | ||||
|                                         ), | ||||
|                                       ), | ||||
|                                 ) | ||||
|                                 .padding(top: 8), | ||||
|                           ], | ||||
|                               FlexibleSpaceBar( | ||||
|                                 title: Text( | ||||
|                                   data.nick, | ||||
|                                   style: TextStyle( | ||||
|                                     color: | ||||
|                                         appbarColor.value ?? | ||||
|                                         Theme.of( | ||||
|                                           context, | ||||
|                                         ).appBarTheme.foregroundColor, | ||||
|                                     shadows: [appbarShadow], | ||||
|                                   ), | ||||
|                                 ), | ||||
|                                 background: | ||||
|                                     Container(), // Empty container since background is handled by Stack | ||||
|                               ), | ||||
|                             ], | ||||
|                           ), | ||||
|                         ), | ||||
|                       ), | ||||
|                     ], | ||||
|                   ).padding(horizontal: 24, top: 24), | ||||
|                 ), | ||||
|                 SliverToBoxAdapter( | ||||
|                   child: Column( | ||||
|                     children: [ | ||||
|                       if (badges.value?.isNotEmpty ?? false) | ||||
|                         BadgeList(badges: badges.value!).padding(top: 16), | ||||
|                       if (data.verification != null) | ||||
|                         VerificationStatusCard( | ||||
|                           mark: data.verification!, | ||||
|                         ).padding(top: 16), | ||||
|                     ], | ||||
|                   ).padding(horizontal: 24), | ||||
|                 ), | ||||
|                 SliverToBoxAdapter( | ||||
|                   child: const Divider(height: 1).padding(vertical: 24), | ||||
|                 ), | ||||
|                 SliverToBoxAdapter( | ||||
|                   child: Column( | ||||
|                     crossAxisAlignment: CrossAxisAlignment.stretch, | ||||
|                     children: [ | ||||
|                       Text('bio').tr().bold(), | ||||
|                       Text( | ||||
|                         data.bio.isEmpty ? 'descriptionNone'.tr() : data.bio, | ||||
|                       ), | ||||
|                     ], | ||||
|                   ).padding(horizontal: 24), | ||||
|                 ), | ||||
|                 SliverToBoxAdapter( | ||||
|                   child: const Divider(height: 1).padding(top: 24), | ||||
|                 ), | ||||
|                 SliverPostList(pubName: name), | ||||
|                 SliverGap(MediaQuery.of(context).padding.bottom + 16), | ||||
|               ], | ||||
|             ), | ||||
|                         SliverToBoxAdapter( | ||||
|                           child: publisherBasisWidget(data).padding(bottom: 8), | ||||
|                         ), | ||||
|                         SliverToBoxAdapter(child: publisherBadgesWidget(data)), | ||||
|                         SliverToBoxAdapter( | ||||
|                           child: publisherVerificationWidget(data), | ||||
|                         ), | ||||
|                         SliverToBoxAdapter(child: publisherBioWidget(data)), | ||||
|                         SliverToBoxAdapter(child: publisherCategoryTabWidget()), | ||||
|                         SliverPostList( | ||||
|                           key: ValueKey(categoryTab.value), | ||||
|                           pubName: name, | ||||
|                           type: switch (categoryTab.value) { | ||||
|                             1 => 0, | ||||
|                             2 => 1, | ||||
|                             _ => null, | ||||
|                           }, | ||||
|                         ), | ||||
|                         SliverGap(MediaQuery.of(context).padding.bottom + 16), | ||||
|                       ], | ||||
|                     ), | ||||
|           ), | ||||
|       error: | ||||
|           (error, stackTrace) => AppScaffold( | ||||
|             isNoBackground: false, | ||||
|             appBar: AppBar(leading: const PageBackButton()), | ||||
|             body: Center(child: Text(error.toString())), | ||||
|           ), | ||||
|       loading: | ||||
|           () => AppScaffold( | ||||
|             isNoBackground: false, | ||||
|             appBar: AppBar(leading: const PageBackButton()), | ||||
|             body: Center(child: CircularProgressIndicator()), | ||||
|           ), | ||||
|   | ||||
| @@ -6,7 +6,7 @@ part of 'pub_profile.dart'; | ||||
| // RiverpodGenerator | ||||
| // ************************************************************************** | ||||
|  | ||||
| String _$publisherHash() => r'3a7ae4d48765170aea42c7d6f4502d68f984dfab'; | ||||
| String _$publisherHash() => r'a1da21f0275421382e2882fd52c4e061c4675cf7'; | ||||
|  | ||||
| /// Copied from Dart SDK | ||||
| class _SystemHash { | ||||
| @@ -145,7 +145,7 @@ class _PublisherProviderElement | ||||
|   String get uname => (origin as PublisherProvider).uname; | ||||
| } | ||||
|  | ||||
| String _$publisherBadgesHash() => r'a5781deded7e682a781ccd7854418f050438e3f4'; | ||||
| String _$publisherBadgesHash() => r'527efad74225fbacf558ac5db160ecce53a60c62'; | ||||
|  | ||||
| /// See also [publisherBadges]. | ||||
| @ProviderFor(publisherBadges) | ||||
| @@ -268,7 +268,7 @@ class _PublisherBadgesProviderElement | ||||
| } | ||||
|  | ||||
| String _$publisherSubscriptionStatusHash() => | ||||
|     r'4eb6741c40775c814e71b6a98b8f1e2d84bf7e30'; | ||||
|     r'634262ce519e1c8288267df11e08e1d4acaa4a44'; | ||||
|  | ||||
| /// See also [publisherSubscriptionStatus]. | ||||
| @ProviderFor(publisherSubscriptionStatus) | ||||
|   | ||||
| @@ -79,7 +79,7 @@ class RealmDetailScreen extends HookConsumerWidget { | ||||
|     ); | ||||
|  | ||||
|     return AppScaffold( | ||||
|       noBackground: false, | ||||
|       isNoBackground: false, | ||||
|       body: realmState.when( | ||||
|         loading: () => const Center(child: CircularProgressIndicator()), | ||||
|         error: (error, _) => Center(child: Text('Error: $error')), | ||||
| @@ -321,10 +321,10 @@ class _RealmActionMenu extends HookConsumerWidget { | ||||
|                               showConfirmAlert( | ||||
|                                 'leaveRealmHint'.tr(), | ||||
|                                 'leaveRealm'.tr(), | ||||
|                               ).then((confirm) { | ||||
|                               ).then((confirm) async { | ||||
|                                 if (confirm) { | ||||
|                                   final client = ref.watch(apiClientProvider); | ||||
|                                   client.delete( | ||||
|                                   await client.delete( | ||||
|                                     '/sphere/realms/$realmSlug/members/me', | ||||
|                                   ); | ||||
|                                   ref.invalidate(realmsJoinedProvider); | ||||
| @@ -361,10 +361,12 @@ class _RealmActionMenu extends HookConsumerWidget { | ||||
|                       showConfirmAlert( | ||||
|                         'leaveRealmHint'.tr(), | ||||
|                         'leaveRealm'.tr(), | ||||
|                       ).then((confirm) { | ||||
|                       ).then((confirm) async { | ||||
|                         if (confirm) { | ||||
|                           final client = ref.watch(apiClientProvider); | ||||
|                           client.delete('/sphere/realms/$realmSlug/members/me'); | ||||
|                           await client.delete( | ||||
|                             '/sphere/realms/$realmSlug/members/me', | ||||
|                           ); | ||||
|                           ref.invalidate(realmsJoinedProvider); | ||||
|                           if (context.mounted) { | ||||
|                             context.pop(true); | ||||
|   | ||||
| @@ -41,7 +41,7 @@ class RealmListScreen extends HookConsumerWidget { | ||||
|     final realmInvites = ref.watch(realmInvitesProvider); | ||||
|  | ||||
|     return AppScaffold( | ||||
|       noBackground: false, | ||||
|       isNoBackground: false, | ||||
|       appBar: AppBar( | ||||
|         title: const Text('realms').tr(), | ||||
|         actions: [ | ||||
| @@ -279,7 +279,7 @@ class EditRealmScreen extends HookConsumerWidget { | ||||
|     } | ||||
|  | ||||
|     return AppScaffold( | ||||
|       noBackground: false, | ||||
|       isNoBackground: false, | ||||
|       appBar: AppBar( | ||||
|         title: Text(slug == null ? 'createRealm'.tr() : 'editRealm'.tr()), | ||||
|         leading: const PageBackButton(), | ||||
|   | ||||
| @@ -552,7 +552,7 @@ class SettingsScreen extends HookConsumerWidget { | ||||
|     } | ||||
|  | ||||
|     return AppScaffold( | ||||
|       noBackground: false, | ||||
|       isNoBackground: false, | ||||
|       appBar: AppBar( | ||||
|         title: Text('settings').tr(), | ||||
|         actions: | ||||
|   | ||||
| @@ -15,6 +15,7 @@ Future<XFile?> cropImage( | ||||
|   BuildContext context, { | ||||
|   required XFile image, | ||||
|   List<CropAspectRatio?>? allowedAspectRatios, | ||||
|   bool replacePath = false, | ||||
| }) async { | ||||
|   final result = await showMaterialImageCropper( | ||||
|     context, | ||||
| @@ -34,7 +35,7 @@ Future<XFile?> cropImage( | ||||
|   croppedFile.dispose(); | ||||
|   return XFile.fromData( | ||||
|     croppedBytes.buffer.asUint8List(), | ||||
|     path: image.path, | ||||
|     path: !replacePath ? image.path : null, | ||||
|     mimeType: image.mimeType, | ||||
|   ); | ||||
| } | ||||
|   | ||||
| @@ -20,6 +20,33 @@ extension DurationFormatter on Duration { | ||||
|     return '${isNegative ? '-' : ''}$hours:$minutes:$seconds'; | ||||
|   } | ||||
|  | ||||
|   String formatShortDuration() { | ||||
|     final isNegative = inMicroseconds < 0; | ||||
|     final positiveDuration = isNegative ? -this : this; | ||||
|  | ||||
|     final hours = positiveDuration.inHours; | ||||
|     final minutes = (positiveDuration.inMinutes % 60).toString().padLeft( | ||||
|       2, | ||||
|       '0', | ||||
|     ); | ||||
|     final seconds = (positiveDuration.inSeconds % 60).toString().padLeft( | ||||
|       2, | ||||
|       '0', | ||||
|     ); | ||||
|     final milliseconds = (positiveDuration.inMilliseconds % 1000) | ||||
|         .toString() | ||||
|         .padLeft(3, '0'); | ||||
|  | ||||
|     String result; | ||||
|     if (hours > 0) { | ||||
|       result = | ||||
|           '${isNegative ? '-' : ''}${hours.toString().padLeft(2, '0')}:$minutes:$seconds.$milliseconds'; | ||||
|     } else { | ||||
|       result = '${isNegative ? '-' : ''}$minutes:$seconds.$milliseconds'; | ||||
|     } | ||||
|     return result; | ||||
|   } | ||||
|  | ||||
|   String formatOffset() { | ||||
|     final isNegative = inMicroseconds < 0; | ||||
|     final positiveDuration = isNegative ? -this : this; | ||||
|   | ||||
							
								
								
									
										30
									
								
								lib/utils/mapping.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										30
									
								
								lib/utils/mapping.dart
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,30 @@ | ||||
| String _upperCamelToLowerSnake(String input) { | ||||
|   final regex = RegExp(r'(?<=[a-z0-9])([A-Z])'); | ||||
|   return input | ||||
|       .replaceAllMapped(regex, (match) => '_${match.group(0)}') | ||||
|       .toLowerCase(); | ||||
| } | ||||
|  | ||||
| Map<String, dynamic> convertMapKeysToSnakeCase(Map<String, dynamic> input) { | ||||
|   final result = <String, dynamic>{}; | ||||
|  | ||||
|   input.forEach((key, value) { | ||||
|     final newKey = _upperCamelToLowerSnake(key); | ||||
|  | ||||
|     if (value is Map<String, dynamic>) { | ||||
|       result[newKey] = convertMapKeysToSnakeCase(value); | ||||
|     } else if (value is List) { | ||||
|       result[newKey] = | ||||
|           value.map((item) { | ||||
|             if (item is Map<String, dynamic>) { | ||||
|               return convertMapKeysToSnakeCase(item); | ||||
|             } | ||||
|             return item; | ||||
|           }).toList(); | ||||
|     } else { | ||||
|       result[newKey] = value; | ||||
|     } | ||||
|   }); | ||||
|  | ||||
|   return result; | ||||
| } | ||||
| @@ -22,11 +22,11 @@ class AccountName extends StatelessWidget { | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     var nameStyle = (style ?? TextStyle()); | ||||
|     if (account.profile.stellarMembership != null) { | ||||
|     if (account.perkSubscription != null) { | ||||
|       nameStyle = nameStyle.copyWith( | ||||
|         color: (switch (account.profile.stellarMembership!.identifier) { | ||||
|         color: (switch (account.perkSubscription!.identifier) { | ||||
|           'solian.stellar.primary' => Colors.blueAccent, | ||||
|           'solian.stellar.nova' => Colors.indigoAccent, | ||||
|           'solian.stellar.nova' => Color.fromRGBO(57, 197, 187, 1), | ||||
|           'solian.stellar.supernova' => Colors.amberAccent, | ||||
|           _ => null, | ||||
|         }), | ||||
| @@ -38,8 +38,8 @@ class AccountName extends StatelessWidget { | ||||
|       spacing: 4, | ||||
|       children: [ | ||||
|         Flexible(child: Text(account.nick, style: nameStyle)), | ||||
|         if (account.profile.stellarMembership != null) | ||||
|           StellarMembershipMark(membership: account.profile.stellarMembership!), | ||||
|         if (account.perkSubscription != null) | ||||
|           StellarMembershipMark(membership: account.perkSubscription!), | ||||
|         if (account.profile.verification != null) | ||||
|           VerificationMark(mark: account.profile.verification!), | ||||
|       ], | ||||
| @@ -101,7 +101,7 @@ class StellarMembershipMark extends StatelessWidget { | ||||
|       case 'solian.stellar.primary': | ||||
|         return Colors.blue; | ||||
|       case 'solian.stellar.nova': | ||||
|         return Colors.indigo; | ||||
|         return Color.fromRGBO(57, 197, 187, 1); | ||||
|       case 'solian.stellar.supernova': | ||||
|         return Colors.amber; | ||||
|       default: | ||||
| @@ -115,7 +115,7 @@ class StellarMembershipMark extends StatelessWidget { | ||||
|  | ||||
|     final tierName = _getMembershipTierName(membership.identifier); | ||||
|     final tierColor = _getMembershipTierColor(membership.identifier); | ||||
|     final tierIcon = Symbols.award_star; | ||||
|     final tierIcon = Symbols.kid_star; | ||||
|  | ||||
|     return Tooltip( | ||||
|       richMessage: TextSpan( | ||||
| @@ -140,30 +140,27 @@ class VerificationStatusCard extends StatelessWidget { | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     return Card( | ||||
|       margin: EdgeInsets.zero, | ||||
|       child: Column( | ||||
|         crossAxisAlignment: CrossAxisAlignment.start, | ||||
|         children: [ | ||||
|           Icon( | ||||
|             mark.type == 4 | ||||
|                 ? Symbols.play_circle | ||||
|                 : mark.type == 0 | ||||
|                 ? Symbols.build_circle | ||||
|                 : Symbols.verified, | ||||
|             size: 32, | ||||
|             color: kVerificationMarkColors[mark.type], | ||||
|             fill: 1, | ||||
|           ), | ||||
|           const Gap(8), | ||||
|           Text(mark.title ?? 'No title').bold(), | ||||
|           Text(mark.description ?? 'descriptionNone'.tr()), | ||||
|           const Gap(6), | ||||
|           Text( | ||||
|             'Verified by\n${mark.verifiedBy ?? 'No one verified it'}', | ||||
|           ).fontSize(11).opacity(0.8), | ||||
|         ], | ||||
|       ).padding(horizontal: 24, vertical: 16), | ||||
|     ); | ||||
|     return Column( | ||||
|       crossAxisAlignment: CrossAxisAlignment.start, | ||||
|       children: [ | ||||
|         Icon( | ||||
|           mark.type == 4 | ||||
|               ? Symbols.play_circle | ||||
|               : mark.type == 0 | ||||
|               ? Symbols.build_circle | ||||
|               : Symbols.verified, | ||||
|           size: 32, | ||||
|           color: kVerificationMarkColors[mark.type], | ||||
|           fill: 1, | ||||
|         ), | ||||
|         const Gap(8), | ||||
|         Text(mark.title ?? 'No title').bold(), | ||||
|         Text(mark.description ?? 'descriptionNone'.tr()), | ||||
|         const Gap(6), | ||||
|         Text( | ||||
|           'Verified by\n${mark.verifiedBy ?? 'No one verified it'}', | ||||
|         ).fontSize(11).opacity(0.8), | ||||
|       ], | ||||
|     ).padding(horizontal: 24, vertical: 16); | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -26,7 +26,7 @@ class AccountProfileCard extends HookConsumerWidget { | ||||
|   Widget build(BuildContext context, WidgetRef ref) { | ||||
|     final account = ref.watch(accountProvider(uname)); | ||||
|     final width = | ||||
|         math.max(MediaQuery.of(context).size.width - 80, 360).toDouble(); | ||||
|         math.min(MediaQuery.of(context).size.width - 80, 360).toDouble(); | ||||
|     return PopupCard( | ||||
|       elevation: 8, | ||||
|       shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(12.0)), | ||||
| @@ -105,7 +105,10 @@ class AccountProfileCard extends HookConsumerWidget { | ||||
|                       FilledButton.tonalIcon( | ||||
|                         onPressed: () { | ||||
|                           Navigator.pop(context); | ||||
|                           context.pushNamed('accountProfile', pathParameters: {'name': data.name}); | ||||
|                           context.pushNamed( | ||||
|                             'accountProfile', | ||||
|                             pathParameters: {'name': data.name}, | ||||
|                           ); | ||||
|                         }, | ||||
|                         icon: const Icon(Symbols.launch), | ||||
|                         label: Text('accountProfileView').tr(), | ||||
| @@ -164,6 +167,7 @@ Future<void> showAccountProfileCard( | ||||
|     offset: offset ?? Offset.zero, | ||||
|     context: context, | ||||
|     builder: (context) => AccountProfileCard(uname: uname), | ||||
|     alignment: Alignment.center, | ||||
|     dimBackground: true, | ||||
|   ); | ||||
| } | ||||
|   | ||||
| @@ -32,12 +32,12 @@ class BadgeItem extends StatelessWidget { | ||||
|       child: Container( | ||||
|         padding: const EdgeInsets.all(4), | ||||
|         decoration: BoxDecoration( | ||||
|           color: (template?.color ?? Colors.blue).withOpacity(0.1), | ||||
|           color: (template?.color ?? Colors.blue).withOpacity(0.2), | ||||
|           shape: BoxShape.circle, | ||||
|         ), | ||||
|         child: Icon( | ||||
|           template?.icon ?? Icons.stars, | ||||
|           color: template?.color ?? Colors.orange, | ||||
|           color: template?.color ?? Colors.blue, | ||||
|           size: 20, | ||||
|         ), | ||||
|       ), | ||||
|   | ||||
| @@ -26,6 +26,8 @@ class FortuneGraphWidget extends HookConsumerWidget { | ||||
|  | ||||
|   final String? eventCalanderUser; | ||||
|  | ||||
|   final EdgeInsets? margin; | ||||
|  | ||||
|   const FortuneGraphWidget({ | ||||
|     super.key, | ||||
|     required this.events, | ||||
| @@ -34,6 +36,7 @@ class FortuneGraphWidget extends HookConsumerWidget { | ||||
|     this.height = 180, | ||||
|     this.onPointSelected, | ||||
|     this.eventCalanderUser, | ||||
|     this.margin, | ||||
|   }); | ||||
|  | ||||
|   @override | ||||
| @@ -249,7 +252,7 @@ class FortuneGraphWidget extends HookConsumerWidget { | ||||
|     if (constrainWidth) { | ||||
|       return ConstrainedBox( | ||||
|         constraints: BoxConstraints(maxWidth: maxWidth), | ||||
|         child: Card(margin: EdgeInsets.all(16), child: content), | ||||
|         child: Card(margin: margin ?? EdgeInsets.all(16), child: content), | ||||
|       ).center(); | ||||
|     } | ||||
|  | ||||
|   | ||||
| @@ -32,7 +32,7 @@ class RestorePurchaseSheet extends HookConsumerWidget { | ||||
|       try { | ||||
|         final client = ref.read(apiClientProvider); | ||||
|         await client.post( | ||||
|           '/subscriptions/order/restore/${selectedProvider.value!}', | ||||
|           '/id/subscriptions/order/restore/${selectedProvider.value!}', | ||||
|           data: {'order_id': orderIdController.text.trim()}, | ||||
|         ); | ||||
|  | ||||
|   | ||||
| @@ -86,6 +86,8 @@ class AccountStatusCreationWidget extends HookConsumerWidget { | ||||
|       onTap: () { | ||||
|         showModalBottomSheet( | ||||
|           context: context, | ||||
|           isScrollControlled: true, | ||||
|           useRootNavigator: true, | ||||
|           builder: | ||||
|               (context) => AccountStatusCreationSheet( | ||||
|                 initialStatus: | ||||
|   | ||||
| @@ -9,6 +9,7 @@ import 'package:island/pods/network.dart'; | ||||
| import 'package:island/pods/userinfo.dart'; | ||||
| import 'package:island/widgets/account/status.dart'; | ||||
| import 'package:island/widgets/alert.dart'; | ||||
| import 'package:island/widgets/content/sheet.dart'; | ||||
| import 'package:material_symbols_icons/symbols.dart'; | ||||
|  | ||||
| class AccountStatusCreationSheet extends HookConsumerWidget { | ||||
| @@ -49,7 +50,7 @@ class AccountStatusCreationSheet extends HookConsumerWidget { | ||||
|         final user = ref.watch(userInfoProvider); | ||||
|         final apiClient = ref.read(apiClientProvider); | ||||
|         await apiClient.request( | ||||
|           '/accounts/me/statuses', | ||||
|           '/id/accounts/me/statuses', | ||||
|           data: { | ||||
|             'attitude': attitude.value, | ||||
|             'is_invisible': isInvisible.value, | ||||
| @@ -71,178 +72,145 @@ class AccountStatusCreationSheet extends HookConsumerWidget { | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     return Container( | ||||
|       constraints: BoxConstraints( | ||||
|         maxHeight: MediaQuery.of(context).size.height * 0.8, | ||||
|       ), | ||||
|       child: Column( | ||||
|         children: [ | ||||
|           Padding( | ||||
|             padding: EdgeInsets.only(top: 16, left: 20, right: 16, bottom: 12), | ||||
|             child: Row( | ||||
|               children: [ | ||||
|                 Text( | ||||
|                   initialStatus == null | ||||
|                       ? 'statusCreate'.tr() | ||||
|                       : 'statusUpdate'.tr(), | ||||
|                   style: Theme.of(context).textTheme.headlineSmall?.copyWith( | ||||
|                     fontWeight: FontWeight.w600, | ||||
|                     letterSpacing: -0.5, | ||||
|                   ), | ||||
|     return SheetScaffold( | ||||
|       heightFactor: 0.6, | ||||
|       titleText: | ||||
|           initialStatus == null ? 'statusCreate'.tr() : 'statusUpdate'.tr(), | ||||
|       actions: [ | ||||
|         TextButton.icon( | ||||
|           onPressed: | ||||
|               submitting.value | ||||
|                   ? null | ||||
|                   : () { | ||||
|                     submitStatus(); | ||||
|                   }, | ||||
|           icon: const Icon(Symbols.upload), | ||||
|           label: Text(initialStatus == null ? 'create' : 'update').tr(), | ||||
|           style: ButtonStyle( | ||||
|             visualDensity: VisualDensity( | ||||
|               horizontal: VisualDensity.minimumDensity, | ||||
|             ), | ||||
|             foregroundColor: WidgetStatePropertyAll( | ||||
|               Theme.of(context).colorScheme.onSurface, | ||||
|             ), | ||||
|           ), | ||||
|         ), | ||||
|         if (initialStatus != null) | ||||
|           IconButton( | ||||
|             icon: const Icon(Symbols.delete), | ||||
|             onPressed: submitting.value ? null : () => clearStatus(), | ||||
|             style: IconButton.styleFrom(minimumSize: const Size(36, 36)), | ||||
|           ), | ||||
|       ], | ||||
|       child: SingleChildScrollView( | ||||
|         padding: const EdgeInsets.symmetric(horizontal: 20), | ||||
|         child: Column( | ||||
|           crossAxisAlignment: CrossAxisAlignment.stretch, | ||||
|           children: [ | ||||
|             const Gap(24), | ||||
|             TextField( | ||||
|               controller: labelController, | ||||
|               decoration: InputDecoration( | ||||
|                 labelText: 'statusLabel'.tr(), | ||||
|                 border: const OutlineInputBorder( | ||||
|                   borderRadius: BorderRadius.all(Radius.circular(12)), | ||||
|                 ), | ||||
|                 const Spacer(), | ||||
|                 TextButton.icon( | ||||
|                   onPressed: | ||||
|                       submitting.value | ||||
|                           ? null | ||||
|                           : () { | ||||
|                             submitStatus(); | ||||
|                           }, | ||||
|                   icon: const Icon(Symbols.upload), | ||||
|                   label: Text(initialStatus == null ? 'create' : 'update').tr(), | ||||
|                   style: ButtonStyle( | ||||
|                     visualDensity: VisualDensity( | ||||
|                       horizontal: VisualDensity.minimumDensity, | ||||
|                     ), | ||||
|                     foregroundColor: WidgetStatePropertyAll( | ||||
|                       Theme.of(context).colorScheme.onSurface, | ||||
|                     ), | ||||
|                   ), | ||||
|               ), | ||||
|               onTapOutside: | ||||
|                   (_) => FocusManager.instance.primaryFocus?.unfocus(), | ||||
|             ), | ||||
|             const SizedBox(height: 24), | ||||
|             Text( | ||||
|               'statusAttitude'.tr(), | ||||
|               style: Theme.of(context).textTheme.titleMedium, | ||||
|             ), | ||||
|             const SizedBox(height: 8), | ||||
|             SegmentedButton( | ||||
|               segments: [ | ||||
|                 ButtonSegment( | ||||
|                   value: 0, | ||||
|                   icon: const Icon(Symbols.sentiment_satisfied), | ||||
|                   label: Text('attitudePositive'.tr()), | ||||
|                 ), | ||||
|                 if (initialStatus != null) | ||||
|                   IconButton( | ||||
|                     icon: const Icon(Symbols.delete), | ||||
|                     onPressed: submitting.value ? null : () => clearStatus(), | ||||
|                     style: IconButton.styleFrom( | ||||
|                       minimumSize: const Size(36, 36), | ||||
|                     ), | ||||
|                   ), | ||||
|                 IconButton( | ||||
|                   icon: const Icon(Symbols.close), | ||||
|                   onPressed: () => Navigator.pop(context), | ||||
|                   style: IconButton.styleFrom(minimumSize: const Size(36, 36)), | ||||
|                 ButtonSegment( | ||||
|                   value: 1, | ||||
|                   icon: const Icon(Symbols.sentiment_stressed), | ||||
|                   label: Text('attitudeNeutral'.tr()), | ||||
|                 ), | ||||
|                 ButtonSegment( | ||||
|                   value: 2, | ||||
|                   icon: const Icon(Symbols.sentiment_sad), | ||||
|                   label: Text('attitudeNegative'.tr()), | ||||
|                 ), | ||||
|               ], | ||||
|               selected: {attitude.value}, | ||||
|               onSelectionChanged: (Set<int> newSelection) { | ||||
|                 attitude.value = newSelection.first; | ||||
|               }, | ||||
|             ), | ||||
|           ), | ||||
|           const Divider(height: 1), | ||||
|           Expanded( | ||||
|             child: SingleChildScrollView( | ||||
|               padding: const EdgeInsets.symmetric(horizontal: 20), | ||||
|               child: Column( | ||||
|                 crossAxisAlignment: CrossAxisAlignment.start, | ||||
|                 children: [ | ||||
|                   const Gap(24), | ||||
|                   TextField( | ||||
|                     controller: labelController, | ||||
|                     decoration: InputDecoration( | ||||
|                       labelText: 'statusLabel'.tr(), | ||||
|                       border: const OutlineInputBorder( | ||||
|                         borderRadius: BorderRadius.all(Radius.circular(12)), | ||||
|                       ), | ||||
|                     ), | ||||
|                     onTapOutside: | ||||
|                         (_) => FocusManager.instance.primaryFocus?.unfocus(), | ||||
|                   ), | ||||
|                   const SizedBox(height: 24), | ||||
|                   Text( | ||||
|                     'statusAttitude'.tr(), | ||||
|                     style: Theme.of(context).textTheme.titleMedium, | ||||
|                   ), | ||||
|                   const SizedBox(height: 8), | ||||
|                   SegmentedButton( | ||||
|                     segments: [ | ||||
|                       ButtonSegment( | ||||
|                         value: 0, | ||||
|                         icon: const Icon(Symbols.sentiment_satisfied), | ||||
|                         label: Text('attitudePositive'.tr()), | ||||
|                       ), | ||||
|                       ButtonSegment( | ||||
|                         value: 1, | ||||
|                         icon: const Icon(Symbols.sentiment_stressed), | ||||
|                         label: Text('attitudeNeutral'.tr()), | ||||
|                       ), | ||||
|                       ButtonSegment( | ||||
|                         value: 2, | ||||
|                         icon: const Icon(Symbols.sentiment_sad), | ||||
|                         label: Text('attitudeNegative'.tr()), | ||||
|                       ), | ||||
|                     ], | ||||
|                     selected: {attitude.value}, | ||||
|                     onSelectionChanged: (Set<int> newSelection) { | ||||
|                       attitude.value = newSelection.first; | ||||
|                     }, | ||||
|                   ), | ||||
|                   const Gap(12), | ||||
|                   SwitchListTile( | ||||
|                     title: Text('statusInvisible'.tr()), | ||||
|                     subtitle: Text('statusInvisibleDescription'.tr()), | ||||
|                     value: isInvisible.value, | ||||
|                     contentPadding: EdgeInsets.symmetric(horizontal: 8), | ||||
|                     onChanged: (bool value) { | ||||
|                       isInvisible.value = value; | ||||
|                     }, | ||||
|                   ), | ||||
|                   SwitchListTile( | ||||
|                     title: Text('statusNotDisturb'.tr()), | ||||
|                     subtitle: Text('statusNotDisturbDescription'.tr()), | ||||
|                     value: isNotDisturb.value, | ||||
|                     contentPadding: EdgeInsets.symmetric(horizontal: 8), | ||||
|                     onChanged: (bool value) { | ||||
|                       isNotDisturb.value = value; | ||||
|                     }, | ||||
|                   ), | ||||
|                   const SizedBox(height: 24), | ||||
|                   Text( | ||||
|                     'statusClearTime'.tr(), | ||||
|                     style: Theme.of(context).textTheme.titleMedium, | ||||
|                   ), | ||||
|                   const SizedBox(height: 8), | ||||
|                   ListTile( | ||||
|                     title: Text( | ||||
|                       clearedAt.value == null | ||||
|                           ? 'statusNoAutoClear'.tr() | ||||
|                           : DateFormat.yMMMd().add_jm().format( | ||||
|                             clearedAt.value!, | ||||
|                           ), | ||||
|                     ), | ||||
|                     trailing: const Icon(Symbols.schedule), | ||||
|                     shape: RoundedRectangleBorder( | ||||
|                       borderRadius: BorderRadius.circular(8), | ||||
|                       side: BorderSide( | ||||
|                         color: Theme.of(context).colorScheme.outline, | ||||
|                       ), | ||||
|                     ), | ||||
|                     onTap: () async { | ||||
|                       final now = DateTime.now(); | ||||
|                       final date = await showDatePicker( | ||||
|                         context: context, | ||||
|                         initialDate: now, | ||||
|                         firstDate: now, | ||||
|                         lastDate: now.add(const Duration(days: 365)), | ||||
|                       ); | ||||
|                       if (date == null) return; | ||||
|                       if (!context.mounted) return; | ||||
|                       final time = await showTimePicker( | ||||
|                         context: context, | ||||
|                         initialTime: TimeOfDay.now(), | ||||
|                       ); | ||||
|                       if (time == null) return; | ||||
|                       clearedAt.value = DateTime( | ||||
|                         date.year, | ||||
|                         date.month, | ||||
|                         date.day, | ||||
|                         time.hour, | ||||
|                         time.minute, | ||||
|                       ); | ||||
|                     }, | ||||
|                   ), | ||||
|                   Gap(MediaQuery.of(context).padding.bottom + 24), | ||||
|                 ], | ||||
|             const Gap(12), | ||||
|             SwitchListTile( | ||||
|               title: Text('statusInvisible'.tr()), | ||||
|               subtitle: Text('statusInvisibleDescription'.tr()), | ||||
|               value: isInvisible.value, | ||||
|               contentPadding: EdgeInsets.symmetric(horizontal: 8), | ||||
|               onChanged: (bool value) { | ||||
|                 isInvisible.value = value; | ||||
|               }, | ||||
|             ), | ||||
|             SwitchListTile( | ||||
|               title: Text('statusNotDisturb'.tr()), | ||||
|               subtitle: Text('statusNotDisturbDescription'.tr()), | ||||
|               value: isNotDisturb.value, | ||||
|               contentPadding: EdgeInsets.symmetric(horizontal: 8), | ||||
|               onChanged: (bool value) { | ||||
|                 isNotDisturb.value = value; | ||||
|               }, | ||||
|             ), | ||||
|             const SizedBox(height: 24), | ||||
|             Text( | ||||
|               'statusClearTime'.tr(), | ||||
|               style: Theme.of(context).textTheme.titleMedium, | ||||
|             ), | ||||
|             const SizedBox(height: 8), | ||||
|             ListTile( | ||||
|               title: Text( | ||||
|                 clearedAt.value == null | ||||
|                     ? 'statusNoAutoClear'.tr() | ||||
|                     : DateFormat.yMMMd().add_jm().format(clearedAt.value!), | ||||
|               ), | ||||
|               trailing: const Icon(Symbols.schedule), | ||||
|               shape: RoundedRectangleBorder( | ||||
|                 borderRadius: BorderRadius.circular(8), | ||||
|                 side: BorderSide(color: Theme.of(context).colorScheme.outline), | ||||
|               ), | ||||
|               onTap: () async { | ||||
|                 final now = DateTime.now(); | ||||
|                 final date = await showDatePicker( | ||||
|                   context: context, | ||||
|                   initialDate: now, | ||||
|                   firstDate: now, | ||||
|                   lastDate: now.add(const Duration(days: 365)), | ||||
|                 ); | ||||
|                 if (date == null) return; | ||||
|                 if (!context.mounted) return; | ||||
|                 final time = await showTimePicker( | ||||
|                   context: context, | ||||
|                   initialTime: TimeOfDay.now(), | ||||
|                 ); | ||||
|                 if (time == null) return; | ||||
|                 clearedAt.value = DateTime( | ||||
|                   date.year, | ||||
|                   date.month, | ||||
|                   date.day, | ||||
|                   time.hour, | ||||
|                   time.minute, | ||||
|                 ); | ||||
|               }, | ||||
|             ), | ||||
|           ), | ||||
|         ], | ||||
|             Gap(MediaQuery.of(context).padding.bottom + 24), | ||||
|           ], | ||||
|         ), | ||||
|       ), | ||||
|     ); | ||||
|   } | ||||
|   | ||||
| @@ -165,7 +165,7 @@ class AppScaffold extends StatelessWidget { | ||||
|   final AppBar? appBar; | ||||
|   final DrawerCallback? onDrawerChanged; | ||||
|   final DrawerCallback? onEndDrawerChanged; | ||||
|   final bool? noBackground; | ||||
|   final bool? isNoBackground; | ||||
|   final bool? extendBody; | ||||
|  | ||||
|   const AppScaffold({ | ||||
| @@ -181,7 +181,7 @@ class AppScaffold extends StatelessWidget { | ||||
|     this.endDrawer, | ||||
|     this.onDrawerChanged, | ||||
|     this.onEndDrawerChanged, | ||||
|     this.noBackground, | ||||
|     this.isNoBackground, | ||||
|     this.extendBody, | ||||
|   }); | ||||
|  | ||||
| @@ -190,7 +190,7 @@ class AppScaffold extends StatelessWidget { | ||||
|     final appBarHeight = appBar?.preferredSize.height ?? 0; | ||||
|     final safeTop = MediaQuery.of(context).padding.top; | ||||
|  | ||||
|     final noBackground = this.noBackground ?? isWideScreen(context); | ||||
|     final noBackground = isNoBackground ?? isWideScreen(context); | ||||
|  | ||||
|     final content = Column( | ||||
|       children: [ | ||||
|   | ||||
| @@ -4,9 +4,11 @@ import 'package:go_router/go_router.dart'; | ||||
| import 'package:gap/gap.dart'; | ||||
| import 'package:hooks_riverpod/hooks_riverpod.dart'; | ||||
| import 'package:island/pods/call.dart'; | ||||
| import 'package:island/pods/network.dart'; | ||||
| import 'package:island/widgets/alert.dart'; | ||||
| import 'package:island/widgets/chat/call_participant_tile.dart'; | ||||
| import 'package:island/widgets/content/sheet.dart'; | ||||
| import 'package:material_symbols_icons/symbols.dart'; | ||||
| import 'package:styled_widget/styled_widget.dart'; | ||||
| import 'package:livekit_client/livekit_client.dart'; | ||||
|  | ||||
| @@ -20,8 +22,10 @@ class CallControlsBar extends HookConsumerWidget { | ||||
|  | ||||
|     return Container( | ||||
|       padding: const EdgeInsets.symmetric(horizontal: 20, vertical: 16), | ||||
|       child: Row( | ||||
|         mainAxisAlignment: MainAxisAlignment.center, | ||||
|       child: Wrap( | ||||
|         alignment: WrapAlignment.center, | ||||
|         runSpacing: 16, | ||||
|         spacing: 16, | ||||
|         children: [ | ||||
|           _buildCircularButtonWithDropdown( | ||||
|             context: context, | ||||
| @@ -33,7 +37,6 @@ class CallControlsBar extends HookConsumerWidget { | ||||
|             hasDropdown: true, | ||||
|             deviceType: 'videoinput', | ||||
|           ), | ||||
|           const Gap(16), | ||||
|           _buildCircularButton( | ||||
|             icon: | ||||
|                 callState.isScreenSharing | ||||
| @@ -42,7 +45,6 @@ class CallControlsBar extends HookConsumerWidget { | ||||
|             onPressed: () => callNotifier.toggleScreenShare(), | ||||
|             backgroundColor: const Color(0xFF424242), | ||||
|           ), | ||||
|           const Gap(16), | ||||
|           _buildCircularButtonWithDropdown( | ||||
|             context: context, | ||||
|             ref: ref, | ||||
| @@ -52,10 +54,62 @@ class CallControlsBar extends HookConsumerWidget { | ||||
|             hasDropdown: true, | ||||
|             deviceType: 'audioinput', | ||||
|           ), | ||||
|           const Gap(16), | ||||
|           _buildCircularButton( | ||||
|             icon: | ||||
|                 callState.isSpeakerphone | ||||
|                     ? Symbols.mobile_speaker | ||||
|                     : Symbols.ear_sound, | ||||
|             onPressed: () => callNotifier.toggleSpeakerphone(), | ||||
|             backgroundColor: const Color(0xFF424242), | ||||
|           ), | ||||
|           _buildCircularButton( | ||||
|             icon: Icons.call_end, | ||||
|             onPressed: () => callNotifier.disconnect(), | ||||
|             onPressed: | ||||
|                 () => showModalBottomSheet( | ||||
|                   context: context, | ||||
|                   isScrollControlled: true, | ||||
|                   useRootNavigator: true, | ||||
|                   builder: | ||||
|                       (innerContext) => Column( | ||||
|                         mainAxisSize: MainAxisSize.min, | ||||
|                         children: [ | ||||
|                           ListTile( | ||||
|                             leading: const Icon(Symbols.logout, fill: 1), | ||||
|                             title: Text('callLeave').tr(), | ||||
|                             onTap: () { | ||||
|                               callNotifier.disconnect(); | ||||
|                               Navigator.of(context).pop(); | ||||
|                               Navigator.of(innerContext).pop(); | ||||
|                             }, | ||||
|                           ), | ||||
|                           ListTile( | ||||
|                             leading: const Icon(Symbols.call_end, fill: 1), | ||||
|                             iconColor: Colors.red, | ||||
|                             title: Text('callEnd').tr(), | ||||
|                             onTap: () async { | ||||
|                               callNotifier.disconnect(); | ||||
|                               final apiClient = ref.watch(apiClientProvider); | ||||
|                               try { | ||||
|                                 showLoadingModal(context); | ||||
|                                 await apiClient.delete( | ||||
|                                   '/sphere/chat/realtime/${callNotifier.roomId}', | ||||
|                                 ); | ||||
|                                 callNotifier.dispose(); | ||||
|                                 if (context.mounted) { | ||||
|                                   Navigator.of(context).pop(); | ||||
|                                   Navigator.of(innerContext).pop(); | ||||
|                                 } | ||||
|                               } catch (err) { | ||||
|                                 showErrorAlert(err); | ||||
|                               } finally { | ||||
|                                 if (context.mounted) hideLoadingModal(context); | ||||
|                               } | ||||
|                             }, | ||||
|                           ), | ||||
|                           Gap(MediaQuery.of(context).padding.bottom), | ||||
|                         ], | ||||
|                       ), | ||||
|                 ), | ||||
|             backgroundColor: const Color(0xFFE53E3E), | ||||
|             iconColor: Colors.white, | ||||
|           ), | ||||
| @@ -212,24 +266,14 @@ class CallControlsBar extends HookConsumerWidget { | ||||
|       } | ||||
|  | ||||
|       if (context.mounted) { | ||||
|         ScaffoldMessenger.of(context).showSnackBar( | ||||
|           SnackBar( | ||||
|             content: Text( | ||||
|               '${'switchedTo'.tr()} ${device.label.isNotEmpty ? device.label : 'selectedDevice'.tr()}', | ||||
|             ), | ||||
|             backgroundColor: Colors.green, | ||||
|           ), | ||||
|         ); | ||||
|       } | ||||
|     } catch (e) { | ||||
|       if (context.mounted) { | ||||
|         ScaffoldMessenger.of(context).showSnackBar( | ||||
|           SnackBar( | ||||
|             content: Text('${'failedToSwitchDevice'.tr()}: $e'), | ||||
|             backgroundColor: Colors.red, | ||||
|         showSnackBar( | ||||
|           'switchedTo'.tr( | ||||
|             args: [device.label.isNotEmpty ? device.label : 'device'], | ||||
|           ), | ||||
|         ); | ||||
|       } | ||||
|     } catch (err) { | ||||
|       showErrorAlert(err); | ||||
|     } | ||||
|   } | ||||
| } | ||||
| @@ -279,7 +323,7 @@ class CallOverlayBar extends HookConsumerWidget { | ||||
|       child: Card( | ||||
|         margin: EdgeInsets.zero, | ||||
|         child: Row( | ||||
|           crossAxisAlignment: CrossAxisAlignment.start, | ||||
|           crossAxisAlignment: CrossAxisAlignment.center, | ||||
|           children: [ | ||||
|             Expanded( | ||||
|               child: Row( | ||||
| @@ -294,17 +338,7 @@ class CallOverlayBar extends HookConsumerWidget { | ||||
|                         height: 40, | ||||
|                         child: | ||||
|                             SpeakingRippleAvatar( | ||||
|                               isSpeaking: lastSpeaker.isSpeaking, | ||||
|                               audioLevel: | ||||
|                                   lastSpeaker.remoteParticipant.audioLevel, | ||||
|                               pictureId: | ||||
|                                   lastSpeaker | ||||
|                                       .participant | ||||
|                                       .profile | ||||
|                                       ?.account | ||||
|                                       .profile | ||||
|                                       .picture | ||||
|                                       ?.id, | ||||
|                               live: lastSpeaker, | ||||
|                               size: 36, | ||||
|                             ).center(), | ||||
|                       ); | ||||
| @@ -314,10 +348,7 @@ class CallOverlayBar extends HookConsumerWidget { | ||||
|                   Column( | ||||
|                     crossAxisAlignment: CrossAxisAlignment.start, | ||||
|                     children: [ | ||||
|                       Text( | ||||
|                         lastSpeaker.participant.profile?.account.nick ?? | ||||
|                             'unknown'.tr(), | ||||
|                       ).bold(), | ||||
|                       Text('@${lastSpeaker.participant.identity}').bold(), | ||||
|                       Text( | ||||
|                         formatDuration(callState.duration), | ||||
|                         style: Theme.of(context).textTheme.bodySmall, | ||||
| @@ -360,7 +391,10 @@ class CallOverlayBar extends HookConsumerWidget { | ||||
|         ).padding(all: 16), | ||||
|       ), | ||||
|       onTap: () { | ||||
|         context.pushNamed('chatCall', pathParameters: {'id': callNotifier.roomId!}); | ||||
|         context.pushNamed( | ||||
|           'chatCall', | ||||
|           pathParameters: {'id': callNotifier.roomId!}, | ||||
|         ); | ||||
|       }, | ||||
|     ); | ||||
|   } | ||||
|   | ||||
							
								
								
									
										127
									
								
								lib/widgets/chat/call_participant_card.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										127
									
								
								lib/widgets/chat/call_participant_card.dart
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,127 @@ | ||||
| import 'dart:math' as math; | ||||
|  | ||||
| import 'package:flutter/material.dart'; | ||||
| import 'package:flutter_hooks/flutter_hooks.dart'; | ||||
| import 'package:flutter_popup_card/flutter_popup_card.dart'; | ||||
| import 'package:gap/gap.dart'; | ||||
| import 'package:hooks_riverpod/hooks_riverpod.dart'; | ||||
| import 'package:island/pods/call.dart'; | ||||
| import 'package:island/widgets/account/account_nameplate.dart'; | ||||
| import 'package:livekit_client/livekit_client.dart'; | ||||
| import 'package:material_symbols_icons/material_symbols_icons.dart'; | ||||
| import 'package:styled_widget/styled_widget.dart'; | ||||
|  | ||||
| class CallParticipantCard extends HookConsumerWidget { | ||||
|   final CallParticipantLive live; | ||||
|   const CallParticipantCard({super.key, required this.live}); | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context, WidgetRef ref) { | ||||
|     final width = | ||||
|         math.min(MediaQuery.of(context).size.width - 80, 360).toDouble(); | ||||
|     final callNotifier = ref.watch(callNotifierProvider.notifier); | ||||
|  | ||||
|     final volumeSliderValue = useState(callNotifier.getParticipantVolume(live)); | ||||
|  | ||||
|     return PopupCard( | ||||
|       elevation: 8, | ||||
|       shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(12.0)), | ||||
|       child: SizedBox( | ||||
|         width: width, | ||||
|         child: Column( | ||||
|           mainAxisSize: MainAxisSize.min, | ||||
|           crossAxisAlignment: CrossAxisAlignment.stretch, | ||||
|           children: [ | ||||
|             Column( | ||||
|               spacing: 4, | ||||
|               children: [ | ||||
|                 Row( | ||||
|                   children: [ | ||||
|                     const Icon(Symbols.sound_detection_loud_sound, size: 16), | ||||
|                     const Gap(8), | ||||
|                     Expanded( | ||||
|                       child: Slider( | ||||
|                         max: 2, | ||||
|                         value: volumeSliderValue.value, | ||||
|                         onChanged: (value) { | ||||
|                           volumeSliderValue.value = value; | ||||
|                         }, | ||||
|                         onChangeEnd: (value) { | ||||
|                           callNotifier.setParticipantVolume(live, value); | ||||
|                         }, | ||||
|                         year2023: true, | ||||
|                         padding: EdgeInsets.zero, | ||||
|                       ), | ||||
|                     ), | ||||
|                     const Gap(16), | ||||
|                     SizedBox( | ||||
|                       width: 40, | ||||
|                       child: Text( | ||||
|                         '${(volumeSliderValue.value * 100).toStringAsFixed(0)}%', | ||||
|                       ), | ||||
|                     ), | ||||
|                   ], | ||||
|                 ), | ||||
|                 Row( | ||||
|                   children: [ | ||||
|                     const Icon(Symbols.wifi, size: 16), | ||||
|                     const Gap(8), | ||||
|                     Text(switch (live.remoteParticipant.connectionQuality) { | ||||
|                       ConnectionQuality.excellent => 'Excellent', | ||||
|                       ConnectionQuality.good => 'Good', | ||||
|                       ConnectionQuality.poor => 'Bad', | ||||
|                       ConnectionQuality.lost => 'Lost', | ||||
|                       _ => 'Connecting', | ||||
|                     }), | ||||
|                   ], | ||||
|                 ), | ||||
|               ], | ||||
|             ).padding(horizontal: 20, top: 16), | ||||
|             AccountNameplate( | ||||
|               name: live.participant.identity, | ||||
|               isOutlined: false, | ||||
|             ), | ||||
|           ], | ||||
|         ), | ||||
|       ), | ||||
|     ); | ||||
|   } | ||||
| } | ||||
|  | ||||
| class CallParticipantGestureDetector extends StatelessWidget { | ||||
|   final CallParticipantLive participant; | ||||
|   final Widget child; | ||||
|   const CallParticipantGestureDetector({ | ||||
|     super.key, | ||||
|     required this.participant, | ||||
|     required this.child, | ||||
|   }); | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     return GestureDetector( | ||||
|       child: child, | ||||
|       onTapDown: (details) { | ||||
|         showCallParticipantCard( | ||||
|           context, | ||||
|           participant, | ||||
|           offset: details.localPosition, | ||||
|         ); | ||||
|       }, | ||||
|     ); | ||||
|   } | ||||
| } | ||||
|  | ||||
| Future<void> showCallParticipantCard( | ||||
|   BuildContext context, | ||||
|   CallParticipantLive participant, { | ||||
|   Offset? offset, | ||||
| }) async { | ||||
|   await showPopupCard<void>( | ||||
|     offset: offset ?? Offset.zero, | ||||
|     context: context, | ||||
|     builder: (context) => CallParticipantCard(live: participant), | ||||
|     alignment: Alignment.center, | ||||
|     dimBackground: true, | ||||
|   ); | ||||
| } | ||||
| @@ -1,92 +1,127 @@ | ||||
| import 'package:easy_localization/easy_localization.dart'; | ||||
| import 'package:flutter/material.dart'; | ||||
| import 'package:hooks_riverpod/hooks_riverpod.dart'; | ||||
| import 'package:island/pods/call.dart'; | ||||
| import 'package:island/screens/account/profile.dart'; | ||||
| import 'package:island/widgets/chat/call_participant_card.dart'; | ||||
| import 'package:island/widgets/content/cloud_files.dart'; | ||||
| import 'package:livekit_client/livekit_client.dart'; | ||||
| import 'package:material_symbols_icons/symbols.dart'; | ||||
| import 'package:styled_widget/styled_widget.dart'; | ||||
|  | ||||
| class SpeakingRippleAvatar extends StatelessWidget { | ||||
|   final bool isSpeaking; | ||||
|   final double audioLevel; | ||||
|   final String? pictureId; | ||||
| class SpeakingRippleAvatar extends HookConsumerWidget { | ||||
|   final CallParticipantLive live; | ||||
|   final double size; | ||||
|  | ||||
|   const SpeakingRippleAvatar({ | ||||
|     super.key, | ||||
|     required this.isSpeaking, | ||||
|     required this.audioLevel, | ||||
|     required this.pictureId, | ||||
|     this.size = 96, | ||||
|   }); | ||||
|   const SpeakingRippleAvatar({super.key, required this.live, this.size = 96}); | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|   Widget build(BuildContext context, WidgetRef ref) { | ||||
|     final account = ref.watch(accountProvider(live.participant.identity)); | ||||
|  | ||||
|     final avatarRadius = size / 2; | ||||
|     final clampedLevel = audioLevel.clamp(0.0, 1.0); | ||||
|     final clampedLevel = live.remoteParticipant.audioLevel.clamp(0.0, 1.0); | ||||
|     final rippleRadius = avatarRadius + clampedLevel * (size * 0.333); | ||||
|     return TweenAnimationBuilder<double>( | ||||
|       tween: Tween<double>( | ||||
|         begin: avatarRadius, | ||||
|         end: isSpeaking ? rippleRadius : avatarRadius, | ||||
|       ), | ||||
|       duration: const Duration(milliseconds: 250), | ||||
|       curve: Curves.easeOut, | ||||
|       builder: (context, animatedRadius, child) { | ||||
|         return Stack( | ||||
|           alignment: Alignment.center, | ||||
|           children: [ | ||||
|             if (isSpeaking) | ||||
|     return SizedBox( | ||||
|       width: size + 8, | ||||
|       height: size + 8, | ||||
|       child: TweenAnimationBuilder<double>( | ||||
|         tween: Tween<double>( | ||||
|           begin: avatarRadius, | ||||
|           end: live.remoteParticipant.isSpeaking ? rippleRadius : avatarRadius, | ||||
|         ), | ||||
|         duration: const Duration(milliseconds: 250), | ||||
|         curve: Curves.easeOut, | ||||
|         builder: (context, animatedRadius, child) { | ||||
|           return Stack( | ||||
|             alignment: Alignment.center, | ||||
|             children: [ | ||||
|               if (live.remoteParticipant.isSpeaking) | ||||
|                 Container( | ||||
|                   width: animatedRadius * 2, | ||||
|                   height: animatedRadius * 2, | ||||
|                   decoration: BoxDecoration( | ||||
|                     shape: BoxShape.circle, | ||||
|                     color: Colors.green.withOpacity(0.75 + 0.25 * clampedLevel), | ||||
|                   ), | ||||
|                 ), | ||||
|               Container( | ||||
|                 width: animatedRadius * 2, | ||||
|                 height: animatedRadius * 2, | ||||
|                 decoration: BoxDecoration( | ||||
|                   shape: BoxShape.circle, | ||||
|                   color: Colors.green.withOpacity(0.75 + 0.25 * clampedLevel), | ||||
|                 width: size, | ||||
|                 height: size, | ||||
|                 alignment: Alignment.center, | ||||
|                 decoration: BoxDecoration(shape: BoxShape.circle), | ||||
|                 child: account.when( | ||||
|                   data: | ||||
|                       (value) => CallParticipantGestureDetector( | ||||
|                         participant: live, | ||||
|                         child: ProfilePictureWidget( | ||||
|                           file: value.profile.picture, | ||||
|                           radius: size / 2, | ||||
|                         ), | ||||
|                       ), | ||||
|                   error: | ||||
|                       (_, _) => CircleAvatar( | ||||
|                         radius: size / 2, | ||||
|                         child: const Icon(Symbols.person_remove), | ||||
|                       ), | ||||
|                   loading: | ||||
|                       () => CircleAvatar( | ||||
|                         radius: size / 2, | ||||
|                         child: CircularProgressIndicator(), | ||||
|                       ), | ||||
|                 ), | ||||
|               ), | ||||
|             Container( | ||||
|               width: size, | ||||
|               height: size, | ||||
|               alignment: Alignment.center, | ||||
|               decoration: BoxDecoration(shape: BoxShape.circle), | ||||
|               child: ProfilePictureWidget(fileId: pictureId, radius: size / 2), | ||||
|             ), | ||||
|           ], | ||||
|         ); | ||||
|       }, | ||||
|               if (live.remoteParticipant.isMuted) | ||||
|                 Positioned( | ||||
|                   bottom: 4, | ||||
|                   right: 4, | ||||
|                   child: Container( | ||||
|                     width: 20, | ||||
|                     height: 20, | ||||
|                     decoration: BoxDecoration( | ||||
|                       color: Colors.red, | ||||
|                       borderRadius: BorderRadius.all(Radius.circular(10)), | ||||
|                     ), | ||||
|                     child: const Icon( | ||||
|                       Symbols.mic_off, | ||||
|                       size: 14, | ||||
|                       fill: 1, | ||||
|                     ).padding(left: 1.5, top: 1.5), | ||||
|                   ), | ||||
|                 ), | ||||
|             ], | ||||
|           ); | ||||
|         }, | ||||
|       ), | ||||
|     ); | ||||
|   } | ||||
| } | ||||
|  | ||||
| class CallParticipantTile extends StatelessWidget { | ||||
| class CallParticipantTile extends HookConsumerWidget { | ||||
|   final CallParticipantLive live; | ||||
|  | ||||
|   const CallParticipantTile({super.key, required this.live}); | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|   Widget build(BuildContext context, WidgetRef ref) { | ||||
|     final hasVideo = | ||||
|         live.hasVideo && | ||||
|         live.remoteParticipant.trackPublications.values | ||||
|             .where((pub) => pub.track != null && pub.kind == TrackType.VIDEO) | ||||
|             .isNotEmpty; | ||||
|     final audioLevel = live.remoteParticipant.audioLevel; | ||||
|  | ||||
|     if (hasVideo) { | ||||
|       return Stack( | ||||
|         fit: StackFit.loose, | ||||
|         children: [ | ||||
|           Container( | ||||
|             color: Theme.of(context).colorScheme.surfaceContainerHigh, | ||||
|             child: AspectRatio( | ||||
|               aspectRatio: 16 / 9, | ||||
|               child: VideoTrackRenderer( | ||||
|                 live.remoteParticipant.trackPublications.values | ||||
|                         .where((track) => track.kind == TrackType.VIDEO) | ||||
|                         .first | ||||
|                         .track | ||||
|                     as VideoTrack, | ||||
|                 renderMode: VideoRenderMode.platformView, | ||||
|               ), | ||||
|           AspectRatio( | ||||
|             aspectRatio: 16 / 9, | ||||
|             child: VideoTrackRenderer( | ||||
|               live.remoteParticipant.trackPublications.values | ||||
|                       .where((track) => track.kind == TrackType.VIDEO) | ||||
|                       .first | ||||
|                       .track | ||||
|                   as VideoTrack, | ||||
|               renderMode: VideoRenderMode.platformView, | ||||
|             ), | ||||
|           ), | ||||
|           Positioned( | ||||
| @@ -94,21 +129,26 @@ class CallParticipantTile extends StatelessWidget { | ||||
|             right: 8, | ||||
|             bottom: 8, | ||||
|             child: Text( | ||||
|               live.participant.profile?.account.nick ?? | ||||
|                   '${'unknown'.tr()}\'s video', | ||||
|               '@${live.participant.name}', | ||||
|               textAlign: TextAlign.center, | ||||
|               style: const TextStyle(fontSize: 14, color: Colors.white), | ||||
|               style: const TextStyle( | ||||
|                 fontSize: 14, | ||||
|                 color: Colors.white, | ||||
|                 shadows: [ | ||||
|                   BoxShadow( | ||||
|                     color: Colors.black54, | ||||
|                     offset: Offset(1, 1), | ||||
|                     spreadRadius: 8, | ||||
|                     blurRadius: 8, | ||||
|                   ), | ||||
|                 ], | ||||
|               ), | ||||
|             ), | ||||
|           ), | ||||
|         ], | ||||
|       ); | ||||
|     } else { | ||||
|       return SpeakingRippleAvatar( | ||||
|         isSpeaking: live.isSpeaking, | ||||
|         audioLevel: audioLevel, | ||||
|         pictureId: live.participant.profile?.account.profile.picture?.id, | ||||
|         size: 84, | ||||
|       ); | ||||
|       return SpeakingRippleAvatar(size: 84, live: live); | ||||
|     } | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -5,15 +5,20 @@ import 'package:easy_localization/easy_localization.dart'; | ||||
| import 'package:flutter/foundation.dart'; | ||||
| import 'package:flutter/material.dart'; | ||||
| import 'package:flutter/services.dart'; | ||||
| import 'package:flutter_hooks/flutter_hooks.dart'; | ||||
| import 'package:gap/gap.dart'; | ||||
| import 'package:hooks_riverpod/hooks_riverpod.dart'; | ||||
| import 'package:island/database/message.dart'; | ||||
| import 'package:island/models/chat.dart'; | ||||
| import 'package:island/models/embed.dart'; | ||||
| import 'package:island/pods/call.dart'; | ||||
| import 'package:island/pods/translate.dart'; | ||||
| import 'package:island/screens/chat/room.dart'; | ||||
| import 'package:island/utils/mapping.dart'; | ||||
| import 'package:island/widgets/account/account_name.dart'; | ||||
| import 'package:island/widgets/account/account_pfc.dart'; | ||||
| import 'package:island/widgets/app_scaffold.dart'; | ||||
| import 'package:island/widgets/content/alert.native.dart'; | ||||
| import 'package:island/widgets/content/cloud_file_collection.dart'; | ||||
| import 'package:island/widgets/content/cloud_files.dart'; | ||||
| import 'package:island/widgets/content/embed/link.dart'; | ||||
| @@ -66,6 +71,46 @@ class MessageItem extends HookConsumerWidget { | ||||
|  | ||||
|     final isMobile = !kIsWeb && (Platform.isAndroid || Platform.isIOS); | ||||
|  | ||||
|     final messageLanguage = | ||||
|         remoteMessage.content != null | ||||
|             ? ref.watch(detectStringLanguageProvider(remoteMessage.content!)) | ||||
|             : null; | ||||
|  | ||||
|     final currentLanguage = context.locale.toString(); | ||||
|     final translatableLanguage = | ||||
|         messageLanguage != null | ||||
|             ? messageLanguage.substring(0, 2) != currentLanguage.substring(0, 2) | ||||
|             : false; | ||||
|  | ||||
|     final translating = useState(false); | ||||
|     final translatedText = useState<String?>(null); | ||||
|  | ||||
|     Future<void> translate() async { | ||||
|       if (translatedText.value != null) { | ||||
|         translatedText.value = null; | ||||
|         return; | ||||
|       } | ||||
|  | ||||
|       if (translating.value) return; | ||||
|       if (remoteMessage.content == null) return; | ||||
|       translating.value = true; | ||||
|       try { | ||||
|         final text = await ref.watch( | ||||
|           translateStringProvider( | ||||
|             TranslateQuery( | ||||
|               text: remoteMessage.content!, | ||||
|               lang: currentLanguage.substring(0, 2), | ||||
|             ), | ||||
|           ).future, | ||||
|         ); | ||||
|         translatedText.value = text; | ||||
|       } catch (err) { | ||||
|         showErrorAlert(err); | ||||
|       } finally { | ||||
|         translating.value = false; | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     return ContextMenuWidget( | ||||
|       menuProvider: (_) { | ||||
|         if (onAction == null) return Menu(children: []); | ||||
| @@ -102,6 +147,18 @@ class MessageItem extends HookConsumerWidget { | ||||
|                 onAction!.call(MessageItemAction.forward); | ||||
|               }, | ||||
|             ), | ||||
|             if (translatableLanguage) MenuSeparator(), | ||||
|             if (translatableLanguage) | ||||
|               MenuAction( | ||||
|                 title: | ||||
|                     translatedText.value == null | ||||
|                         ? 'translate'.tr() | ||||
|                         : translating.value | ||||
|                         ? 'translating'.tr() | ||||
|                         : 'translated'.tr(), | ||||
|                 image: MenuImage.icon(Symbols.translate), | ||||
|                 callback: translate, | ||||
|               ), | ||||
|             if (isMobile) MenuSeparator(), | ||||
|             if (isMobile) | ||||
|               MenuAction( | ||||
| @@ -166,8 +223,8 @@ class MessageItem extends HookConsumerWidget { | ||||
|                           mainAxisSize: MainAxisSize.min, | ||||
|                           spacing: 5, | ||||
|                           children: [ | ||||
|                             Text( | ||||
|                               sender.account.nick, | ||||
|                             AccountName( | ||||
|                               account: sender.account, | ||||
|                               style: Theme.of(context).textTheme.bodySmall, | ||||
|                             ), | ||||
|                             Badge( | ||||
| @@ -220,24 +277,27 @@ class MessageItem extends HookConsumerWidget { | ||||
|                               isReply: false, | ||||
|                             ).padding(vertical: 4), | ||||
|                           if (_MessageItemContent.hasContent(remoteMessage)) | ||||
|                             _MessageItemContent(item: remoteMessage), | ||||
|                             _MessageItemContent( | ||||
|                               item: remoteMessage, | ||||
|                               translatedText: translatedText.value, | ||||
|                             ), | ||||
|                           if (remoteMessage.attachments.isNotEmpty) | ||||
|                             LayoutBuilder( | ||||
|                               builder: (context, constraints) { | ||||
|                                 return CloudFileList( | ||||
|                                   files: remoteMessage.attachments, | ||||
|                                   maxWidth: constraints.maxWidth, | ||||
|                                 ).padding(vertical: 4); | ||||
|                                   padding: EdgeInsets.symmetric(vertical: 4), | ||||
|                                 ); | ||||
|                               }, | ||||
|                             ), | ||||
|                           if (remoteMessage.meta['embeds'] != null) | ||||
|                             ...((remoteMessage.meta['embeds'] as List<dynamic>) | ||||
|                                 .where((embed) => embed['Type'] == 'link') | ||||
|                                 .map( | ||||
|                                   (embed) => SnEmbedLink.fromJson( | ||||
|                                     embed as Map<String, dynamic>, | ||||
|                                   ), | ||||
|                                   (embed) => convertMapKeysToSnakeCase(embed), | ||||
|                                 ) | ||||
|                                 .where((embed) => embed['type'] == 'link') | ||||
|                                 .map((embed) => SnScrappedLink.fromJson(embed)) | ||||
|                                 .map( | ||||
|                                   (link) => LayoutBuilder( | ||||
|                                     builder: (context, constraints) { | ||||
| @@ -480,7 +540,8 @@ class MessageQuoteWidget extends HookConsumerWidget { | ||||
|  | ||||
| class _MessageItemContent extends StatelessWidget { | ||||
|   final SnChatMessage item; | ||||
|   const _MessageItemContent({required this.item}); | ||||
|   final String? translatedText; | ||||
|   const _MessageItemContent({required this.item, this.translatedText}); | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
| @@ -493,10 +554,40 @@ class _MessageItemContent extends StatelessWidget { | ||||
|         ); | ||||
|       case 'text': | ||||
|       default: | ||||
|         return MarkdownTextContent( | ||||
|           content: item.content!, | ||||
|           isSelectable: true, | ||||
|           linesMargin: EdgeInsets.zero, | ||||
|         return Column( | ||||
|           mainAxisSize: MainAxisSize.min, | ||||
|           crossAxisAlignment: CrossAxisAlignment.start, | ||||
|           children: [ | ||||
|             MarkdownTextContent( | ||||
|               content: item.content!, | ||||
|               isSelectable: true, | ||||
|               linesMargin: EdgeInsets.zero, | ||||
|             ), | ||||
|             if (translatedText?.isNotEmpty ?? false) | ||||
|               ...([ | ||||
|                 ConstrainedBox( | ||||
|                   constraints: BoxConstraints( | ||||
|                     maxWidth: math.min( | ||||
|                       280, | ||||
|                       MediaQuery.of(context).size.width * 0.4, | ||||
|                     ), | ||||
|                   ), | ||||
|                   child: Row( | ||||
|                     mainAxisSize: MainAxisSize.min, | ||||
|                     children: [ | ||||
|                       Text('translated').tr().fontSize(11).opacity(0.75), | ||||
|                       const Gap(8), | ||||
|                       Flexible(child: Divider()), | ||||
|                     ], | ||||
|                   ).padding(vertical: 4), | ||||
|                 ), | ||||
|                 MarkdownTextContent( | ||||
|                   content: translatedText!, | ||||
|                   isSelectable: true, | ||||
|                   linesMargin: EdgeInsets.zero, | ||||
|                 ), | ||||
|               ]), | ||||
|           ], | ||||
|         ); | ||||
|     } | ||||
|   } | ||||
|   | ||||
| @@ -35,7 +35,9 @@ Future<SnCheckInResult?> checkInResultToday(Ref ref) async { | ||||
| } | ||||
|  | ||||
| class CheckInWidget extends HookConsumerWidget { | ||||
|   const CheckInWidget({super.key}); | ||||
|   final EdgeInsets? margin; | ||||
|   final VoidCallback? onChecked; | ||||
|   const CheckInWidget({super.key, this.margin, this.onChecked}); | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context, WidgetRef ref) { | ||||
| @@ -51,6 +53,7 @@ class CheckInWidget extends HookConsumerWidget { | ||||
|         ref.invalidate(checkInResultTodayProvider); | ||||
|         final userNotifier = ref.read(userInfoProvider.notifier); | ||||
|         userNotifier.fetchUser(); | ||||
|         onChecked?.call(); | ||||
|       } catch (err) { | ||||
|         if (err is DioException) { | ||||
|           if (err.response?.statusCode == 423 && context.mounted) { | ||||
| @@ -66,7 +69,8 @@ class CheckInWidget extends HookConsumerWidget { | ||||
|     } | ||||
|  | ||||
|     return Card( | ||||
|       margin: EdgeInsets.only(left: 16, right: 16, top: 16, bottom: 8), | ||||
|       margin: | ||||
|           margin ?? EdgeInsets.only(left: 16, right: 16, top: 16, bottom: 8), | ||||
|       child: Row( | ||||
|         crossAxisAlignment: CrossAxisAlignment.center, | ||||
|         spacing: 16, | ||||
|   | ||||
| @@ -1,3 +1,4 @@ | ||||
| import 'dart:convert'; | ||||
| import 'dart:io'; | ||||
|  | ||||
| import 'package:cross_file/cross_file.dart'; | ||||
| @@ -5,18 +6,89 @@ import 'package:easy_localization/easy_localization.dart'; | ||||
| import 'package:flutter/foundation.dart'; | ||||
| import 'package:flutter/material.dart'; | ||||
| import 'package:gap/gap.dart'; | ||||
| import 'package:hooks_riverpod/hooks_riverpod.dart'; | ||||
| import 'package:island/models/file.dart'; | ||||
| import 'package:island/pods/network.dart'; | ||||
| import 'package:island/services/file.dart'; | ||||
| import 'package:island/widgets/alert.dart'; | ||||
| import 'package:island/widgets/content/cloud_files.dart'; | ||||
| import 'package:island/widgets/content/sheet.dart'; | ||||
| import 'package:material_symbols_icons/symbols.dart'; | ||||
| import 'package:styled_widget/styled_widget.dart'; | ||||
| import 'package:super_context_menu/super_context_menu.dart'; | ||||
|  | ||||
| class AttachmentPreview extends StatelessWidget { | ||||
| import 'sensitive.dart'; | ||||
|  | ||||
| class SensitiveMarksSelector extends StatefulWidget { | ||||
|   final List<int> initial; | ||||
|   final ValueChanged<List<int>>? onChanged; | ||||
|  | ||||
|   const SensitiveMarksSelector({ | ||||
|     super.key, | ||||
|     required this.initial, | ||||
|     this.onChanged, | ||||
|   }); | ||||
|  | ||||
|   @override | ||||
|   State<SensitiveMarksSelector> createState() => SensitiveMarksSelectorState(); | ||||
| } | ||||
|  | ||||
| class SensitiveMarksSelectorState extends State<SensitiveMarksSelector> { | ||||
|   late List<int> _selected; | ||||
|  | ||||
|   List<int> get current => _selected; | ||||
|  | ||||
|   @override | ||||
|   void initState() { | ||||
|     super.initState(); | ||||
|     _selected = [...widget.initial]; | ||||
|   } | ||||
|  | ||||
|   void _toggle(int value) { | ||||
|     setState(() { | ||||
|       if (_selected.contains(value)) { | ||||
|         _selected.remove(value); | ||||
|       } else { | ||||
|         _selected.add(value); | ||||
|       } | ||||
|     }); | ||||
|     widget.onChanged?.call([..._selected]); | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     // Build a list of all categories in fixed order as int list indices | ||||
|     final categories = kSensitiveCategoriesOrdered; | ||||
|  | ||||
|     return Column( | ||||
|       crossAxisAlignment: CrossAxisAlignment.stretch, | ||||
|       children: [ | ||||
|         Wrap( | ||||
|           spacing: 8, | ||||
|           children: [ | ||||
|             for (var i = 0; i < categories.length; i++) | ||||
|               FilterChip( | ||||
|                 label: Text(categories[i].i18nKey.tr()), | ||||
|                 avatar: Text(categories[i].symbol), | ||||
|                 selected: _selected.contains(i), | ||||
|                 onSelected: (_) => _toggle(i), | ||||
|               ), | ||||
|           ], | ||||
|         ), | ||||
|       ], | ||||
|     ); | ||||
|   } | ||||
| } | ||||
|  | ||||
| class AttachmentPreview extends HookConsumerWidget { | ||||
|   final UniversalFile item; | ||||
|   final double? progress; | ||||
|   final Function(int)? onMove; | ||||
|   final Function? onDelete; | ||||
|   final Function? onInsert; | ||||
|   final Function(UniversalFile)? onUpdate; | ||||
|   final Function? onRequestUpload; | ||||
|  | ||||
|   const AttachmentPreview({ | ||||
|     super.key, | ||||
|     required this.item, | ||||
| @@ -24,226 +96,437 @@ class AttachmentPreview extends StatelessWidget { | ||||
|     this.onRequestUpload, | ||||
|     this.onMove, | ||||
|     this.onDelete, | ||||
|     this.onUpdate, | ||||
|     this.onInsert, | ||||
|   }); | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     var ratio = | ||||
|         (item.isOnCloud ? (item.data.fileMeta?['ratio'] ?? 1) : 1).toDouble(); | ||||
|     if (ratio == 0) ratio = 1.0; | ||||
|   // GlobalKey for selector | ||||
|   static final GlobalKey<SensitiveMarksSelectorState> _sensitiveSelectorKey = | ||||
|       GlobalKey<SensitiveMarksSelectorState>(); | ||||
|  | ||||
|     return AspectRatio( | ||||
|       aspectRatio: ratio, | ||||
|       child: ClipRRect( | ||||
|         borderRadius: BorderRadius.circular(8), | ||||
|         child: Stack( | ||||
|           fit: StackFit.expand, | ||||
|           children: [ | ||||
|             Container( | ||||
|               color: Theme.of(context).colorScheme.surfaceContainerHigh, | ||||
|               child: Builder( | ||||
|                 builder: (context) { | ||||
|                   if (item.isOnCloud) { | ||||
|                     return CloudFileWidget(item: item.data); | ||||
|                   } else if (item.data is XFile) { | ||||
|                     if (item.type == UniversalFileType.image) { | ||||
|                       final file = item.data as XFile; | ||||
|                       if (file.path.isEmpty) { | ||||
|                         return FutureBuilder<Uint8List>( | ||||
|                           future: file.readAsBytes(), | ||||
|                           builder: (context, snapshot) { | ||||
|                             if (snapshot.hasData) { | ||||
|                               return Image.memory(snapshot.data!); | ||||
|                             } | ||||
|                             return const Center( | ||||
|                               child: CircularProgressIndicator(), | ||||
|                             ); | ||||
|                           }, | ||||
|                         ); | ||||
|                       } | ||||
|                       return kIsWeb | ||||
|                           ? Image.network(file.path) | ||||
|                           : Image.file(File(file.path)); | ||||
|                     } else { | ||||
|                       return Center( | ||||
|                         child: Text( | ||||
|                           'Preview is not supported for ${item.type}', | ||||
|                           textAlign: TextAlign.center, | ||||
|                         ), | ||||
|                       ); | ||||
|                     } | ||||
|                   } else if (item is List<int> || item is Uint8List) { | ||||
|                     if (item.type == UniversalFileType.image) { | ||||
|                       return Image.memory(item.data); | ||||
|                     } else { | ||||
|                       return Center( | ||||
|                         child: Text( | ||||
|                           'Preview is not supported for ${item.type}', | ||||
|                           textAlign: TextAlign.center, | ||||
|                         ), | ||||
|                       ); | ||||
|                     } | ||||
|                   } | ||||
|                   return Placeholder(); | ||||
|                 }, | ||||
|               ), | ||||
|   Future<void> _showRenameDialog(BuildContext context, WidgetRef ref) async { | ||||
|     final nameController = TextEditingController(text: item.data.name); | ||||
|     String? errorMessage; | ||||
|  | ||||
|     await showModalBottomSheet( | ||||
|       context: context, | ||||
|       isScrollControlled: true, | ||||
|       builder: | ||||
|           (context) => SheetScaffold( | ||||
|             heightFactor: 0.6, | ||||
|             titleText: 'rename'.tr(), | ||||
|             child: Column( | ||||
|               mainAxisSize: MainAxisSize.min, | ||||
|               crossAxisAlignment: CrossAxisAlignment.stretch, | ||||
|               children: [ | ||||
|                 Padding( | ||||
|                   padding: const EdgeInsets.symmetric( | ||||
|                     horizontal: 24, | ||||
|                     vertical: 24, | ||||
|                   ), | ||||
|                   child: TextField( | ||||
|                     controller: nameController, | ||||
|                     decoration: InputDecoration( | ||||
|                       labelText: 'fileName'.tr(), | ||||
|                       border: const OutlineInputBorder(), | ||||
|                       errorText: errorMessage, | ||||
|                     ), | ||||
|                   ), | ||||
|                 ), | ||||
|                 Row( | ||||
|                   mainAxisAlignment: MainAxisAlignment.end, | ||||
|                   children: [ | ||||
|                     TextButton( | ||||
|                       onPressed: () => Navigator.pop(context), | ||||
|                       child: Text('cancel'.tr()), | ||||
|                     ), | ||||
|                     const Gap(8), | ||||
|                     TextButton( | ||||
|                       onPressed: () async { | ||||
|                         final newName = nameController.text.trim(); | ||||
|                         if (newName.isEmpty) { | ||||
|                           errorMessage = 'fieldCannotBeEmpty'.tr(); | ||||
|                           return; | ||||
|                         } | ||||
|  | ||||
|                         try { | ||||
|                           showLoadingModal(context); | ||||
|                           final apiClient = ref.watch(apiClientProvider); | ||||
|                           await apiClient.patch( | ||||
|                             '/drive/files/${item.data.id}/name', | ||||
|                             data: jsonEncode(newName), | ||||
|                           ); | ||||
|                           final newData = item.data; | ||||
|                           newData.name = newName; | ||||
|                           final updatedFile = item.copyWith(data: newData); | ||||
|                           onUpdate?.call(item.copyWith(data: updatedFile)); | ||||
|                           if (context.mounted) Navigator.pop(context); | ||||
|                         } catch (err) { | ||||
|                           showErrorAlert(err); | ||||
|                         } finally { | ||||
|                           if (context.mounted) hideLoadingModal(context); | ||||
|                         } | ||||
|                       }, | ||||
|                       child: Text('rename'.tr()), | ||||
|                     ), | ||||
|                   ], | ||||
|                 ).padding(horizontal: 16, vertical: 8), | ||||
|               ], | ||||
|             ), | ||||
|             if (progress != null) | ||||
|               Positioned.fill( | ||||
|                 child: Container( | ||||
|                   color: Colors.black.withOpacity(0.3), | ||||
|                   padding: EdgeInsets.symmetric(horizontal: 40, vertical: 16), | ||||
|           ), | ||||
|     ); | ||||
|   } | ||||
|  | ||||
|   Future<void> _showSensitiveDialog(BuildContext context, WidgetRef ref) async { | ||||
|     await showModalBottomSheet( | ||||
|       context: context, | ||||
|       isScrollControlled: true, | ||||
|       builder: | ||||
|           (context) => SheetScaffold( | ||||
|             heightFactor: 0.6, | ||||
|             titleText: 'markAsSensitive'.tr(), | ||||
|             child: Column( | ||||
|               mainAxisSize: MainAxisSize.min, | ||||
|               crossAxisAlignment: CrossAxisAlignment.stretch, | ||||
|               children: [ | ||||
|                 Padding( | ||||
|                   padding: const EdgeInsets.symmetric( | ||||
|                     horizontal: 24, | ||||
|                     vertical: 24, | ||||
|                   ), | ||||
|                   child: Column( | ||||
|                     mainAxisAlignment: MainAxisAlignment.center, | ||||
|                     crossAxisAlignment: CrossAxisAlignment.center, | ||||
|                     children: [ | ||||
|                       if (progress != null) | ||||
|                         Text( | ||||
|                           '${progress!.toStringAsFixed(2)}%', | ||||
|                           style: TextStyle(color: Colors.white), | ||||
|                         ) | ||||
|                       else | ||||
|                         Text( | ||||
|                           'uploading'.tr(), | ||||
|                           style: TextStyle(color: Colors.white), | ||||
|                         ), | ||||
|                       Gap(6), | ||||
|                       Center( | ||||
|                         child: LinearProgressIndicator( | ||||
|                           value: progress != null ? progress! / 100.0 : null, | ||||
|                         ), | ||||
|                       // Sensitive categories checklist | ||||
|                       SensitiveMarksSelector( | ||||
|                         key: _sensitiveSelectorKey, | ||||
|                         initial: | ||||
|                             (item.data.sensitiveMarks ?? []) | ||||
|                                 .map((e) => e as int) | ||||
|                                 .cast<int>() | ||||
|                                 .toList(), | ||||
|                         onChanged: (marks) { | ||||
|                           // Update local data immediately (optimistic) | ||||
|                           final newData = item.data; | ||||
|                           newData.sensitiveMarks = marks; | ||||
|                           final updatedFile = item.copyWith(data: newData); | ||||
|                           onUpdate?.call(item.copyWith(data: updatedFile)); | ||||
|                         }, | ||||
|                       ), | ||||
|                     ], | ||||
|                   ), | ||||
|                 ), | ||||
|               ), | ||||
|             Positioned( | ||||
|               left: 8, | ||||
|               top: 8, | ||||
|               child: ClipRRect( | ||||
|                 borderRadius: BorderRadius.circular(8), | ||||
|                 child: Container( | ||||
|                   color: Colors.black.withOpacity(0.5), | ||||
|                   child: Material( | ||||
|                     color: Colors.transparent, | ||||
|                     child: Row( | ||||
|                       mainAxisSize: MainAxisSize.min, | ||||
|                       children: [ | ||||
|                         if (onDelete != null) | ||||
|                           InkWell( | ||||
|                             borderRadius: BorderRadius.circular(8), | ||||
|                             child: const Icon( | ||||
|                               Symbols.delete, | ||||
|                               size: 14, | ||||
|                               color: Colors.white, | ||||
|                             ).padding(horizontal: 8, vertical: 6), | ||||
|                             onTap: () { | ||||
|                               onDelete?.call(); | ||||
|                             }, | ||||
|                           ), | ||||
|                         if (onDelete != null && onMove != null) | ||||
|                           SizedBox( | ||||
|                             height: 26, | ||||
|                             child: const VerticalDivider( | ||||
|                               width: 0.3, | ||||
|                               color: Colors.white, | ||||
|                               thickness: 0.3, | ||||
|                 Row( | ||||
|                   mainAxisAlignment: MainAxisAlignment.end, | ||||
|                   children: [ | ||||
|                     TextButton( | ||||
|                       onPressed: () => Navigator.pop(context), | ||||
|                       child: Text('cancel'.tr()), | ||||
|                     ), | ||||
|                     const Gap(8), | ||||
|                     TextButton( | ||||
|                       onPressed: () async { | ||||
|                         try { | ||||
|                           showLoadingModal(context); | ||||
|                           final apiClient = ref.watch(apiClientProvider); | ||||
|                           // Use the current selections from stateful selector via GlobalKey | ||||
|                           final selectorState = | ||||
|                               _sensitiveSelectorKey.currentState; | ||||
|                           final marks = selectorState?.current ?? <int>[]; | ||||
|                           await apiClient.put( | ||||
|                             '/drive/files/${item.data.id}/marks', | ||||
|                             data: jsonEncode({'sensitive_marks': marks}), | ||||
|                           ); | ||||
|                           final newData = item.data as SnCloudFile; | ||||
|                           final updatedFile = item.copyWith( | ||||
|                             data: newData.copyWith(sensitiveMarks: marks), | ||||
|                           ); | ||||
|                           onUpdate?.call(updatedFile); | ||||
|                           if (context.mounted) Navigator.pop(context); | ||||
|                         } catch (err) { | ||||
|                           showErrorAlert(err); | ||||
|                         } finally { | ||||
|                           if (context.mounted) hideLoadingModal(context); | ||||
|                         } | ||||
|                       }, | ||||
|                       child: Text('confirm'.tr()), | ||||
|                     ), | ||||
|                   ], | ||||
|                 ).padding(horizontal: 16, vertical: 8), | ||||
|               ], | ||||
|             ), | ||||
|           ), | ||||
|     ); | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context, WidgetRef ref) { | ||||
|     var ratio = | ||||
|         item.isOnCloud | ||||
|             ? (item.data.fileMeta?['ratio'] is num | ||||
|                 ? item.data.fileMeta!['ratio'].toDouble() | ||||
|                 : 1.0) | ||||
|             : 1.0; | ||||
|     if (ratio == 0) ratio = 1.0; | ||||
|  | ||||
|     final contentWidget = ClipRRect( | ||||
|       borderRadius: BorderRadius.circular(8), | ||||
|       child: Container( | ||||
|         color: Theme.of(context).colorScheme.surfaceContainer, | ||||
|         child: Column( | ||||
|           children: [ | ||||
|             Row( | ||||
|               mainAxisAlignment: MainAxisAlignment.spaceBetween, | ||||
|               children: [ | ||||
|                 ClipRRect( | ||||
|                   borderRadius: BorderRadius.circular(8), | ||||
|                   child: Container( | ||||
|                     color: Colors.black.withOpacity(0.5), | ||||
|                     child: Material( | ||||
|                       color: Colors.transparent, | ||||
|                       child: Row( | ||||
|                         mainAxisSize: MainAxisSize.min, | ||||
|                         children: [ | ||||
|                           if (onDelete != null) | ||||
|                             InkWell( | ||||
|                               borderRadius: BorderRadius.circular(8), | ||||
|                               child: Icon( | ||||
|                                 item.isLink ? Symbols.link_off : Symbols.delete, | ||||
|                                 size: 14, | ||||
|                                 color: Colors.white, | ||||
|                               ).padding(horizontal: 8, vertical: 6), | ||||
|                               onTap: () { | ||||
|                                 onDelete?.call(); | ||||
|                               }, | ||||
|                             ), | ||||
|                           ).padding(horizontal: 2), | ||||
|                         if (onMove != null) | ||||
|                           InkWell( | ||||
|                             borderRadius: BorderRadius.circular(8), | ||||
|                             child: const Icon( | ||||
|                               Symbols.keyboard_arrow_up, | ||||
|                               size: 14, | ||||
|                               color: Colors.white, | ||||
|                             ).padding(horizontal: 8, vertical: 6), | ||||
|                             onTap: () { | ||||
|                               onMove?.call(-1); | ||||
|                             }, | ||||
|                           ), | ||||
|                         if (onMove != null) | ||||
|                           InkWell( | ||||
|                             borderRadius: BorderRadius.circular(8), | ||||
|                             child: const Icon( | ||||
|                               Symbols.keyboard_arrow_down, | ||||
|                               size: 14, | ||||
|                               color: Colors.white, | ||||
|                             ).padding(horizontal: 8, vertical: 6), | ||||
|                             onTap: () { | ||||
|                               onMove?.call(1); | ||||
|                             }, | ||||
|                           ), | ||||
|                         if (onInsert != null) | ||||
|                           InkWell( | ||||
|                             borderRadius: BorderRadius.circular(8), | ||||
|                             child: const Icon( | ||||
|                               Symbols.add, | ||||
|                               size: 14, | ||||
|                               color: Colors.white, | ||||
|                             ).padding(horizontal: 8, vertical: 6), | ||||
|                             onTap: () { | ||||
|                               onInsert?.call(); | ||||
|                             }, | ||||
|                           ), | ||||
|                       ], | ||||
|                           if (onDelete != null && onMove != null) | ||||
|                             SizedBox( | ||||
|                               height: 26, | ||||
|                               child: const VerticalDivider( | ||||
|                                 width: 0.3, | ||||
|                                 color: Colors.white, | ||||
|                                 thickness: 0.3, | ||||
|                               ), | ||||
|                             ).padding(horizontal: 2), | ||||
|                           if (onMove != null) | ||||
|                             InkWell( | ||||
|                               borderRadius: BorderRadius.circular(8), | ||||
|                               child: const Icon( | ||||
|                                 Symbols.keyboard_arrow_up, | ||||
|                                 size: 14, | ||||
|                                 color: Colors.white, | ||||
|                               ).padding(horizontal: 8, vertical: 6), | ||||
|                               onTap: () { | ||||
|                                 onMove?.call(-1); | ||||
|                               }, | ||||
|                             ), | ||||
|                           if (onMove != null) | ||||
|                             InkWell( | ||||
|                               borderRadius: BorderRadius.circular(8), | ||||
|                               child: const Icon( | ||||
|                                 Symbols.keyboard_arrow_down, | ||||
|                                 size: 14, | ||||
|                                 color: Colors.white, | ||||
|                               ).padding(horizontal: 8, vertical: 6), | ||||
|                               onTap: () { | ||||
|                                 onMove?.call(1); | ||||
|                               }, | ||||
|                             ), | ||||
|                           if (onInsert != null) | ||||
|                             InkWell( | ||||
|                               borderRadius: BorderRadius.circular(8), | ||||
|                               child: const Icon( | ||||
|                                 Symbols.add, | ||||
|                                 size: 14, | ||||
|                                 color: Colors.white, | ||||
|                               ).padding(horizontal: 8, vertical: 6), | ||||
|                               onTap: () { | ||||
|                                 onInsert?.call(); | ||||
|                               }, | ||||
|                             ), | ||||
|                         ], | ||||
|                       ), | ||||
|                     ), | ||||
|                   ), | ||||
|                 ), | ||||
|                 if (onRequestUpload != null) | ||||
|                   InkWell( | ||||
|                     borderRadius: BorderRadius.circular(8), | ||||
|                     onTap: () => onRequestUpload?.call(), | ||||
|                     child: ClipRRect( | ||||
|                       borderRadius: BorderRadius.circular(8), | ||||
|                       child: Container( | ||||
|                         color: Colors.black.withOpacity(0.5), | ||||
|                         padding: EdgeInsets.symmetric( | ||||
|                           horizontal: 8, | ||||
|                           vertical: 4, | ||||
|                         ), | ||||
|                         child: | ||||
|                             (item.isOnCloud) | ||||
|                                 ? Row( | ||||
|                                   mainAxisSize: MainAxisSize.min, | ||||
|                                   children: [ | ||||
|                                     Icon( | ||||
|                                       Symbols.cloud, | ||||
|                                       size: 16, | ||||
|                                       color: Colors.white, | ||||
|                                     ), | ||||
|                                     const Gap(8), | ||||
|                                     Text( | ||||
|                                       'On-cloud', | ||||
|                                       style: TextStyle(color: Colors.white), | ||||
|                                     ), | ||||
|                                   ], | ||||
|                                 ) | ||||
|                                 : Row( | ||||
|                                   mainAxisSize: MainAxisSize.min, | ||||
|                                   children: [ | ||||
|                                     Icon( | ||||
|                                       Symbols.cloud_off, | ||||
|                                       size: 16, | ||||
|                                       color: Colors.white, | ||||
|                                     ), | ||||
|                                     const Gap(8), | ||||
|                                     Text( | ||||
|                                       'On-device', | ||||
|                                       style: TextStyle(color: Colors.white), | ||||
|                                     ), | ||||
|                                   ], | ||||
|                                 ), | ||||
|                       ), | ||||
|                     ), | ||||
|                   ), | ||||
|               ], | ||||
|             ).padding(horizontal: 12, vertical: 8), | ||||
|             AspectRatio( | ||||
|               aspectRatio: ratio, | ||||
|               child: Stack( | ||||
|                 fit: StackFit.expand, | ||||
|                 children: [ | ||||
|                   Builder( | ||||
|                     key: ValueKey(item.hashCode), | ||||
|                     builder: (context) { | ||||
|                       if (item.isOnCloud) { | ||||
|                         return CloudFileWidget(item: item.data); | ||||
|                       } else if (item.data is XFile) { | ||||
|                         final file = item.data as XFile; | ||||
|                         if (file.path.isEmpty) { | ||||
|                           return FutureBuilder<Uint8List>( | ||||
|                             future: file.readAsBytes(), | ||||
|                             builder: (context, snapshot) { | ||||
|                               if (snapshot.hasData) { | ||||
|                                 return Image.memory(snapshot.data!); | ||||
|                               } | ||||
|                               return const Center( | ||||
|                                 child: CircularProgressIndicator(), | ||||
|                               ); | ||||
|                             }, | ||||
|                           ); | ||||
|                         } | ||||
|  | ||||
|                         switch (item.type) { | ||||
|                           case UniversalFileType.image: | ||||
|                             return kIsWeb | ||||
|                                 ? Image.network(file.path) | ||||
|                                 : Image.file(File(file.path)); | ||||
|                           default: | ||||
|                             return Column( | ||||
|                               children: [ | ||||
|                                 const Icon(Symbols.document_scanner), | ||||
|                                 Text(file.name), | ||||
|                               ], | ||||
|                             ); | ||||
|                         } | ||||
|                       } else if (item is List<int> || item is Uint8List) { | ||||
|                         switch (item.type) { | ||||
|                           case UniversalFileType.image: | ||||
|                             return Image.memory(item.data); | ||||
|                           default: | ||||
|                             return Column( | ||||
|                               children: [const Icon(Symbols.document_scanner)], | ||||
|                             ); | ||||
|                         } | ||||
|                       } | ||||
|                       return Placeholder(); | ||||
|                     }, | ||||
|                   ), | ||||
|                   if (progress != null) | ||||
|                     Positioned.fill( | ||||
|                       child: Container( | ||||
|                         color: Colors.black.withOpacity(0.3), | ||||
|                         padding: EdgeInsets.symmetric( | ||||
|                           horizontal: 40, | ||||
|                           vertical: 16, | ||||
|                         ), | ||||
|                         child: Column( | ||||
|                           mainAxisAlignment: MainAxisAlignment.center, | ||||
|                           crossAxisAlignment: CrossAxisAlignment.center, | ||||
|                           children: [ | ||||
|                             if (progress != null) | ||||
|                               Text( | ||||
|                                 '${progress!.toStringAsFixed(2)}%', | ||||
|                                 style: TextStyle(color: Colors.white), | ||||
|                               ) | ||||
|                             else | ||||
|                               Text( | ||||
|                                 'uploading'.tr(), | ||||
|                                 style: TextStyle(color: Colors.white), | ||||
|                               ), | ||||
|                             Gap(6), | ||||
|                             Center( | ||||
|                               child: LinearProgressIndicator( | ||||
|                                 value: | ||||
|                                     progress != null ? progress! / 100.0 : null, | ||||
|                               ), | ||||
|                             ), | ||||
|                           ], | ||||
|                         ), | ||||
|                       ), | ||||
|                     ), | ||||
|                 ], | ||||
|               ), | ||||
|             ), | ||||
|             if (onRequestUpload != null) | ||||
|               Positioned( | ||||
|                 top: 8, | ||||
|                 right: 8, | ||||
|                 child: InkWell( | ||||
|                   borderRadius: BorderRadius.circular(8), | ||||
|                   onTap: () => onRequestUpload?.call(), | ||||
|                   child: ClipRRect( | ||||
|                     borderRadius: BorderRadius.circular(8), | ||||
|                     child: Container( | ||||
|                       color: Colors.black.withOpacity(0.5), | ||||
|                       padding: EdgeInsets.symmetric(horizontal: 8, vertical: 4), | ||||
|                       child: | ||||
|                           (item.isOnCloud) | ||||
|                               ? Row( | ||||
|                                 mainAxisSize: MainAxisSize.min, | ||||
|                                 children: [ | ||||
|                                   Icon( | ||||
|                                     Symbols.cloud, | ||||
|                                     size: 16, | ||||
|                                     color: Colors.white, | ||||
|                                   ), | ||||
|                                   const Gap(8), | ||||
|                                   Text( | ||||
|                                     'On-cloud', | ||||
|                                     style: TextStyle(color: Colors.white), | ||||
|                                   ), | ||||
|                                 ], | ||||
|                               ) | ||||
|                               : Row( | ||||
|                                 mainAxisSize: MainAxisSize.min, | ||||
|                                 children: [ | ||||
|                                   Icon( | ||||
|                                     Symbols.cloud_off, | ||||
|                                     size: 16, | ||||
|                                     color: Colors.white, | ||||
|                                   ), | ||||
|                                   const Gap(8), | ||||
|                                   Text( | ||||
|                                     'On-device', | ||||
|                                     style: TextStyle(color: Colors.white), | ||||
|                                   ), | ||||
|                                 ], | ||||
|                               ), | ||||
|                     ), | ||||
|                   ), | ||||
|                 ), | ||||
|               ), | ||||
|           ], | ||||
|         ), | ||||
|       ), | ||||
|     ); | ||||
|  | ||||
|     return ContextMenuWidget( | ||||
|       menuProvider: | ||||
|           (MenuRequest request) => Menu( | ||||
|             children: [ | ||||
|               if (item.isOnDevice && item.type == UniversalFileType.image) | ||||
|                 MenuAction( | ||||
|                   title: 'crop'.tr(), | ||||
|                   image: MenuImage.icon(Symbols.crop), | ||||
|                   callback: () async { | ||||
|                     final result = await cropImage( | ||||
|                       context, | ||||
|                       image: item.data, | ||||
|                       replacePath: true, | ||||
|                     ); | ||||
|                     if (result == null) return; | ||||
|                     onUpdate?.call(item.copyWith(data: result)); | ||||
|                   }, | ||||
|                 ), | ||||
|               if (item.isOnCloud) | ||||
|                 MenuAction( | ||||
|                   title: 'rename'.tr(), | ||||
|                   image: MenuImage.icon(Symbols.edit), | ||||
|                   callback: () async { | ||||
|                     await _showRenameDialog(context, ref); | ||||
|                   }, | ||||
|                 ), | ||||
|               if (item.isOnCloud) | ||||
|                 MenuAction( | ||||
|                   title: 'markAsSensitive'.tr(), | ||||
|                   image: MenuImage.icon(Symbols.no_adult_content), | ||||
|                   callback: () async { | ||||
|                     await _showSensitiveDialog(context, ref); | ||||
|                   }, | ||||
|                 ), | ||||
|             ], | ||||
|           ), | ||||
|       child: contentWidget, | ||||
|     ); | ||||
|   } | ||||
| } | ||||
|   | ||||
							
								
								
									
										168
									
								
								lib/widgets/content/audio.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										168
									
								
								lib/widgets/content/audio.dart
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,168 @@ | ||||
| import 'dart:developer'; | ||||
|  | ||||
| import 'package:flutter/material.dart'; | ||||
| import 'package:flutter_cache_manager/flutter_cache_manager.dart'; | ||||
| import 'package:gap/gap.dart'; | ||||
| import 'package:hooks_riverpod/hooks_riverpod.dart'; | ||||
| import 'package:island/pods/network.dart'; | ||||
| import 'package:island/services/time.dart'; | ||||
| import 'package:material_symbols_icons/symbols.dart'; | ||||
| import 'package:media_kit/media_kit.dart'; | ||||
| import 'package:styled_widget/styled_widget.dart'; | ||||
|  | ||||
| class UniversalAudio extends ConsumerStatefulWidget { | ||||
|   final String uri; | ||||
|   final String filename; | ||||
|   final bool autoplay; | ||||
|   const UniversalAudio({ | ||||
|     super.key, | ||||
|     required this.uri, | ||||
|     required this.filename, | ||||
|     this.autoplay = false, | ||||
|   }); | ||||
|  | ||||
|   @override | ||||
|   ConsumerState<UniversalAudio> createState() => _UniversalAudioState(); | ||||
| } | ||||
|  | ||||
| class _UniversalAudioState extends ConsumerState<UniversalAudio> { | ||||
|   Player? _player; | ||||
|  | ||||
|   Duration _duration = Duration(seconds: 1); | ||||
|   Duration _duartionBuffered = Duration(seconds: 1); | ||||
|   Duration _position = Duration(seconds: 0); | ||||
|  | ||||
|   bool _sliderWorking = false; | ||||
|   Duration _sliderPosition = Duration(seconds: 0); | ||||
|  | ||||
|   void _openAudio() async { | ||||
|     final url = widget.uri; | ||||
|     MediaKit.ensureInitialized(); | ||||
|  | ||||
|     _player = Player(); | ||||
|     _player!.stream.position.listen((value) { | ||||
|       _position = value; | ||||
|       if (!_sliderWorking) _sliderPosition = _position; | ||||
|       setState(() {}); | ||||
|     }); | ||||
|     _player!.stream.buffer.listen((value) { | ||||
|       _duartionBuffered = value; | ||||
|       setState(() {}); | ||||
|     }); | ||||
|     _player!.stream.duration.listen((value) { | ||||
|       _duration = value; | ||||
|       setState(() {}); | ||||
|     }); | ||||
|  | ||||
|     String? uri; | ||||
|     final inCacheInfo = await DefaultCacheManager().getFileFromCache(url); | ||||
|     if (inCacheInfo == null) { | ||||
|       log('[MediaPlayer] Miss cache: $url'); | ||||
|       final token = ref.watch(tokenProvider)?.token; | ||||
|       DefaultCacheManager().downloadFile( | ||||
|         url, | ||||
|         authHeaders: {'Authorization': 'AtField $token'}, | ||||
|       ); | ||||
|       uri = url; | ||||
|     } else { | ||||
|       uri = inCacheInfo.file.path; | ||||
|       log('[MediaPlayer] Hit cache: $url'); | ||||
|     } | ||||
|  | ||||
|     _player!.open(Media(uri), play: widget.autoplay); | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   void initState() { | ||||
|     super.initState(); | ||||
|     _openAudio(); | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   void dispose() { | ||||
|     super.dispose(); | ||||
|     _player?.dispose(); | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     if (_player == null) { | ||||
|       return Center(child: CircularProgressIndicator()); | ||||
|     } | ||||
|  | ||||
|     return Card( | ||||
|       color: Theme.of(context).colorScheme.surfaceContainerLowest, | ||||
|       child: Row( | ||||
|         children: [ | ||||
|           IconButton.filled( | ||||
|             onPressed: () { | ||||
|               _player!.playOrPause().then((_) { | ||||
|                 if (mounted) setState(() {}); | ||||
|               }); | ||||
|             }, | ||||
|             icon: | ||||
|                 _player!.state.playing | ||||
|                     ? const Icon(Symbols.pause, fill: 1, color: Colors.white) | ||||
|                     : const Icon( | ||||
|                       Symbols.play_arrow, | ||||
|                       fill: 1, | ||||
|                       color: Colors.white, | ||||
|                     ), | ||||
|           ), | ||||
|           const Gap(20), | ||||
|           Expanded( | ||||
|             child: Column( | ||||
|               mainAxisSize: MainAxisSize.min, | ||||
|               crossAxisAlignment: CrossAxisAlignment.start, | ||||
|               children: [ | ||||
|                 AnimatedSwitcher( | ||||
|                   duration: const Duration(milliseconds: 300), | ||||
|                   child: | ||||
|                       (_player!.state.playing || _sliderWorking) | ||||
|                           ? SizedBox( | ||||
|                             width: double.infinity, | ||||
|                             key: const ValueKey('playing'), | ||||
|                             child: Text( | ||||
|                               '${_position.formatShortDuration()} / ${_duration.formatShortDuration()}', | ||||
|                             ), | ||||
|                           ) | ||||
|                           : SizedBox( | ||||
|                             width: double.infinity, | ||||
|                             key: const ValueKey('filename'), | ||||
|                             child: Text( | ||||
|                               widget.filename.isEmpty | ||||
|                                   ? 'Audio' | ||||
|                                   : widget.filename, | ||||
|                               maxLines: 1, | ||||
|                               overflow: TextOverflow.ellipsis, | ||||
|                             ), | ||||
|                           ), | ||||
|                 ), | ||||
|                 Slider( | ||||
|                   value: _sliderPosition.inMilliseconds.toDouble(), | ||||
|                   secondaryTrackValue: | ||||
|                       _duartionBuffered.inMilliseconds.toDouble(), | ||||
|                   max: _duration.inMilliseconds.toDouble(), | ||||
|                   onChangeStart: (_) { | ||||
|                     _sliderWorking = true; | ||||
|                   }, | ||||
|                   onChanged: (value) { | ||||
|                     _sliderPosition = Duration(milliseconds: value.toInt()); | ||||
|                     setState(() {}); | ||||
|                   }, | ||||
|                   onChangeEnd: (value) { | ||||
|                     _sliderPosition = Duration(milliseconds: value.toInt()); | ||||
|                     _sliderWorking = false; | ||||
|                     _player!.seek(_sliderPosition); | ||||
|                   }, | ||||
|                   year2023: true, | ||||
|                   padding: EdgeInsets.zero, | ||||
|                 ), | ||||
|               ], | ||||
|             ), | ||||
|           ), | ||||
|         ], | ||||
|       ).padding(horizontal: 24, vertical: 16), | ||||
|     ); | ||||
|   } | ||||
| } | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user