Compare commits
	
		
			65 Commits
		
	
	
		
			c7f5b63fe5
			...
			3.2.0+131
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 6f9d51673b | |||
| f8c6887769 | |||
| cd2a507b7f | |||
| 3cafce00a2 | |||
| 837f3fbe98 | |||
| ca7cc5d7ee | |||
| ef2c14daa2 | |||
| 3a17837cc6 | |||
| 2617a64acf | |||
| afe1e12a3b | |||
| be80f5ff85 | |||
| 3281d69eba | |||
| 77b6ce9937 | |||
| 39275f61b5 | |||
| 72193ba8f3 | |||
| 98dd9b6617 | |||
| a22b94a263 | |||
| 9c75eafdb3 | |||
| 28fda3d0c7 | |||
| 187c2ea43e | |||
| ae7d967461 | |||
| 1ce71f1fa1 | |||
| 9b68808c77 | |||
|  | 99b7bf8199 | ||
|  | eb9bb73c31 | ||
|  | a8c3830d67 | ||
|  | 07a5a19141 | ||
| ecc100ac45 | |||
| 573b76d3ff | |||
| f7dad5e419 | |||
| 9f2f1c0848 | |||
| 580d9fd979 | |||
| 3b375abc09 | |||
| c527b5e67c | |||
| e9f09bbe54 | |||
| 3aece9316c | |||
| a61c889c6c | |||
| 0dd3221a56 | |||
| 66918521f8 | |||
| bb1846e462 | |||
| a976a6eaf4 | |||
| 4252f66fd3 | |||
| f2d780b48f | |||
| 300541f9bb | |||
| 43787bb813 | |||
| 3417c51a3b | |||
| f98e603e82 | |||
| c9b71701c8 | |||
| 28e98488f1 | |||
| b4d476613e | |||
| b48a1aac44 | |||
| 596d212593 | |||
| 54f290327e | |||
| 16f248ceab | |||
| 856d811187 | |||
| d07b194c04 | |||
| 2554b58be6 | |||
| a627b5838e | |||
| c479a9f381 | |||
| 02057e663b | |||
| 6501594100 | |||
| c6599edc3d | |||
| 709a0620b6 | |||
| f9b2a96c7c | |||
| 4dca6189cb | 
| @@ -62,4 +62,3 @@ If you want to build the release version, use the flutter build command. Learn m | |||||||
| ```bash | ```bash | ||||||
| flutter build <platform> | flutter build <platform> | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
|   | |||||||
| @@ -195,6 +195,7 @@ | |||||||
|   "checkInResultLevel2": "A Normal Day", |   "checkInResultLevel2": "A Normal Day", | ||||||
|   "checkInResultLevel3": "Good Luck", |   "checkInResultLevel3": "Good Luck", | ||||||
|   "checkInResultLevel4": "Best Luck", |   "checkInResultLevel4": "Best Luck", | ||||||
|  |   "checkInResultLevel5": "Happy Birthday 🥳", | ||||||
|   "checkInActivityTitle": "{} checked in on {} and got a {}", |   "checkInActivityTitle": "{} checked in on {} and got a {}", | ||||||
|   "eventCalander": "Event Calander", |   "eventCalander": "Event Calander", | ||||||
|   "eventCalanderEmpty": "No events on that day.", |   "eventCalanderEmpty": "No events on that day.", | ||||||
| @@ -228,6 +229,8 @@ | |||||||
|   "settings": "Settings", |   "settings": "Settings", | ||||||
|   "language": "Language", |   "language": "Language", | ||||||
|   "accountLanguageHint": "This language will be used for email and push notifications.", |   "accountLanguageHint": "This language will be used for email and push notifications.", | ||||||
|  |   "region": "Region", | ||||||
|  |   "accountRegionHint": "This region will be used for content delivery and localization.", | ||||||
|   "settingsDisplayLanguage": "Display Language", |   "settingsDisplayLanguage": "Display Language", | ||||||
|   "languageFollowSystem": "Follow System", |   "languageFollowSystem": "Follow System", | ||||||
|   "postsCreatedCount": "Posts", |   "postsCreatedCount": "Posts", | ||||||
| @@ -338,6 +341,7 @@ | |||||||
|   "notifications": "Notifications", |   "notifications": "Notifications", | ||||||
|   "posts": "Posts", |   "posts": "Posts", | ||||||
|   "settingsBackgroundImage": "Background Image", |   "settingsBackgroundImage": "Background Image", | ||||||
|  |   "settingsBackgroundImageEnable": "Show Background Image", | ||||||
|   "settingsBackgroundImageClear": "Clear Background Image", |   "settingsBackgroundImageClear": "Clear Background Image", | ||||||
|   "settingsBackgroundGenerateColor": "Generate color scheme from Bacground Image", |   "settingsBackgroundGenerateColor": "Generate color scheme from Bacground Image", | ||||||
|   "messageNone": "No content to display", |   "messageNone": "No content to display", | ||||||
| @@ -348,6 +352,8 @@ | |||||||
|   "chatBreakNone": "None", |   "chatBreakNone": "None", | ||||||
|   "settingsRealmCompactView": "Compact Realm View", |   "settingsRealmCompactView": "Compact Realm View", | ||||||
|   "settingsMixedFeed": "Mixed Feed", |   "settingsMixedFeed": "Mixed Feed", | ||||||
|  |   "settingsDataSavingMode": "Data Saving Mode", | ||||||
|  |   "dataSavingHint": "Data Saving Mode", | ||||||
|   "settingsAutoTranslate": "Auto Translate", |   "settingsAutoTranslate": "Auto Translate", | ||||||
|   "settingsHideBottomNav": "Hide Bottom Navigation", |   "settingsHideBottomNav": "Hide Bottom Navigation", | ||||||
|   "settingsSoundEffects": "Sound Effects", |   "settingsSoundEffects": "Sound Effects", | ||||||
| @@ -386,6 +392,7 @@ | |||||||
|   "postSettings": "Settings", |   "postSettings": "Settings", | ||||||
|   "postPublisherUnselected": "Publisher Unspecified", |   "postPublisherUnselected": "Publisher Unspecified", | ||||||
|   "postType": "Post Type", |   "postType": "Post Type", | ||||||
|  |   "postTypePost": "Post", | ||||||
|   "articleAttachmentHint": "Attachments must be uploaded and inserted into the article body to be visible.", |   "articleAttachmentHint": "Attachments must be uploaded and inserted into the article body to be visible.", | ||||||
|   "postVisibility": "Post Visibility", |   "postVisibility": "Post Visibility", | ||||||
|   "postVisibilityPublic": "Public", |   "postVisibilityPublic": "Public", | ||||||
| @@ -633,8 +640,9 @@ | |||||||
|   "chatJoin": "Join the Chat", |   "chatJoin": "Join the Chat", | ||||||
|   "realmJoin": "Join the Realm", |   "realmJoin": "Join the Realm", | ||||||
|   "realmJoinSuccess": "Successfully joined the realm.", |   "realmJoinSuccess": "Successfully joined the realm.", | ||||||
|   "discoverRealms": "Discover realms", |   "discoverRealms": "Realms", | ||||||
|   "discoverPublishers": "Discover publishers", |   "discoverPublishers": "Publishers", | ||||||
|  |   "discoverShuffledPost": "Random Posts", | ||||||
|   "search": "Search", |   "search": "Search", | ||||||
|   "publisherMembers": "Collaborators", |   "publisherMembers": "Collaborators", | ||||||
|   "developerHub": "Developer Hub", |   "developerHub": "Developer Hub", | ||||||
| @@ -692,7 +700,7 @@ | |||||||
|   "publisherFeatureDevelopDescription": "Unlock development abilities for your publisher, including custom apps, API keys, and more.", |   "publisherFeatureDevelopDescription": "Unlock development abilities for your publisher, including custom apps, API keys, and more.", | ||||||
|   "publisherFeatureDevelopHint": "Currently, this feature is under active development, you need send a request to unlock this feature.", |   "publisherFeatureDevelopHint": "Currently, this feature is under active development, you need send a request to unlock this feature.", | ||||||
|   "learnMore": "Learn More", |   "learnMore": "Learn More", | ||||||
|   "discoverWebArticles": "Articles from external sites", |   "discoverWebArticles": "Web Feed Articles", | ||||||
|   "webArticlesStand": "Article Stand", |   "webArticlesStand": "Article Stand", | ||||||
|   "about": "About", |   "about": "About", | ||||||
|   "membershipCancel": "Cancel Membership", |   "membershipCancel": "Cancel Membership", | ||||||
| @@ -883,6 +891,7 @@ | |||||||
|   "stellarProgram": "Stellar Program", |   "stellarProgram": "Stellar Program", | ||||||
|   "socialCredits": "Social Credits", |   "socialCredits": "Social Credits", | ||||||
|   "credits": "Credits", |   "credits": "Credits", | ||||||
|  |   "creditsStatus": "Credits Status", | ||||||
|   "socialCreditsDescription": "Social Credit is a way for Solar Network to evaluate users. It is calculated based on their behavior and interactions. With a base score of 100, higher scores indicate a user's credibility within the community. Scores change over time to reflect a user's recent behavior. Users with higher credit ratings enjoy more benefits, while users with lower credit ratings may have some functionality restricted.", |   "socialCreditsDescription": "Social Credit is a way for Solar Network to evaluate users. It is calculated based on their behavior and interactions. With a base score of 100, higher scores indicate a user's credibility within the community. Scores change over time to reflect a user's recent behavior. Users with higher credit ratings enjoy more benefits, while users with lower credit ratings may have some functionality restricted.", | ||||||
|   "socialCreditsLevelPoor": "Poor", |   "socialCreditsLevelPoor": "Poor", | ||||||
|   "socialCreditsLevelNormal": "Normal", |   "socialCreditsLevelNormal": "Normal", | ||||||
| @@ -926,5 +935,47 @@ | |||||||
|   "newSecretGenerated": "New Secret Generated", |   "newSecretGenerated": "New Secret Generated", | ||||||
|   "copySecretHint": "Please copy this secret and store it somewhere safe. You will not be able to see it again.", |   "copySecretHint": "Please copy this secret and store it somewhere safe. You will not be able to see it again.", | ||||||
|   "expiresIn": "Expires In (seconds)", |   "expiresIn": "Expires In (seconds)", | ||||||
|   "isOidc": "OIDC Compliant" |   "isOidc": "OIDC Compliant", | ||||||
| } |   "pinPost": "Pin Post", | ||||||
|  |   "unpinPost": "Unpin Post", | ||||||
|  |   "pinnedPost": "Pinned", | ||||||
|  |   "publisherPage": "Publisher Page", | ||||||
|  |   "realmPage": "Realm Page", | ||||||
|  |   "replyPage": "Reply Page", | ||||||
|  |   "pinPostPublisherHint": "Pin this post to your publisher page", | ||||||
|  |   "pinPostRealmHint": "Pin this post to the realm page", | ||||||
|  |   "pinPostRealmDisabledHint": "This post doesn't belong to any realm", | ||||||
|  |   "pinPostReplyHint": "Pin this post to the reply page", | ||||||
|  |   "pinPostReplyDisabledHint": "This post is not a reply", | ||||||
|  |   "pin": "Pin", | ||||||
|  |   "unpinPostHint": "Are you sure you want to unpin this post?", | ||||||
|  |   "all": "All", | ||||||
|  |   "statusPresent": "Present", | ||||||
|  |   "accountAutomated": "Automated", | ||||||
|  |   "chatBreakClearButton": "Clear", | ||||||
|  |   "chatBreak5m": "5m", | ||||||
|  |   "chatBreak10m": "10m", | ||||||
|  |   "chatBreak15m": "15m", | ||||||
|  |   "chatBreak30m": "30m", | ||||||
|  |   "chatBreakCustomMinutes": "Custom (minutes)", | ||||||
|  |   "errorGeneric": "Error: {}", | ||||||
|  |   "searchMessages": "Search Messages", | ||||||
|  |   "messagesCount": "{} messages", | ||||||
|  |   "dotSeparator": "·", | ||||||
|  |   "roleValidationHint": "Role must be between 0 and 100", | ||||||
|  |   "searchMessagesHint": "Search messages...", | ||||||
|  |   "searchLinks": "Links", | ||||||
|  |   "searchAttachments": "Attachments", | ||||||
|  |   "noMessagesFound": "No messages found", | ||||||
|  |   "openInBrowser": "Open in Browser", | ||||||
|  |   "highlightPost": "Highlight Post", | ||||||
|  |   "filters": "Filters", | ||||||
|  |   "apply": "Apply", | ||||||
|  |   "pubName": "Pub Name", | ||||||
|  |   "realm": "Realm", | ||||||
|  |   "shuffle": "Shuffle", | ||||||
|  |   "pinned": "Pinned", | ||||||
|  |   "noResultsFound": "No results found", | ||||||
|  |   "toggleFilters": "Toggle filters", | ||||||
|  |   "notableDayNext": "{} is in" | ||||||
|  | } | ||||||
|   | |||||||
| @@ -158,11 +158,12 @@ | |||||||
|   "checkIn": "签到", |   "checkIn": "签到", | ||||||
|   "checkInNone": "尚未签到", |   "checkInNone": "尚未签到", | ||||||
|   "checkInNoneHint": "通过签到获取您的财富提示和每日奖励。", |   "checkInNoneHint": "通过签到获取您的财富提示和每日奖励。", | ||||||
|   "checkInResultLevel0": "最差运气", |   "checkInResultLevel0": "大凶", | ||||||
|   "checkInResultLevel1": "坏运气", |   "checkInResultLevel1": "凶", | ||||||
|   "checkInResultLevel2": "一个普通的日常", |   "checkInResultLevel2": "中平", | ||||||
|   "checkInResultLevel3": "好运", |   "checkInResultLevel3": "吉", | ||||||
|   "checkInResultLevel4": "最佳运气", |   "checkInResultLevel4": "大吉", | ||||||
|  |   "checkInResultLevel5": "生日快乐 🥳", | ||||||
|   "checkInActivityTitle": "{} 在 {} 签到并获得了 {}", |   "checkInActivityTitle": "{} 在 {} 签到并获得了 {}", | ||||||
|   "eventCalander": "活动日历", |   "eventCalander": "活动日历", | ||||||
|   "eventCalanderEmpty": "该日无活动。", |   "eventCalanderEmpty": "该日无活动。", | ||||||
| @@ -304,6 +305,7 @@ | |||||||
|   "notifications": "通知", |   "notifications": "通知", | ||||||
|   "posts": "帖子", |   "posts": "帖子", | ||||||
|   "settingsBackgroundImage": "背景图片", |   "settingsBackgroundImage": "背景图片", | ||||||
|  |   "settingsBackgroundImageEnable": "显示背景图片", | ||||||
|   "settingsBackgroundImageClear": "清除背景图片", |   "settingsBackgroundImageClear": "清除背景图片", | ||||||
|   "settingsBackgroundGenerateColor": "从背景图像生成主题色", |   "settingsBackgroundGenerateColor": "从背景图像生成主题色", | ||||||
|   "messageNone": "没有内容可显示", |   "messageNone": "没有内容可显示", | ||||||
| @@ -314,6 +316,8 @@ | |||||||
|   "chatBreakNone": "无", |   "chatBreakNone": "无", | ||||||
|   "settingsRealmCompactView": "紧凑领域视图", |   "settingsRealmCompactView": "紧凑领域视图", | ||||||
|   "settingsMixedFeed": "混合动态", |   "settingsMixedFeed": "混合动态", | ||||||
|  |   "settingsDataSavingMode": "流量节省模式", | ||||||
|  |   "dataSavingHint": "流量节省模式", | ||||||
|   "settingsAutoTranslate": "自动翻译", |   "settingsAutoTranslate": "自动翻译", | ||||||
|   "settingsHideBottomNav": "隐藏底部导航", |   "settingsHideBottomNav": "隐藏底部导航", | ||||||
|   "settingsSoundEffects": "音效", |   "settingsSoundEffects": "音效", | ||||||
| @@ -855,5 +859,10 @@ | |||||||
|   "newSecretGenerated": "已生成新密钥", |   "newSecretGenerated": "已生成新密钥", | ||||||
|   "copySecretHint": "请复制此密钥并将其存放在安全的地方。您将无法再次看到它。", |   "copySecretHint": "请复制此密钥并将其存放在安全的地方。您将无法再次看到它。", | ||||||
|   "expiresIn": "过期时间(秒)", |   "expiresIn": "过期时间(秒)", | ||||||
|   "isOidc": "OIDC 兼容" |   "isOidc": "OIDC 兼容", | ||||||
|  |   "statusPresent": "至今", | ||||||
|  |   "accountAutomated": "机器人", | ||||||
|  |   "openInBrowser": "在浏览器中打开", | ||||||
|  |   "highlightPost": "精选帖子", | ||||||
|  |   "notableDayNext": "距离 {} 还有" | ||||||
| } | } | ||||||
|   | |||||||
| @@ -303,7 +303,8 @@ | |||||||
|     "notifications": "通知", |     "notifications": "通知", | ||||||
|     "posts": "帖子", |     "posts": "帖子", | ||||||
|     "settingsBackgroundImage": "背景圖片", |     "settingsBackgroundImage": "背景圖片", | ||||||
|     "settingsBackgroundImageClear": "清除背景圖片", |   "settingsBackgroundImageEnable": "顯示背景圖片", | ||||||
|  |   "settingsBackgroundImageClear": "清除背景圖片", | ||||||
|     "settingsBackgroundGenerateColor": "從背景圖像生成主題色", |     "settingsBackgroundGenerateColor": "從背景圖像生成主題色", | ||||||
|     "messageNone": "沒有內容可顯示", |     "messageNone": "沒有內容可顯示", | ||||||
|     "unreadMessages": { |     "unreadMessages": { | ||||||
| @@ -314,6 +315,8 @@ | |||||||
|     "settingsRealmCompactView": "緊湊領域視圖", |     "settingsRealmCompactView": "緊湊領域視圖", | ||||||
|     "settingsMixedFeed": "混合動態", |     "settingsMixedFeed": "混合動態", | ||||||
|     "settingsAutoTranslate": "自動翻譯", |     "settingsAutoTranslate": "自動翻譯", | ||||||
|  |     "settingsDataSavingMode": "低數據模式", | ||||||
|  |     "dataSavingHint": "低數據模式", | ||||||
|     "settingsHideBottomNav": "隱藏底部導航", |     "settingsHideBottomNav": "隱藏底部導航", | ||||||
|     "settingsSoundEffects": "音效", |     "settingsSoundEffects": "音效", | ||||||
|     "settingsAprilFoolFeatures": "愚人節功能", |     "settingsAprilFoolFeatures": "愚人節功能", | ||||||
| @@ -824,4 +827,4 @@ | |||||||
|     "copySecretHint": "請複製此密鑰並將其存放在安全的地方。您將無法再次看到它。", |     "copySecretHint": "請複製此密鑰並將其存放在安全的地方。您將無法再次看到它。", | ||||||
|     "expiresIn": "過期時間(秒)", |     "expiresIn": "過期時間(秒)", | ||||||
|     "isOidc": "OIDC 相容" |     "isOidc": "OIDC 相容" | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										12
									
								
								assets/icons/icon-outline.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								assets/icons/icon-outline.svg
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,12 @@ | |||||||
|  | <svg xmlns="http://www.w3.org/2000/svg" width="192" height="192" fill="none"> | ||||||
|  |     <path stroke="#fff" stroke-linecap="round" stroke-linejoin="round" stroke-width="12" | ||||||
|  |         d="M54 147h86" /> | ||||||
|  |     <path stroke="#fff" stroke-linecap="round" stroke-linejoin="round" stroke-width="10" | ||||||
|  |         d="M57 111s-2-4.5-2-10m22 22s-4 7-11 4m9-22s-2-4.5-2-10" /> | ||||||
|  |     <path stroke="#fff" stroke-linecap="round" stroke-linejoin="round" stroke-width="12" | ||||||
|  |         d="M54 147a32 32 0 0 1-11.999-61.665A39 39 0 0 1 81 46m59 101a30 30 0 0 0 29.933-28" /> | ||||||
|  |     <circle cx="132" cy="75" r="4" stroke="#fff" stroke-linecap="round" stroke-linejoin="round" | ||||||
|  |         stroke-width="8" /> | ||||||
|  |     <path stroke="#fff" stroke-linecap="round" stroke-linejoin="round" stroke-width="10" | ||||||
|  |         d="M112.5 41.217C100.843 47.961 93 60.564 93 75c0 6.375 1.53 12.393 4.242 17.707m69.513-35.419A38.84 38.84 0 0 1 171 75c0 14.433-7.84 27.034-19.493 33.779m-.793-43.317A20.9 20.9 0 0 1 153 75c0 7.77-4.221 14.556-10.495 18.188m-21.003-36.38C115.224 60.44 111 67.226 111 75a20.9 20.9 0 0 0 2.284 9.533" /> | ||||||
|  | </svg> | ||||||
| After Width: | Height: | Size: 1.0 KiB | 
							
								
								
									
										
											BIN
										
									
								
								assets/images/media-offline.jpg
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								assets/images/media-offline.jpg
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 461 KiB | 
										
											Binary file not shown.
										
									
								
							| Before Width: | Height: | Size: 307 KiB | 
| @@ -21,6 +21,6 @@ | |||||||
|   <key>CFBundleVersion</key> |   <key>CFBundleVersion</key> | ||||||
|   <string>1.0</string> |   <string>1.0</string> | ||||||
|   <key>MinimumOSVersion</key> |   <key>MinimumOSVersion</key> | ||||||
|   <string>12.0</string> |   <string>13.0</string> | ||||||
| </dict> | </dict> | ||||||
| </plist> | </plist> | ||||||
|   | |||||||
							
								
								
									
										168
									
								
								ios/Podfile.lock
									
									
									
									
									
								
							
							
						
						
									
										168
									
								
								ios/Podfile.lock
									
									
									
									
									
								
							| @@ -40,83 +40,85 @@ PODS: | |||||||
|   - file_picker (0.0.1): |   - file_picker (0.0.1): | ||||||
|     - DKImagePickerController/PhotoGallery |     - DKImagePickerController/PhotoGallery | ||||||
|     - Flutter |     - Flutter | ||||||
|   - Firebase/CoreOnly (12.0.0): |   - file_saver (0.0.1): | ||||||
|     - FirebaseCore (~> 12.0.0) |     - Flutter | ||||||
|   - Firebase/Crashlytics (12.0.0): |   - Firebase/CoreOnly (12.2.0): | ||||||
|  |     - FirebaseCore (~> 12.2.0) | ||||||
|  |   - Firebase/Crashlytics (12.2.0): | ||||||
|     - Firebase/CoreOnly |     - Firebase/CoreOnly | ||||||
|     - FirebaseCrashlytics (~> 12.0.0) |     - FirebaseCrashlytics (~> 12.2.0) | ||||||
|   - Firebase/Messaging (12.0.0): |   - Firebase/Messaging (12.2.0): | ||||||
|     - Firebase/CoreOnly |     - Firebase/CoreOnly | ||||||
|     - FirebaseMessaging (~> 12.0.0) |     - FirebaseMessaging (~> 12.2.0) | ||||||
|   - firebase_analytics (12.0.0): |   - firebase_analytics (12.0.1): | ||||||
|     - firebase_core |     - firebase_core | ||||||
|     - FirebaseAnalytics (= 12.0.0) |     - FirebaseAnalytics (= 12.2.0) | ||||||
|     - Flutter |     - Flutter | ||||||
|   - firebase_core (4.0.0): |   - firebase_core (4.1.0): | ||||||
|     - Firebase/CoreOnly (= 12.0.0) |     - Firebase/CoreOnly (= 12.2.0) | ||||||
|     - Flutter |     - Flutter | ||||||
|   - firebase_crashlytics (5.0.0): |   - firebase_crashlytics (5.0.1): | ||||||
|     - Firebase/Crashlytics (= 12.0.0) |     - Firebase/Crashlytics (= 12.2.0) | ||||||
|     - firebase_core |     - firebase_core | ||||||
|     - Flutter |     - Flutter | ||||||
|   - firebase_messaging (16.0.0): |   - firebase_messaging (16.0.1): | ||||||
|     - Firebase/Messaging (= 12.0.0) |     - Firebase/Messaging (= 12.2.0) | ||||||
|     - firebase_core |     - firebase_core | ||||||
|     - Flutter |     - Flutter | ||||||
|   - FirebaseAnalytics (12.0.0): |   - FirebaseAnalytics (12.2.0): | ||||||
|     - FirebaseAnalytics/Default (= 12.0.0) |     - FirebaseAnalytics/Default (= 12.2.0) | ||||||
|     - FirebaseCore (~> 12.0.0) |     - FirebaseCore (~> 12.2.0) | ||||||
|     - FirebaseInstallations (~> 12.0.0) |     - FirebaseInstallations (~> 12.2.0) | ||||||
|     - GoogleUtilities/AppDelegateSwizzler (~> 8.1) |     - GoogleUtilities/AppDelegateSwizzler (~> 8.1) | ||||||
|     - GoogleUtilities/MethodSwizzler (~> 8.1) |     - GoogleUtilities/MethodSwizzler (~> 8.1) | ||||||
|     - GoogleUtilities/Network (~> 8.1) |     - GoogleUtilities/Network (~> 8.1) | ||||||
|     - "GoogleUtilities/NSData+zlib (~> 8.1)" |     - "GoogleUtilities/NSData+zlib (~> 8.1)" | ||||||
|     - nanopb (~> 3.30910.0) |     - nanopb (~> 3.30910.0) | ||||||
|   - FirebaseAnalytics/Default (12.0.0): |   - FirebaseAnalytics/Default (12.2.0): | ||||||
|     - FirebaseCore (~> 12.0.0) |     - FirebaseCore (~> 12.2.0) | ||||||
|     - FirebaseInstallations (~> 12.0.0) |     - FirebaseInstallations (~> 12.2.0) | ||||||
|     - GoogleAppMeasurement/Default (= 12.0.0) |     - GoogleAppMeasurement/Default (= 12.2.0) | ||||||
|     - GoogleUtilities/AppDelegateSwizzler (~> 8.1) |     - GoogleUtilities/AppDelegateSwizzler (~> 8.1) | ||||||
|     - GoogleUtilities/MethodSwizzler (~> 8.1) |     - GoogleUtilities/MethodSwizzler (~> 8.1) | ||||||
|     - GoogleUtilities/Network (~> 8.1) |     - GoogleUtilities/Network (~> 8.1) | ||||||
|     - "GoogleUtilities/NSData+zlib (~> 8.1)" |     - "GoogleUtilities/NSData+zlib (~> 8.1)" | ||||||
|     - nanopb (~> 3.30910.0) |     - nanopb (~> 3.30910.0) | ||||||
|   - FirebaseCore (12.0.0): |   - FirebaseCore (12.2.0): | ||||||
|     - FirebaseCoreInternal (~> 12.0.0) |     - FirebaseCoreInternal (~> 12.2.0) | ||||||
|     - GoogleUtilities/Environment (~> 8.1) |     - GoogleUtilities/Environment (~> 8.1) | ||||||
|     - GoogleUtilities/Logger (~> 8.1) |     - GoogleUtilities/Logger (~> 8.1) | ||||||
|   - FirebaseCoreExtension (12.0.0): |   - FirebaseCoreExtension (12.2.0): | ||||||
|     - FirebaseCore (~> 12.0.0) |     - FirebaseCore (~> 12.2.0) | ||||||
|   - FirebaseCoreInternal (12.0.0): |   - FirebaseCoreInternal (12.2.0): | ||||||
|     - "GoogleUtilities/NSData+zlib (~> 8.1)" |     - "GoogleUtilities/NSData+zlib (~> 8.1)" | ||||||
|   - FirebaseCrashlytics (12.0.0): |   - FirebaseCrashlytics (12.2.0): | ||||||
|     - FirebaseCore (~> 12.0.0) |     - FirebaseCore (~> 12.2.0) | ||||||
|     - FirebaseInstallations (~> 12.0.0) |     - FirebaseInstallations (~> 12.2.0) | ||||||
|     - FirebaseRemoteConfigInterop (~> 12.0.0) |     - FirebaseRemoteConfigInterop (~> 12.2.0) | ||||||
|     - FirebaseSessions (~> 12.0.0) |     - FirebaseSessions (~> 12.2.0) | ||||||
|     - GoogleDataTransport (~> 10.1) |     - GoogleDataTransport (~> 10.1) | ||||||
|     - GoogleUtilities/Environment (~> 8.1) |     - GoogleUtilities/Environment (~> 8.1) | ||||||
|     - nanopb (~> 3.30910.0) |     - nanopb (~> 3.30910.0) | ||||||
|     - PromisesObjC (~> 2.4) |     - PromisesObjC (~> 2.4) | ||||||
|   - FirebaseInstallations (12.0.0): |   - FirebaseInstallations (12.2.0): | ||||||
|     - FirebaseCore (~> 12.0.0) |     - FirebaseCore (~> 12.2.0) | ||||||
|     - GoogleUtilities/Environment (~> 8.1) |     - GoogleUtilities/Environment (~> 8.1) | ||||||
|     - GoogleUtilities/UserDefaults (~> 8.1) |     - GoogleUtilities/UserDefaults (~> 8.1) | ||||||
|     - PromisesObjC (~> 2.4) |     - PromisesObjC (~> 2.4) | ||||||
|   - FirebaseMessaging (12.0.0): |   - FirebaseMessaging (12.2.0): | ||||||
|     - FirebaseCore (~> 12.0.0) |     - FirebaseCore (~> 12.2.0) | ||||||
|     - FirebaseInstallations (~> 12.0.0) |     - FirebaseInstallations (~> 12.2.0) | ||||||
|     - GoogleDataTransport (~> 10.1) |     - GoogleDataTransport (~> 10.1) | ||||||
|     - GoogleUtilities/AppDelegateSwizzler (~> 8.1) |     - GoogleUtilities/AppDelegateSwizzler (~> 8.1) | ||||||
|     - GoogleUtilities/Environment (~> 8.1) |     - GoogleUtilities/Environment (~> 8.1) | ||||||
|     - GoogleUtilities/Reachability (~> 8.1) |     - GoogleUtilities/Reachability (~> 8.1) | ||||||
|     - GoogleUtilities/UserDefaults (~> 8.1) |     - GoogleUtilities/UserDefaults (~> 8.1) | ||||||
|     - nanopb (~> 3.30910.0) |     - nanopb (~> 3.30910.0) | ||||||
|   - FirebaseRemoteConfigInterop (12.0.0) |   - FirebaseRemoteConfigInterop (12.2.0) | ||||||
|   - FirebaseSessions (12.0.0): |   - FirebaseSessions (12.2.0): | ||||||
|     - FirebaseCore (~> 12.0.0) |     - FirebaseCore (~> 12.2.0) | ||||||
|     - FirebaseCoreExtension (~> 12.0.0) |     - FirebaseCoreExtension (~> 12.2.0) | ||||||
|     - FirebaseInstallations (~> 12.0.0) |     - FirebaseInstallations (~> 12.2.0) | ||||||
|     - GoogleDataTransport (~> 10.1) |     - GoogleDataTransport (~> 10.1) | ||||||
|     - GoogleUtilities/Environment (~> 8.1) |     - GoogleUtilities/Environment (~> 8.1) | ||||||
|     - GoogleUtilities/UserDefaults (~> 8.1) |     - GoogleUtilities/UserDefaults (~> 8.1) | ||||||
| @@ -134,6 +136,8 @@ PODS: | |||||||
|     - OrderedSet (~> 6.0.3) |     - OrderedSet (~> 6.0.3) | ||||||
|   - flutter_keyboard_visibility (0.0.1): |   - flutter_keyboard_visibility (0.0.1): | ||||||
|     - Flutter |     - Flutter | ||||||
|  |   - flutter_local_notifications (0.0.1): | ||||||
|  |     - Flutter | ||||||
|   - flutter_native_splash (2.4.3): |   - flutter_native_splash (2.4.3): | ||||||
|     - Flutter |     - Flutter | ||||||
|   - flutter_platform_alert (0.0.1): |   - flutter_platform_alert (0.0.1): | ||||||
| @@ -145,33 +149,33 @@ PODS: | |||||||
|   - flutter_udid (0.0.1): |   - flutter_udid (0.0.1): | ||||||
|     - Flutter |     - Flutter | ||||||
|     - SAMKeychain |     - SAMKeychain | ||||||
|   - flutter_webrtc (1.0.0): |   - flutter_webrtc (1.1.0): | ||||||
|     - Flutter |     - Flutter | ||||||
|     - WebRTC-SDK (= 137.7151.02) |     - WebRTC-SDK (= 137.7151.03) | ||||||
|   - gal (1.0.0): |   - gal (1.0.0): | ||||||
|     - Flutter |     - Flutter | ||||||
|     - FlutterMacOS |     - FlutterMacOS | ||||||
|   - GoogleAdsOnDeviceConversion (2.1.0): |   - GoogleAdsOnDeviceConversion (2.3.0): | ||||||
|     - GoogleUtilities/Logger (~> 8.1) |     - GoogleUtilities/Logger (~> 8.1) | ||||||
|     - GoogleUtilities/Network (~> 8.1) |     - GoogleUtilities/Network (~> 8.1) | ||||||
|     - nanopb (~> 3.30910.0) |     - nanopb (~> 3.30910.0) | ||||||
|   - GoogleAppMeasurement/Core (12.0.0): |   - GoogleAppMeasurement/Core (12.2.0): | ||||||
|     - GoogleUtilities/AppDelegateSwizzler (~> 8.1) |     - GoogleUtilities/AppDelegateSwizzler (~> 8.1) | ||||||
|     - GoogleUtilities/MethodSwizzler (~> 8.1) |     - GoogleUtilities/MethodSwizzler (~> 8.1) | ||||||
|     - GoogleUtilities/Network (~> 8.1) |     - GoogleUtilities/Network (~> 8.1) | ||||||
|     - "GoogleUtilities/NSData+zlib (~> 8.1)" |     - "GoogleUtilities/NSData+zlib (~> 8.1)" | ||||||
|     - nanopb (~> 3.30910.0) |     - nanopb (~> 3.30910.0) | ||||||
|   - GoogleAppMeasurement/Default (12.0.0): |   - GoogleAppMeasurement/Default (12.2.0): | ||||||
|     - GoogleAdsOnDeviceConversion (= 2.1.0) |     - GoogleAdsOnDeviceConversion (= 2.3.0) | ||||||
|     - GoogleAppMeasurement/Core (= 12.0.0) |     - GoogleAppMeasurement/Core (= 12.2.0) | ||||||
|     - GoogleAppMeasurement/IdentitySupport (= 12.0.0) |     - GoogleAppMeasurement/IdentitySupport (= 12.2.0) | ||||||
|     - GoogleUtilities/AppDelegateSwizzler (~> 8.1) |     - GoogleUtilities/AppDelegateSwizzler (~> 8.1) | ||||||
|     - GoogleUtilities/MethodSwizzler (~> 8.1) |     - GoogleUtilities/MethodSwizzler (~> 8.1) | ||||||
|     - GoogleUtilities/Network (~> 8.1) |     - GoogleUtilities/Network (~> 8.1) | ||||||
|     - "GoogleUtilities/NSData+zlib (~> 8.1)" |     - "GoogleUtilities/NSData+zlib (~> 8.1)" | ||||||
|     - nanopb (~> 3.30910.0) |     - nanopb (~> 3.30910.0) | ||||||
|   - GoogleAppMeasurement/IdentitySupport (12.0.0): |   - GoogleAppMeasurement/IdentitySupport (12.2.0): | ||||||
|     - GoogleAppMeasurement/Core (= 12.0.0) |     - GoogleAppMeasurement/Core (= 12.2.0) | ||||||
|     - GoogleUtilities/AppDelegateSwizzler (~> 8.1) |     - GoogleUtilities/AppDelegateSwizzler (~> 8.1) | ||||||
|     - GoogleUtilities/MethodSwizzler (~> 8.1) |     - GoogleUtilities/MethodSwizzler (~> 8.1) | ||||||
|     - GoogleUtilities/Network (~> 8.1) |     - GoogleUtilities/Network (~> 8.1) | ||||||
| @@ -215,7 +219,7 @@ PODS: | |||||||
|   - livekit_client (2.5.0): |   - livekit_client (2.5.0): | ||||||
|     - Flutter |     - Flutter | ||||||
|     - flutter_webrtc |     - flutter_webrtc | ||||||
|     - WebRTC-SDK (= 137.7151.02) |     - WebRTC-SDK (= 137.7151.03) | ||||||
|   - local_auth_darwin (0.0.1): |   - local_auth_darwin (0.0.1): | ||||||
|     - Flutter |     - Flutter | ||||||
|     - FlutterMacOS |     - FlutterMacOS | ||||||
| @@ -248,9 +252,9 @@ PODS: | |||||||
|   - record_ios (1.1.0): |   - record_ios (1.1.0): | ||||||
|     - Flutter |     - Flutter | ||||||
|   - SAMKeychain (1.5.3) |   - SAMKeychain (1.5.3) | ||||||
|   - SDWebImage (5.21.1): |   - SDWebImage (5.21.2): | ||||||
|     - SDWebImage/Core (= 5.21.1) |     - SDWebImage/Core (= 5.21.2) | ||||||
|   - SDWebImage/Core (5.21.1) |   - SDWebImage/Core (5.21.2) | ||||||
|   - share_plus (0.0.1): |   - share_plus (0.0.1): | ||||||
|     - Flutter |     - Flutter | ||||||
|   - shared_preferences_foundation (0.0.1): |   - shared_preferences_foundation (0.0.1): | ||||||
| @@ -295,7 +299,7 @@ PODS: | |||||||
|     - Flutter |     - Flutter | ||||||
|   - wakelock_plus (0.0.1): |   - wakelock_plus (0.0.1): | ||||||
|     - Flutter |     - Flutter | ||||||
|   - WebRTC-SDK (137.7151.02) |   - WebRTC-SDK (137.7151.03) | ||||||
|  |  | ||||||
| DEPENDENCIES: | DEPENDENCIES: | ||||||
|   - Alamofire |   - Alamofire | ||||||
| @@ -303,6 +307,7 @@ DEPENDENCIES: | |||||||
|   - croppy (from `.symlinks/plugins/croppy/ios`) |   - croppy (from `.symlinks/plugins/croppy/ios`) | ||||||
|   - device_info_plus (from `.symlinks/plugins/device_info_plus/ios`) |   - device_info_plus (from `.symlinks/plugins/device_info_plus/ios`) | ||||||
|   - file_picker (from `.symlinks/plugins/file_picker/ios`) |   - file_picker (from `.symlinks/plugins/file_picker/ios`) | ||||||
|  |   - file_saver (from `.symlinks/plugins/file_saver/ios`) | ||||||
|   - firebase_analytics (from `.symlinks/plugins/firebase_analytics/ios`) |   - firebase_analytics (from `.symlinks/plugins/firebase_analytics/ios`) | ||||||
|   - firebase_core (from `.symlinks/plugins/firebase_core/ios`) |   - firebase_core (from `.symlinks/plugins/firebase_core/ios`) | ||||||
|   - firebase_crashlytics (from `.symlinks/plugins/firebase_crashlytics/ios`) |   - firebase_crashlytics (from `.symlinks/plugins/firebase_crashlytics/ios`) | ||||||
| @@ -311,6 +316,7 @@ DEPENDENCIES: | |||||||
|   - flutter_app_update (from `.symlinks/plugins/flutter_app_update/ios`) |   - flutter_app_update (from `.symlinks/plugins/flutter_app_update/ios`) | ||||||
|   - flutter_inappwebview_ios (from `.symlinks/plugins/flutter_inappwebview_ios/ios`) |   - flutter_inappwebview_ios (from `.symlinks/plugins/flutter_inappwebview_ios/ios`) | ||||||
|   - flutter_keyboard_visibility (from `.symlinks/plugins/flutter_keyboard_visibility/ios`) |   - flutter_keyboard_visibility (from `.symlinks/plugins/flutter_keyboard_visibility/ios`) | ||||||
|  |   - flutter_local_notifications (from `.symlinks/plugins/flutter_local_notifications/ios`) | ||||||
|   - flutter_native_splash (from `.symlinks/plugins/flutter_native_splash/ios`) |   - flutter_native_splash (from `.symlinks/plugins/flutter_native_splash/ios`) | ||||||
|   - flutter_platform_alert (from `.symlinks/plugins/flutter_platform_alert/ios`) |   - flutter_platform_alert (from `.symlinks/plugins/flutter_platform_alert/ios`) | ||||||
|   - flutter_secure_storage (from `.symlinks/plugins/flutter_secure_storage/ios`) |   - flutter_secure_storage (from `.symlinks/plugins/flutter_secure_storage/ios`) | ||||||
| @@ -381,6 +387,8 @@ EXTERNAL SOURCES: | |||||||
|     :path: ".symlinks/plugins/device_info_plus/ios" |     :path: ".symlinks/plugins/device_info_plus/ios" | ||||||
|   file_picker: |   file_picker: | ||||||
|     :path: ".symlinks/plugins/file_picker/ios" |     :path: ".symlinks/plugins/file_picker/ios" | ||||||
|  |   file_saver: | ||||||
|  |     :path: ".symlinks/plugins/file_saver/ios" | ||||||
|   firebase_analytics: |   firebase_analytics: | ||||||
|     :path: ".symlinks/plugins/firebase_analytics/ios" |     :path: ".symlinks/plugins/firebase_analytics/ios" | ||||||
|   firebase_core: |   firebase_core: | ||||||
| @@ -397,6 +405,8 @@ EXTERNAL SOURCES: | |||||||
|     :path: ".symlinks/plugins/flutter_inappwebview_ios/ios" |     :path: ".symlinks/plugins/flutter_inappwebview_ios/ios" | ||||||
|   flutter_keyboard_visibility: |   flutter_keyboard_visibility: | ||||||
|     :path: ".symlinks/plugins/flutter_keyboard_visibility/ios" |     :path: ".symlinks/plugins/flutter_keyboard_visibility/ios" | ||||||
|  |   flutter_local_notifications: | ||||||
|  |     :path: ".symlinks/plugins/flutter_local_notifications/ios" | ||||||
|   flutter_native_splash: |   flutter_native_splash: | ||||||
|     :path: ".symlinks/plugins/flutter_native_splash/ios" |     :path: ".symlinks/plugins/flutter_native_splash/ios" | ||||||
|   flutter_platform_alert: |   flutter_platform_alert: | ||||||
| @@ -464,39 +474,41 @@ SPEC CHECKSUMS: | |||||||
|   DKImagePickerController: 946cec48c7873164274ecc4624d19e3da4c1ef3c |   DKImagePickerController: 946cec48c7873164274ecc4624d19e3da4c1ef3c | ||||||
|   DKPhotoGallery: b3834fecb755ee09a593d7c9e389d8b5d6deed60 |   DKPhotoGallery: b3834fecb755ee09a593d7c9e389d8b5d6deed60 | ||||||
|   file_picker: a0560bc09d61de87f12d246fc47d2119e6ef37be |   file_picker: a0560bc09d61de87f12d246fc47d2119e6ef37be | ||||||
|   Firebase: 800d487043c0557d9faed71477a38d9aafb08a41 |   file_saver: 6cdbcddd690cb02b0c1a0c225b37cd805c2bf8b6 | ||||||
|   firebase_analytics: cd56fc56f75c1df30a6ff5290cd56e230996a76d |   Firebase: 26f6f8d460603af3df970ad505b16b15f5e2e9a1 | ||||||
|   firebase_core: 633e1851ffe1b9ab875f6467a4f574c79cef02e4 |   firebase_analytics: 111ff65791a430356bd6c7e4d7339537fc6a15ae | ||||||
|   firebase_crashlytics: 2c6c1a17900a38081d938330e9f48e60ec5b255d |   firebase_core: 3ff52146406557dddd01d570e807e203ec7e1302 | ||||||
|   firebase_messaging: d17feef781edc84ebefe62624fb384358ad96361 |   firebase_crashlytics: 3637078b718a52dc9fb4d64e37c969e86b87ff6f | ||||||
|   FirebaseAnalytics: 6d790cd1b159b4eb61a99948df0934ce505a34f7 |   firebase_messaging: 3dcc998dd98e1e54af75d0cccae8606eba43553c | ||||||
|   FirebaseCore: 055f4ab117d5964158c833f3d5e7ec6d91648d4a |   FirebaseAnalytics: e04e23bc070e3014aa5cf4980f9df7ce5cd79ec8 | ||||||
|   FirebaseCoreExtension: 639afb3de6abd611952be78a794c54a47fa0f361 |   FirebaseCore: 311c48a147ad4a0ab7febbaed89e8025c67510cd | ||||||
|   FirebaseCoreInternal: dedc28e569a4be85f38f3d6af1070a2e12018d55 |   FirebaseCoreExtension: 73af080c22a2f7b44cefa391dc08f7e4ee162cb5 | ||||||
|   FirebaseCrashlytics: db75aa0cab8d00f68406fa247c32fe17ade884d7 |   FirebaseCoreInternal: 56ea29f3dad2894f81b060f706f9d53509b6ed3b | ||||||
|   FirebaseInstallations: d4c7c958f99c8860d7fcece786314ae790e2f988 |   FirebaseCrashlytics: f83cbf176d5c637ade108c0aacf1ccbd5ec499bf | ||||||
|   FirebaseMessaging: af49f8d7c0a3d2a017d9302c80946f45a7777dde |   FirebaseInstallations: 3e884b01feabdf67582a80f3250425a00979b4ed | ||||||
|   FirebaseRemoteConfigInterop: bfa0ea72ba3dc5af739777296424e46bd6f42613 |   FirebaseMessaging: 43ec73bbfedd0c385a849bb91593ab4ad4b9e48e | ||||||
|   FirebaseSessions: 4e784acda213108aafef536535cdfc03504acc42 |   FirebaseRemoteConfigInterop: 0896fd52ab72586a355c8f389ff85aaa9e5375e1 | ||||||
|   Flutter: e0871f40cf51350855a761d2e70bf5af5b9b5de7 |   FirebaseSessions: f4692789e770bec66ce17d772c0e9561c4f11737 | ||||||
|  |   Flutter: cabc95a1d2626b1b06e7179b784ebcf0c0cde467 | ||||||
|   flutter_app_update: 816fdb2e30e4832a7c45e3f108d391c42ef040a9 |   flutter_app_update: 816fdb2e30e4832a7c45e3f108d391c42ef040a9 | ||||||
|   flutter_inappwebview_ios: b89ba3482b96fb25e00c967aae065701b66e9b99 |   flutter_inappwebview_ios: b89ba3482b96fb25e00c967aae065701b66e9b99 | ||||||
|   flutter_keyboard_visibility: 4625131e43015dbbe759d9b20daaf77e0e3f6619 |   flutter_keyboard_visibility: 4625131e43015dbbe759d9b20daaf77e0e3f6619 | ||||||
|  |   flutter_local_notifications: a5a732f069baa862e728d839dd2ebb904737effb | ||||||
|   flutter_native_splash: c32d145d68aeda5502d5f543ee38c192065986cf |   flutter_native_splash: c32d145d68aeda5502d5f543ee38c192065986cf | ||||||
|   flutter_platform_alert: bf3b5fcd4ac14bd637e20527e9c471633071afd3 |   flutter_platform_alert: bf3b5fcd4ac14bd637e20527e9c471633071afd3 | ||||||
|   flutter_secure_storage: 1ed9476fba7e7a782b22888f956cce43e2c62f13 |   flutter_secure_storage: 1ed9476fba7e7a782b22888f956cce43e2c62f13 | ||||||
|   flutter_timezone: 7c838e17ffd4645d261e87037e5bebf6d38fe544 |   flutter_timezone: 7c838e17ffd4645d261e87037e5bebf6d38fe544 | ||||||
|   flutter_udid: f7c3884e6ec2951efe4f9de082257fc77c4d15e9 |   flutter_udid: f7c3884e6ec2951efe4f9de082257fc77c4d15e9 | ||||||
|   flutter_webrtc: 6f7da106613d52ade777d5b4875a43f48c28b457 |   flutter_webrtc: b0b2e04411747142962164a1cfa43a1af9a0afac | ||||||
|   gal: baecd024ebfd13c441269ca7404792a7152fde89 |   gal: baecd024ebfd13c441269ca7404792a7152fde89 | ||||||
|   GoogleAdsOnDeviceConversion: 2be6297a4f048459e0ae17fad9bfd2844e10cf64 |   GoogleAdsOnDeviceConversion: 9090c435cde08903e8dd1ba2c77fbec9e46d9afe | ||||||
|   GoogleAppMeasurement: 8f6ab04ad6ae493b53fcf56bd26323fb2f1384f3 |   GoogleAppMeasurement: 09f341dfa8527d1612a09cbfe809a242c0b737af | ||||||
|   GoogleDataTransport: aae35b7ea0c09004c3797d53c8c41f66f219d6a7 |   GoogleDataTransport: aae35b7ea0c09004c3797d53c8c41f66f219d6a7 | ||||||
|   GoogleUtilities: 00c88b9a86066ef77f0da2fab05f65d7768ed8e1 |   GoogleUtilities: 00c88b9a86066ef77f0da2fab05f65d7768ed8e1 | ||||||
|   image_picker_ios: 7fe1ff8e34c1790d6fff70a32484959f563a928a |   image_picker_ios: 7fe1ff8e34c1790d6fff70a32484959f563a928a | ||||||
|   irondash_engine_context: 8e58ca8e0212ee9d1c7dc6a42121849986c88486 |   irondash_engine_context: 8e58ca8e0212ee9d1c7dc6a42121849986c88486 | ||||||
|   Kingfisher: ff0d31a1f07bdff6a1ebb3ba08b8e6e567b6500c |   Kingfisher: ff0d31a1f07bdff6a1ebb3ba08b8e6e567b6500c | ||||||
|   livekit_client: e3b79b99405428aac439b6b76a254cd9a11dbbfb |   livekit_client: f810c81bbbc229a84f60b09e66603ac4e93f7599 | ||||||
|   local_auth_darwin: d2e8c53ef0c4f43c646462e3415432c4dab3ae19 |   local_auth_darwin: d2e8c53ef0c4f43c646462e3415432c4dab3ae19 | ||||||
|   media_kit_libs_ios_video: 5a18affdb97d1f5d466dc79988b13eff6c5e2854 |   media_kit_libs_ios_video: 5a18affdb97d1f5d466dc79988b13eff6c5e2854 | ||||||
|   media_kit_video: 1746e198cb697d1ffb734b1d05ec429d1fcd1474 |   media_kit_video: 1746e198cb697d1ffb734b1d05ec429d1fcd1474 | ||||||
| @@ -512,7 +524,7 @@ SPEC CHECKSUMS: | |||||||
|   receive_sharing_intent: 222384f00ffe7e952bbfabaa9e3967cb87e5fe00 |   receive_sharing_intent: 222384f00ffe7e952bbfabaa9e3967cb87e5fe00 | ||||||
|   record_ios: f75fa1d57f840012775c0e93a38a7f3ceea1a374 |   record_ios: f75fa1d57f840012775c0e93a38a7f3ceea1a374 | ||||||
|   SAMKeychain: 483e1c9f32984d50ca961e26818a534283b4cd5c |   SAMKeychain: 483e1c9f32984d50ca961e26818a534283b4cd5c | ||||||
|   SDWebImage: f29024626962457f3470184232766516dee8dfea |   SDWebImage: 9f177d83116802728e122410fb25ad88f5c7608a | ||||||
|   share_plus: 50da8cb520a8f0f65671c6c6a99b3617ed10a58a |   share_plus: 50da8cb520a8f0f65671c6c6a99b3617ed10a58a | ||||||
|   shared_preferences_foundation: 9e1978ff2562383bd5676f64ec4e9aa8fa06a6f7 |   shared_preferences_foundation: 9e1978ff2562383bd5676f64ec4e9aa8fa06a6f7 | ||||||
|   sign_in_with_apple: c5dcc141574c8c54d5ac99dd2163c0c72ad22418 |   sign_in_with_apple: c5dcc141574c8c54d5ac99dd2163c0c72ad22418 | ||||||
| @@ -524,7 +536,7 @@ SPEC CHECKSUMS: | |||||||
|   url_launcher_ios: 694010445543906933d732453a59da0a173ae33d |   url_launcher_ios: 694010445543906933d732453a59da0a173ae33d | ||||||
|   volume_controller: 3657a1f65bedb98fa41ff7dc5793537919f31b12 |   volume_controller: 3657a1f65bedb98fa41ff7dc5793537919f31b12 | ||||||
|   wakelock_plus: e29112ab3ef0b318e58cfa5c32326458be66b556 |   wakelock_plus: e29112ab3ef0b318e58cfa5c32326458be66b556 | ||||||
|   WebRTC-SDK: d20de357dcbf7c9696b124b39f3ff62125107e4b |   WebRTC-SDK: 69d4e56b0b4b27d788e87bab9b9a1326ed05b1e3 | ||||||
|  |  | ||||||
| PODFILE CHECKSUM: c818292390b02fa379036ea099713a332bd7193f | PODFILE CHECKSUM: c818292390b02fa379036ea099713a332bd7193f | ||||||
|  |  | ||||||
|   | |||||||
| @@ -853,7 +853,7 @@ | |||||||
| 				GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE; | 				GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE; | ||||||
| 				GCC_WARN_UNUSED_FUNCTION = YES; | 				GCC_WARN_UNUSED_FUNCTION = YES; | ||||||
| 				GCC_WARN_UNUSED_VARIABLE = YES; | 				GCC_WARN_UNUSED_VARIABLE = YES; | ||||||
| 				IPHONEOS_DEPLOYMENT_TARGET = 12.0; | 				IPHONEOS_DEPLOYMENT_TARGET = 13.0; | ||||||
| 				MTL_ENABLE_DEBUG_INFO = NO; | 				MTL_ENABLE_DEBUG_INFO = NO; | ||||||
| 				SDKROOT = iphoneos; | 				SDKROOT = iphoneos; | ||||||
| 				SUPPORTED_PLATFORMS = iphoneos; | 				SUPPORTED_PLATFORMS = iphoneos; | ||||||
| @@ -897,6 +897,7 @@ | |||||||
| 				CODE_SIGN_STYLE = Automatic; | 				CODE_SIGN_STYLE = Automatic; | ||||||
| 				CURRENT_PROJECT_VERSION = 1; | 				CURRENT_PROJECT_VERSION = 1; | ||||||
| 				GENERATE_INFOPLIST_FILE = YES; | 				GENERATE_INFOPLIST_FILE = YES; | ||||||
|  | 				IPHONEOS_DEPLOYMENT_TARGET = 15.0; | ||||||
| 				MARKETING_VERSION = 1.0; | 				MARKETING_VERSION = 1.0; | ||||||
| 				PRODUCT_BUNDLE_IDENTIFIER = dev.solsynth.solian.RunnerTests; | 				PRODUCT_BUNDLE_IDENTIFIER = dev.solsynth.solian.RunnerTests; | ||||||
| 				PRODUCT_NAME = "$(TARGET_NAME)"; | 				PRODUCT_NAME = "$(TARGET_NAME)"; | ||||||
| @@ -915,6 +916,7 @@ | |||||||
| 				CODE_SIGN_STYLE = Automatic; | 				CODE_SIGN_STYLE = Automatic; | ||||||
| 				CURRENT_PROJECT_VERSION = 1; | 				CURRENT_PROJECT_VERSION = 1; | ||||||
| 				GENERATE_INFOPLIST_FILE = YES; | 				GENERATE_INFOPLIST_FILE = YES; | ||||||
|  | 				IPHONEOS_DEPLOYMENT_TARGET = 15.0; | ||||||
| 				MARKETING_VERSION = 1.0; | 				MARKETING_VERSION = 1.0; | ||||||
| 				PRODUCT_BUNDLE_IDENTIFIER = dev.solsynth.solian.RunnerTests; | 				PRODUCT_BUNDLE_IDENTIFIER = dev.solsynth.solian.RunnerTests; | ||||||
| 				PRODUCT_NAME = "$(TARGET_NAME)"; | 				PRODUCT_NAME = "$(TARGET_NAME)"; | ||||||
| @@ -931,6 +933,7 @@ | |||||||
| 				CODE_SIGN_STYLE = Automatic; | 				CODE_SIGN_STYLE = Automatic; | ||||||
| 				CURRENT_PROJECT_VERSION = 1; | 				CURRENT_PROJECT_VERSION = 1; | ||||||
| 				GENERATE_INFOPLIST_FILE = YES; | 				GENERATE_INFOPLIST_FILE = YES; | ||||||
|  | 				IPHONEOS_DEPLOYMENT_TARGET = 15.0; | ||||||
| 				MARKETING_VERSION = 1.0; | 				MARKETING_VERSION = 1.0; | ||||||
| 				PRODUCT_BUNDLE_IDENTIFIER = dev.solsynth.solian.RunnerTests; | 				PRODUCT_BUNDLE_IDENTIFIER = dev.solsynth.solian.RunnerTests; | ||||||
| 				PRODUCT_NAME = "$(TARGET_NAME)"; | 				PRODUCT_NAME = "$(TARGET_NAME)"; | ||||||
| @@ -1078,7 +1081,7 @@ | |||||||
| 				INFOPLIST_FILE = SolianShareExtension/Info.plist; | 				INFOPLIST_FILE = SolianShareExtension/Info.plist; | ||||||
| 				INFOPLIST_KEY_CFBundleDisplayName = SolianShareExtension; | 				INFOPLIST_KEY_CFBundleDisplayName = SolianShareExtension; | ||||||
| 				INFOPLIST_KEY_NSHumanReadableCopyright = ""; | 				INFOPLIST_KEY_NSHumanReadableCopyright = ""; | ||||||
| 				IPHONEOS_DEPLOYMENT_TARGET = 13.0; | 				IPHONEOS_DEPLOYMENT_TARGET = 15.0; | ||||||
| 				LD_RUNPATH_SEARCH_PATHS = ( | 				LD_RUNPATH_SEARCH_PATHS = ( | ||||||
| 					"$(inherited)", | 					"$(inherited)", | ||||||
| 					"@executable_path/Frameworks", | 					"@executable_path/Frameworks", | ||||||
| @@ -1121,7 +1124,7 @@ | |||||||
| 				INFOPLIST_FILE = SolianShareExtension/Info.plist; | 				INFOPLIST_FILE = SolianShareExtension/Info.plist; | ||||||
| 				INFOPLIST_KEY_CFBundleDisplayName = SolianShareExtension; | 				INFOPLIST_KEY_CFBundleDisplayName = SolianShareExtension; | ||||||
| 				INFOPLIST_KEY_NSHumanReadableCopyright = ""; | 				INFOPLIST_KEY_NSHumanReadableCopyright = ""; | ||||||
| 				IPHONEOS_DEPLOYMENT_TARGET = 13.0; | 				IPHONEOS_DEPLOYMENT_TARGET = 15.0; | ||||||
| 				LD_RUNPATH_SEARCH_PATHS = ( | 				LD_RUNPATH_SEARCH_PATHS = ( | ||||||
| 					"$(inherited)", | 					"$(inherited)", | ||||||
| 					"@executable_path/Frameworks", | 					"@executable_path/Frameworks", | ||||||
| @@ -1161,7 +1164,7 @@ | |||||||
| 				INFOPLIST_FILE = SolianShareExtension/Info.plist; | 				INFOPLIST_FILE = SolianShareExtension/Info.plist; | ||||||
| 				INFOPLIST_KEY_CFBundleDisplayName = SolianShareExtension; | 				INFOPLIST_KEY_CFBundleDisplayName = SolianShareExtension; | ||||||
| 				INFOPLIST_KEY_NSHumanReadableCopyright = ""; | 				INFOPLIST_KEY_NSHumanReadableCopyright = ""; | ||||||
| 				IPHONEOS_DEPLOYMENT_TARGET = 13.0; | 				IPHONEOS_DEPLOYMENT_TARGET = 15.0; | ||||||
| 				LD_RUNPATH_SEARCH_PATHS = ( | 				LD_RUNPATH_SEARCH_PATHS = ( | ||||||
| 					"$(inherited)", | 					"$(inherited)", | ||||||
| 					"@executable_path/Frameworks", | 					"@executable_path/Frameworks", | ||||||
| @@ -1348,7 +1351,7 @@ | |||||||
| 				GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE; | 				GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE; | ||||||
| 				GCC_WARN_UNUSED_FUNCTION = YES; | 				GCC_WARN_UNUSED_FUNCTION = YES; | ||||||
| 				GCC_WARN_UNUSED_VARIABLE = YES; | 				GCC_WARN_UNUSED_VARIABLE = YES; | ||||||
| 				IPHONEOS_DEPLOYMENT_TARGET = 12.0; | 				IPHONEOS_DEPLOYMENT_TARGET = 13.0; | ||||||
| 				MTL_ENABLE_DEBUG_INFO = YES; | 				MTL_ENABLE_DEBUG_INFO = YES; | ||||||
| 				ONLY_ACTIVE_ARCH = YES; | 				ONLY_ACTIVE_ARCH = YES; | ||||||
| 				SDKROOT = iphoneos; | 				SDKROOT = iphoneos; | ||||||
| @@ -1399,7 +1402,7 @@ | |||||||
| 				GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE; | 				GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE; | ||||||
| 				GCC_WARN_UNUSED_FUNCTION = YES; | 				GCC_WARN_UNUSED_FUNCTION = YES; | ||||||
| 				GCC_WARN_UNUSED_VARIABLE = YES; | 				GCC_WARN_UNUSED_VARIABLE = YES; | ||||||
| 				IPHONEOS_DEPLOYMENT_TARGET = 12.0; | 				IPHONEOS_DEPLOYMENT_TARGET = 13.0; | ||||||
| 				MTL_ENABLE_DEBUG_INFO = NO; | 				MTL_ENABLE_DEBUG_INFO = NO; | ||||||
| 				SDKROOT = iphoneos; | 				SDKROOT = iphoneos; | ||||||
| 				SUPPORTED_PLATFORMS = iphoneos; | 				SUPPORTED_PLATFORMS = iphoneos; | ||||||
|   | |||||||
| @@ -47,6 +47,7 @@ class NotificationService: UNNotificationServiceExtension { | |||||||
|     private func processNotification(request: UNNotificationRequest, content: UNMutableNotificationContent) throws { |     private func processNotification(request: UNNotificationRequest, content: UNMutableNotificationContent) throws { | ||||||
|         switch content.userInfo["type"] as? String { |         switch content.userInfo["type"] as? String { | ||||||
|         case "messages.new": |         case "messages.new": | ||||||
|  |             content.categoryIdentifier = "REPLYABLE_MESSAGE" | ||||||
|             try handleMessagingNotification(request: request, content: content) |             try handleMessagingNotification(request: request, content: content) | ||||||
|         default: |         default: | ||||||
|             try handleDefaultNotification(content: content) |             try handleDefaultNotification(content: content) | ||||||
| @@ -60,8 +61,6 @@ class NotificationService: UNNotificationServiceExtension { | |||||||
|          |          | ||||||
|         let pfpIdentifier = meta["pfp"] as? String |         let pfpIdentifier = meta["pfp"] as? String | ||||||
|          |          | ||||||
|         content.categoryIdentifier = "REPLYABLE_MESSAGE" |  | ||||||
|          |  | ||||||
|         let metaCopy = meta as? [String: Any] ?? [:] |         let metaCopy = meta as? [String: Any] ?? [:] | ||||||
|         let pfpUrl = pfpIdentifier != nil ? getAttachmentUrl(for: pfpIdentifier!) : nil |         let pfpUrl = pfpIdentifier != nil ? getAttachmentUrl(for: pfpIdentifier!) : nil | ||||||
|          |          | ||||||
|   | |||||||
| @@ -68,6 +68,34 @@ class AppDatabase extends _$AppDatabase { | |||||||
|     return (delete(chatMessages)..where((m) => m.id.equals(id))).go(); |     return (delete(chatMessages)..where((m) => m.id.equals(id))).go(); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   Future<int> getTotalMessagesForRoom(String roomId) { | ||||||
|  |     return (select(chatMessages)..where((m) => m.roomId.equals(roomId))).get().then((list) => list.length); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   Future<List<LocalChatMessage>> searchMessages( | ||||||
|  |     String roomId, | ||||||
|  |     String query, | ||||||
|  |   ) async { | ||||||
|  |     var selectStatement = select(chatMessages) | ||||||
|  |       ..where((m) => m.roomId.equals(roomId)); | ||||||
|  |  | ||||||
|  |     if (query.isNotEmpty) { | ||||||
|  |       selectStatement = | ||||||
|  |           selectStatement | ||||||
|  |             ..where((m) => m.content.like('%${query.toLowerCase()}%')); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |      | ||||||
|  |  | ||||||
|  |      | ||||||
|  |  | ||||||
|  |     final messages = | ||||||
|  |         await (selectStatement | ||||||
|  |               ..orderBy([(m) => OrderingTerm.desc(m.createdAt)])) | ||||||
|  |             .get(); | ||||||
|  |     return messages.map((msg) => companionToMessage(msg)).toList(); | ||||||
|  |   } | ||||||
|  |  | ||||||
|   // Convert between Drift and model objects |   // Convert between Drift and model objects | ||||||
|   ChatMessagesCompanion messageToCompanion(LocalChatMessage message) { |   ChatMessagesCompanion messageToCompanion(LocalChatMessage message) { | ||||||
|     return ChatMessagesCompanion( |     return ChatMessagesCompanion( | ||||||
|   | |||||||
| @@ -8,7 +8,7 @@ import 'package:firebase_crashlytics/firebase_crashlytics.dart'; | |||||||
| import 'package:firebase_messaging/firebase_messaging.dart'; | import 'package:firebase_messaging/firebase_messaging.dart'; | ||||||
| import 'package:flutter/foundation.dart'; | import 'package:flutter/foundation.dart'; | ||||||
| import 'package:flutter/material.dart'; | import 'package:flutter/material.dart'; | ||||||
| import 'package:flutter/services.dart'; |  | ||||||
| import 'package:flutter_hooks/flutter_hooks.dart'; | import 'package:flutter_hooks/flutter_hooks.dart'; | ||||||
| import 'package:go_router/go_router.dart'; | import 'package:go_router/go_router.dart'; | ||||||
| import 'package:hooks_riverpod/hooks_riverpod.dart'; | import 'package:hooks_riverpod/hooks_riverpod.dart'; | ||||||
| @@ -30,7 +30,6 @@ import 'package:shared_preferences/shared_preferences.dart'; | |||||||
| import 'package:image_picker_platform_interface/image_picker_platform_interface.dart'; | import 'package:image_picker_platform_interface/image_picker_platform_interface.dart'; | ||||||
| import 'package:flutter_native_splash/flutter_native_splash.dart'; | import 'package:flutter_native_splash/flutter_native_splash.dart'; | ||||||
| import 'package:url_launcher/url_launcher_string.dart'; | import 'package:url_launcher/url_launcher_string.dart'; | ||||||
| import 'package:flutter_langdetect/flutter_langdetect.dart' as langdetect; |  | ||||||
|  |  | ||||||
| @pragma('vm:entry-point') | @pragma('vm:entry-point') | ||||||
| Future<void> _firebaseMessagingBackgroundHandler(RemoteMessage message) async { | Future<void> _firebaseMessagingBackgroundHandler(RemoteMessage message) async { | ||||||
| @@ -52,7 +51,6 @@ void main() async { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   try { |   try { | ||||||
|     await langdetect.initLangDetect(); |  | ||||||
|     await EasyLocalization.ensureInitialized(); |     await EasyLocalization.ensureInitialized(); | ||||||
|  |  | ||||||
|     if (kIsWeb || !Platform.isLinux) { |     if (kIsWeb || !Platform.isLinux) { | ||||||
| @@ -169,12 +167,12 @@ class IslandApp extends HookConsumerWidget { | |||||||
|     final theme = ref.watch(themeProvider); |     final theme = ref.watch(themeProvider); | ||||||
|  |  | ||||||
|     void handleMessage(RemoteMessage notification) { |     void handleMessage(RemoteMessage notification) { | ||||||
|       if (notification.data['action_uri'] != null) { |       if (notification.data['meta']?['action_uri'] != null) { | ||||||
|         var uri = notification.data['action_uri'] as String; |         var uri = notification.data['meta']['action_uri'] as String; | ||||||
|         if (uri.startsWith('/')) { |         if (uri.startsWith('/')) { | ||||||
|           // In-app routes |           // In-app routes | ||||||
|           final router = ref.read(routerProvider); |           final router = ref.read(routerProvider); | ||||||
|           router.go(notification.data['action_uri']); |           router.push(notification.data['meta']['action_uri']); | ||||||
|         } else { |         } else { | ||||||
|           // External links |           // External links | ||||||
|           launchUrlString(uri); |           launchUrlString(uri); | ||||||
| @@ -186,27 +184,6 @@ class IslandApp extends HookConsumerWidget { | |||||||
|       if (!kIsWeb && Platform.isLinux) { |       if (!kIsWeb && Platform.isLinux) { | ||||||
|         return null; |         return null; | ||||||
|       } |       } | ||||||
|       const channel = MethodChannel('dev.solsynth.solian/notifications'); |  | ||||||
|  |  | ||||||
|       Future<void> handleInitialLink() async { |  | ||||||
|         final String? link = await channel.invokeMethod('initialLink'); |  | ||||||
|         if (link != null) { |  | ||||||
|           final router = ref.read(routerProvider); |  | ||||||
|           router.go(link); |  | ||||||
|         } |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       if (!kIsWeb && Platform.isAndroid) { |  | ||||||
|         handleInitialLink(); |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       channel.setMethodCallHandler((call) async { |  | ||||||
|         if (call.method == 'newLink') { |  | ||||||
|           final String link = call.arguments; |  | ||||||
|           final router = ref.read(routerProvider); |  | ||||||
|           router.go(link); |  | ||||||
|         } |  | ||||||
|       }); |  | ||||||
|  |  | ||||||
|       // When the app is opened from a terminated state. |       // When the app is opened from a terminated state. | ||||||
|       FirebaseMessaging.instance.getInitialMessage().then((message) { |       FirebaseMessaging.instance.getInitialMessage().then((message) { | ||||||
| @@ -246,6 +223,7 @@ class IslandApp extends HookConsumerWidget { | |||||||
|           if (user.value != null) { |           if (user.value != null) { | ||||||
|             final apiClient = ref.read(apiClientProvider); |             final apiClient = ref.read(apiClientProvider); | ||||||
|             subscribePushNotification(apiClient); |             subscribePushNotification(apiClient); | ||||||
|  |             initializeLocalNotifications(); | ||||||
|             final wsNotifier = ref.read(websocketStateProvider.notifier); |             final wsNotifier = ref.read(websocketStateProvider.notifier); | ||||||
|             wsNotifier.connect(); |             wsNotifier.connect(); | ||||||
|           } |           } | ||||||
| @@ -262,6 +240,7 @@ class IslandApp extends HookConsumerWidget { | |||||||
|       themeMode: ThemeMode.system, |       themeMode: ThemeMode.system, | ||||||
|       routerConfig: router, |       routerConfig: router, | ||||||
|       supportedLocales: context.supportedLocales, |       supportedLocales: context.supportedLocales, | ||||||
|  |       scrollBehavior: AppScrollBehavior(), | ||||||
|       localizationsDelegates: [ |       localizationsDelegates: [ | ||||||
|         ...context.localizationDelegates, |         ...context.localizationDelegates, | ||||||
|         CroppyLocalizations.delegate, |         CroppyLocalizations.delegate, | ||||||
|   | |||||||
| @@ -13,6 +13,7 @@ sealed class SnAccount with _$SnAccount { | |||||||
|     required String name, |     required String name, | ||||||
|     required String nick, |     required String nick, | ||||||
|     required String language, |     required String language, | ||||||
|  |     @Default("") String region, | ||||||
|     required bool isSuperuser, |     required bool isSuperuser, | ||||||
|     required String? automatedId, |     required String? automatedId, | ||||||
|     required SnAccountProfile profile, |     required SnAccountProfile profile, | ||||||
| @@ -71,6 +72,8 @@ sealed class SnAccountProfile with _$SnAccountProfile { | |||||||
|     SnAccountBadge? activeBadge, |     SnAccountBadge? activeBadge, | ||||||
|     required int experience, |     required int experience, | ||||||
|     required int level, |     required int level, | ||||||
|  |     @Default(100) double socialCredits, | ||||||
|  |     @Default(0) int socialCreditsLevel, | ||||||
|     required double levelingProgress, |     required double levelingProgress, | ||||||
|     required SnCloudFile? picture, |     required SnCloudFile? picture, | ||||||
|     required SnCloudFile? background, |     required SnCloudFile? background, | ||||||
|   | |||||||
| @@ -15,7 +15,7 @@ T _$identity<T>(T value) => value; | |||||||
| /// @nodoc | /// @nodoc | ||||||
| mixin _$SnAccount { | mixin _$SnAccount { | ||||||
|  |  | ||||||
|  String get id; String get name; String get nick; String get language; bool get isSuperuser; String? get automatedId; SnAccountProfile get profile; SnWalletSubscriptionRef? get perkSubscription; 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; String get region; bool get isSuperuser; String? get automatedId; SnAccountProfile get profile; SnWalletSubscriptionRef? get perkSubscription; List<SnAccountBadge> get badges; DateTime get createdAt; DateTime get updatedAt; DateTime? get deletedAt; | ||||||
| /// Create a copy of SnAccount | /// Create a copy of SnAccount | ||||||
| /// with the given fields replaced by the non-null parameter values. | /// with the given fields replaced by the non-null parameter values. | ||||||
| @JsonKey(includeFromJson: false, includeToJson: false) | @JsonKey(includeFromJson: false, includeToJson: false) | ||||||
| @@ -28,16 +28,16 @@ $SnAccountCopyWith<SnAccount> get copyWith => _$SnAccountCopyWithImpl<SnAccount> | |||||||
|  |  | ||||||
| @override | @override | ||||||
| bool operator ==(Object other) { | 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.automatedId, automatedId) || other.automatedId == automatedId)&&(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)); |   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.region, region) || other.region == region)&&(identical(other.isSuperuser, isSuperuser) || other.isSuperuser == isSuperuser)&&(identical(other.automatedId, automatedId) || other.automatedId == automatedId)&&(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) | @JsonKey(includeFromJson: false, includeToJson: false) | ||||||
| @override | @override | ||||||
| int get hashCode => Object.hash(runtimeType,id,name,nick,language,isSuperuser,automatedId,profile,perkSubscription,const DeepCollectionEquality().hash(badges),createdAt,updatedAt,deletedAt); | int get hashCode => Object.hash(runtimeType,id,name,nick,language,region,isSuperuser,automatedId,profile,perkSubscription,const DeepCollectionEquality().hash(badges),createdAt,updatedAt,deletedAt); | ||||||
|  |  | ||||||
| @override | @override | ||||||
| String toString() { | String toString() { | ||||||
|   return 'SnAccount(id: $id, name: $name, nick: $nick, language: $language, isSuperuser: $isSuperuser, automatedId: $automatedId, profile: $profile, perkSubscription: $perkSubscription, badges: $badges, createdAt: $createdAt, updatedAt: $updatedAt, deletedAt: $deletedAt)'; |   return 'SnAccount(id: $id, name: $name, nick: $nick, language: $language, region: $region, isSuperuser: $isSuperuser, automatedId: $automatedId, profile: $profile, perkSubscription: $perkSubscription, badges: $badges, createdAt: $createdAt, updatedAt: $updatedAt, deletedAt: $deletedAt)'; | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -48,7 +48,7 @@ abstract mixin class $SnAccountCopyWith<$Res>  { | |||||||
|   factory $SnAccountCopyWith(SnAccount value, $Res Function(SnAccount) _then) = _$SnAccountCopyWithImpl; |   factory $SnAccountCopyWith(SnAccount value, $Res Function(SnAccount) _then) = _$SnAccountCopyWithImpl; | ||||||
| @useResult | @useResult | ||||||
| $Res call({ | $Res call({ | ||||||
|  String id, String name, String nick, String language, bool isSuperuser, String? automatedId, SnAccountProfile profile, SnWalletSubscriptionRef? perkSubscription, List<SnAccountBadge> badges, DateTime createdAt, DateTime updatedAt, DateTime? deletedAt |  String id, String name, String nick, String language, String region, bool isSuperuser, String? automatedId, SnAccountProfile profile, SnWalletSubscriptionRef? perkSubscription, List<SnAccountBadge> badges, DateTime createdAt, DateTime updatedAt, DateTime? deletedAt | ||||||
| }); | }); | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -65,12 +65,13 @@ class _$SnAccountCopyWithImpl<$Res> | |||||||
|  |  | ||||||
| /// Create a copy of SnAccount | /// Create a copy of SnAccount | ||||||
| /// with the given fields replaced by the non-null parameter values. | /// 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? automatedId = freezed,Object? profile = null,Object? perkSubscription = freezed,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? region = null,Object? isSuperuser = null,Object? automatedId = freezed,Object? profile = null,Object? perkSubscription = freezed,Object? badges = null,Object? createdAt = null,Object? updatedAt = null,Object? deletedAt = freezed,}) { | ||||||
|   return _then(_self.copyWith( |   return _then(_self.copyWith( | ||||||
| id: null == id ? _self.id : id // ignore: cast_nullable_to_non_nullable | 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,name: null == name ? _self.name : name // ignore: cast_nullable_to_non_nullable | ||||||
| as String,nick: null == nick ? _self.nick : nick // ignore: cast_nullable_to_non_nullable | as String,nick: null == nick ? _self.nick : nick // ignore: cast_nullable_to_non_nullable | ||||||
| as String,language: null == language ? _self.language : language // ignore: cast_nullable_to_non_nullable | as String,language: null == language ? _self.language : language // ignore: cast_nullable_to_non_nullable | ||||||
|  | as String,region: null == region ? _self.region : region // ignore: cast_nullable_to_non_nullable | ||||||
| as String,isSuperuser: null == isSuperuser ? _self.isSuperuser : isSuperuser // ignore: cast_nullable_to_non_nullable | as String,isSuperuser: null == isSuperuser ? _self.isSuperuser : isSuperuser // ignore: cast_nullable_to_non_nullable | ||||||
| as bool,automatedId: freezed == automatedId ? _self.automatedId : automatedId // ignore: cast_nullable_to_non_nullable | as bool,automatedId: freezed == automatedId ? _self.automatedId : automatedId // ignore: cast_nullable_to_non_nullable | ||||||
| as String?,profile: null == profile ? _self.profile : profile // ignore: cast_nullable_to_non_nullable | as String?,profile: null == profile ? _self.profile : profile // ignore: cast_nullable_to_non_nullable | ||||||
| @@ -182,10 +183,10 @@ return $default(_that);case _: | |||||||
| /// } | /// } | ||||||
| /// ``` | /// ``` | ||||||
|  |  | ||||||
| @optionalTypeArgs TResult maybeWhen<TResult extends Object?>(TResult Function( String id,  String name,  String nick,  String language,  bool isSuperuser,  String? automatedId,  SnAccountProfile profile,  SnWalletSubscriptionRef? perkSubscription,  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,  String region,  bool isSuperuser,  String? automatedId,  SnAccountProfile profile,  SnWalletSubscriptionRef? perkSubscription,  List<SnAccountBadge> badges,  DateTime createdAt,  DateTime updatedAt,  DateTime? deletedAt)?  $default,{required TResult orElse(),}) {final _that = this; | ||||||
| switch (_that) { | switch (_that) { | ||||||
| case _SnAccount() when $default != null: | case _SnAccount() when $default != null: | ||||||
| return $default(_that.id,_that.name,_that.nick,_that.language,_that.isSuperuser,_that.automatedId,_that.profile,_that.perkSubscription,_that.badges,_that.createdAt,_that.updatedAt,_that.deletedAt);case _: | return $default(_that.id,_that.name,_that.nick,_that.language,_that.region,_that.isSuperuser,_that.automatedId,_that.profile,_that.perkSubscription,_that.badges,_that.createdAt,_that.updatedAt,_that.deletedAt);case _: | ||||||
|   return orElse(); |   return orElse(); | ||||||
|  |  | ||||||
| } | } | ||||||
| @@ -203,10 +204,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,  String? automatedId,  SnAccountProfile profile,  SnWalletSubscriptionRef? perkSubscription,  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,  String region,  bool isSuperuser,  String? automatedId,  SnAccountProfile profile,  SnWalletSubscriptionRef? perkSubscription,  List<SnAccountBadge> badges,  DateTime createdAt,  DateTime updatedAt,  DateTime? deletedAt)  $default,) {final _that = this; | ||||||
| switch (_that) { | switch (_that) { | ||||||
| case _SnAccount(): | case _SnAccount(): | ||||||
| return $default(_that.id,_that.name,_that.nick,_that.language,_that.isSuperuser,_that.automatedId,_that.profile,_that.perkSubscription,_that.badges,_that.createdAt,_that.updatedAt,_that.deletedAt);} | return $default(_that.id,_that.name,_that.nick,_that.language,_that.region,_that.isSuperuser,_that.automatedId,_that.profile,_that.perkSubscription,_that.badges,_that.createdAt,_that.updatedAt,_that.deletedAt);} | ||||||
| } | } | ||||||
| /// A variant of `when` that fallback to returning `null` | /// A variant of `when` that fallback to returning `null` | ||||||
| /// | /// | ||||||
| @@ -220,10 +221,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,  String? automatedId,  SnAccountProfile profile,  SnWalletSubscriptionRef? perkSubscription,  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,  String region,  bool isSuperuser,  String? automatedId,  SnAccountProfile profile,  SnWalletSubscriptionRef? perkSubscription,  List<SnAccountBadge> badges,  DateTime createdAt,  DateTime updatedAt,  DateTime? deletedAt)?  $default,) {final _that = this; | ||||||
| switch (_that) { | switch (_that) { | ||||||
| case _SnAccount() when $default != null: | case _SnAccount() when $default != null: | ||||||
| return $default(_that.id,_that.name,_that.nick,_that.language,_that.isSuperuser,_that.automatedId,_that.profile,_that.perkSubscription,_that.badges,_that.createdAt,_that.updatedAt,_that.deletedAt);case _: | return $default(_that.id,_that.name,_that.nick,_that.language,_that.region,_that.isSuperuser,_that.automatedId,_that.profile,_that.perkSubscription,_that.badges,_that.createdAt,_that.updatedAt,_that.deletedAt);case _: | ||||||
|   return null; |   return null; | ||||||
|  |  | ||||||
| } | } | ||||||
| @@ -235,13 +236,14 @@ return $default(_that.id,_that.name,_that.nick,_that.language,_that.isSuperuser, | |||||||
| @JsonSerializable() | @JsonSerializable() | ||||||
|  |  | ||||||
| class _SnAccount implements SnAccount { | class _SnAccount implements SnAccount { | ||||||
|   const _SnAccount({required this.id, required this.name, required this.nick, required this.language, required this.isSuperuser, required this.automatedId, required this.profile, required this.perkSubscription, 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, this.region = "", required this.isSuperuser, required this.automatedId, 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); |   factory _SnAccount.fromJson(Map<String, dynamic> json) => _$SnAccountFromJson(json); | ||||||
|  |  | ||||||
| @override final  String id; | @override final  String id; | ||||||
| @override final  String name; | @override final  String name; | ||||||
| @override final  String nick; | @override final  String nick; | ||||||
| @override final  String language; | @override final  String language; | ||||||
|  | @override@JsonKey() final  String region; | ||||||
| @override final  bool isSuperuser; | @override final  bool isSuperuser; | ||||||
| @override final  String? automatedId; | @override final  String? automatedId; | ||||||
| @override final  SnAccountProfile profile; | @override final  SnAccountProfile profile; | ||||||
| @@ -270,16 +272,16 @@ Map<String, dynamic> toJson() { | |||||||
|  |  | ||||||
| @override | @override | ||||||
| bool operator ==(Object other) { | 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.automatedId, automatedId) || other.automatedId == automatedId)&&(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)); |   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.region, region) || other.region == region)&&(identical(other.isSuperuser, isSuperuser) || other.isSuperuser == isSuperuser)&&(identical(other.automatedId, automatedId) || other.automatedId == automatedId)&&(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) | @JsonKey(includeFromJson: false, includeToJson: false) | ||||||
| @override | @override | ||||||
| int get hashCode => Object.hash(runtimeType,id,name,nick,language,isSuperuser,automatedId,profile,perkSubscription,const DeepCollectionEquality().hash(_badges),createdAt,updatedAt,deletedAt); | int get hashCode => Object.hash(runtimeType,id,name,nick,language,region,isSuperuser,automatedId,profile,perkSubscription,const DeepCollectionEquality().hash(_badges),createdAt,updatedAt,deletedAt); | ||||||
|  |  | ||||||
| @override | @override | ||||||
| String toString() { | String toString() { | ||||||
|   return 'SnAccount(id: $id, name: $name, nick: $nick, language: $language, isSuperuser: $isSuperuser, automatedId: $automatedId, profile: $profile, perkSubscription: $perkSubscription, badges: $badges, createdAt: $createdAt, updatedAt: $updatedAt, deletedAt: $deletedAt)'; |   return 'SnAccount(id: $id, name: $name, nick: $nick, language: $language, region: $region, isSuperuser: $isSuperuser, automatedId: $automatedId, profile: $profile, perkSubscription: $perkSubscription, badges: $badges, createdAt: $createdAt, updatedAt: $updatedAt, deletedAt: $deletedAt)'; | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -290,7 +292,7 @@ abstract mixin class _$SnAccountCopyWith<$Res> implements $SnAccountCopyWith<$Re | |||||||
|   factory _$SnAccountCopyWith(_SnAccount value, $Res Function(_SnAccount) _then) = __$SnAccountCopyWithImpl; |   factory _$SnAccountCopyWith(_SnAccount value, $Res Function(_SnAccount) _then) = __$SnAccountCopyWithImpl; | ||||||
| @override @useResult | @override @useResult | ||||||
| $Res call({ | $Res call({ | ||||||
|  String id, String name, String nick, String language, bool isSuperuser, String? automatedId, SnAccountProfile profile, SnWalletSubscriptionRef? perkSubscription, List<SnAccountBadge> badges, DateTime createdAt, DateTime updatedAt, DateTime? deletedAt |  String id, String name, String nick, String language, String region, bool isSuperuser, String? automatedId, SnAccountProfile profile, SnWalletSubscriptionRef? perkSubscription, List<SnAccountBadge> badges, DateTime createdAt, DateTime updatedAt, DateTime? deletedAt | ||||||
| }); | }); | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -307,12 +309,13 @@ class __$SnAccountCopyWithImpl<$Res> | |||||||
|  |  | ||||||
| /// Create a copy of SnAccount | /// Create a copy of SnAccount | ||||||
| /// with the given fields replaced by the non-null parameter values. | /// 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? automatedId = freezed,Object? profile = null,Object? perkSubscription = freezed,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? region = null,Object? isSuperuser = null,Object? automatedId = freezed,Object? profile = null,Object? perkSubscription = freezed,Object? badges = null,Object? createdAt = null,Object? updatedAt = null,Object? deletedAt = freezed,}) { | ||||||
|   return _then(_SnAccount( |   return _then(_SnAccount( | ||||||
| id: null == id ? _self.id : id // ignore: cast_nullable_to_non_nullable | 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,name: null == name ? _self.name : name // ignore: cast_nullable_to_non_nullable | ||||||
| as String,nick: null == nick ? _self.nick : nick // ignore: cast_nullable_to_non_nullable | as String,nick: null == nick ? _self.nick : nick // ignore: cast_nullable_to_non_nullable | ||||||
| as String,language: null == language ? _self.language : language // ignore: cast_nullable_to_non_nullable | as String,language: null == language ? _self.language : language // ignore: cast_nullable_to_non_nullable | ||||||
|  | as String,region: null == region ? _self.region : region // ignore: cast_nullable_to_non_nullable | ||||||
| as String,isSuperuser: null == isSuperuser ? _self.isSuperuser : isSuperuser // ignore: cast_nullable_to_non_nullable | as String,isSuperuser: null == isSuperuser ? _self.isSuperuser : isSuperuser // ignore: cast_nullable_to_non_nullable | ||||||
| as bool,automatedId: freezed == automatedId ? _self.automatedId : automatedId // ignore: cast_nullable_to_non_nullable | as bool,automatedId: freezed == automatedId ? _self.automatedId : automatedId // ignore: cast_nullable_to_non_nullable | ||||||
| as String?,profile: null == profile ? _self.profile : profile // ignore: cast_nullable_to_non_nullable | as String?,profile: null == profile ? _self.profile : profile // ignore: cast_nullable_to_non_nullable | ||||||
| @@ -613,7 +616,7 @@ as String, | |||||||
| /// @nodoc | /// @nodoc | ||||||
| mixin _$SnAccountProfile { | 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;@ProfileLinkConverter() List<ProfileLink> get links; 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; |  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;@ProfileLinkConverter() List<ProfileLink> get links; DateTime? get lastSeenAt; SnAccountBadge? get activeBadge; int get experience; int get level; double get socialCredits; int get socialCreditsLevel; 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 | /// Create a copy of SnAccountProfile | ||||||
| /// with the given fields replaced by the non-null parameter values. | /// with the given fields replaced by the non-null parameter values. | ||||||
| @JsonKey(includeFromJson: false, includeToJson: false) | @JsonKey(includeFromJson: false, includeToJson: false) | ||||||
| @@ -626,16 +629,16 @@ $SnAccountProfileCopyWith<SnAccountProfile> get copyWith => _$SnAccountProfileCo | |||||||
|  |  | ||||||
| @override | @override | ||||||
| bool operator ==(Object other) { | 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)&&const DeepCollectionEquality().equals(other.links, links)&&(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)); |   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)&&const DeepCollectionEquality().equals(other.links, links)&&(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.socialCredits, socialCredits) || other.socialCredits == socialCredits)&&(identical(other.socialCreditsLevel, socialCreditsLevel) || other.socialCreditsLevel == socialCreditsLevel)&&(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) | @JsonKey(includeFromJson: false, includeToJson: false) | ||||||
| @override | @override | ||||||
| int get hashCode => Object.hashAll([runtimeType,id,firstName,middleName,lastName,bio,gender,pronouns,location,timeZone,birthday,const DeepCollectionEquality().hash(links),lastSeenAt,activeBadge,experience,level,levelingProgress,picture,background,verification,createdAt,updatedAt,deletedAt]); | int get hashCode => Object.hashAll([runtimeType,id,firstName,middleName,lastName,bio,gender,pronouns,location,timeZone,birthday,const DeepCollectionEquality().hash(links),lastSeenAt,activeBadge,experience,level,socialCredits,socialCreditsLevel,levelingProgress,picture,background,verification,createdAt,updatedAt,deletedAt]); | ||||||
|  |  | ||||||
| @override | @override | ||||||
| String toString() { | String toString() { | ||||||
|   return 'SnAccountProfile(id: $id, firstName: $firstName, middleName: $middleName, lastName: $lastName, bio: $bio, gender: $gender, pronouns: $pronouns, location: $location, timeZone: $timeZone, birthday: $birthday, links: $links, lastSeenAt: $lastSeenAt, activeBadge: $activeBadge, experience: $experience, level: $level, levelingProgress: $levelingProgress, picture: $picture, background: $background, verification: $verification, 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, links: $links, lastSeenAt: $lastSeenAt, activeBadge: $activeBadge, experience: $experience, level: $level, socialCredits: $socialCredits, socialCreditsLevel: $socialCreditsLevel, levelingProgress: $levelingProgress, picture: $picture, background: $background, verification: $verification, createdAt: $createdAt, updatedAt: $updatedAt, deletedAt: $deletedAt)'; | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -646,7 +649,7 @@ abstract mixin class $SnAccountProfileCopyWith<$Res>  { | |||||||
|   factory $SnAccountProfileCopyWith(SnAccountProfile value, $Res Function(SnAccountProfile) _then) = _$SnAccountProfileCopyWithImpl; |   factory $SnAccountProfileCopyWith(SnAccountProfile value, $Res Function(SnAccountProfile) _then) = _$SnAccountProfileCopyWithImpl; | ||||||
| @useResult | @useResult | ||||||
| $Res call({ | $Res call({ | ||||||
|  String id, String firstName, String middleName, String lastName, String bio, String gender, String pronouns, String location, String timeZone, DateTime? birthday,@ProfileLinkConverter() List<ProfileLink> links, DateTime? lastSeenAt, SnAccountBadge? activeBadge, int experience, int level, double levelingProgress, SnCloudFile? picture, SnCloudFile? background, SnVerificationMark? verification, 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,@ProfileLinkConverter() List<ProfileLink> links, DateTime? lastSeenAt, SnAccountBadge? activeBadge, int experience, int level, double socialCredits, int socialCreditsLevel, double levelingProgress, SnCloudFile? picture, SnCloudFile? background, SnVerificationMark? verification, DateTime createdAt, DateTime updatedAt, DateTime? deletedAt | ||||||
| }); | }); | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -663,7 +666,7 @@ class _$SnAccountProfileCopyWithImpl<$Res> | |||||||
|  |  | ||||||
| /// Create a copy of SnAccountProfile | /// Create a copy of SnAccountProfile | ||||||
| /// with the given fields replaced by the non-null parameter values. | /// 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? links = null,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,}) { | @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? links = null,Object? lastSeenAt = freezed,Object? activeBadge = freezed,Object? experience = null,Object? level = null,Object? socialCredits = null,Object? socialCreditsLevel = 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( |   return _then(_self.copyWith( | ||||||
| id: null == id ? _self.id : id // ignore: cast_nullable_to_non_nullable | 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 | as String,firstName: null == firstName ? _self.firstName : firstName // ignore: cast_nullable_to_non_nullable | ||||||
| @@ -680,6 +683,8 @@ as List<ProfileLink>,lastSeenAt: freezed == lastSeenAt ? _self.lastSeenAt : last | |||||||
| as DateTime?,activeBadge: freezed == activeBadge ? _self.activeBadge : activeBadge // ignore: cast_nullable_to_non_nullable | as DateTime?,activeBadge: freezed == activeBadge ? _self.activeBadge : activeBadge // ignore: cast_nullable_to_non_nullable | ||||||
| as SnAccountBadge?,experience: null == experience ? _self.experience : experience // ignore: cast_nullable_to_non_nullable | as SnAccountBadge?,experience: null == experience ? _self.experience : experience // ignore: cast_nullable_to_non_nullable | ||||||
| as int,level: null == level ? _self.level : level // ignore: cast_nullable_to_non_nullable | as int,level: null == level ? _self.level : level // ignore: cast_nullable_to_non_nullable | ||||||
|  | as int,socialCredits: null == socialCredits ? _self.socialCredits : socialCredits // ignore: cast_nullable_to_non_nullable | ||||||
|  | as double,socialCreditsLevel: null == socialCreditsLevel ? _self.socialCreditsLevel : socialCreditsLevel // ignore: cast_nullable_to_non_nullable | ||||||
| as int,levelingProgress: null == levelingProgress ? _self.levelingProgress : levelingProgress // ignore: cast_nullable_to_non_nullable | as int,levelingProgress: null == levelingProgress ? _self.levelingProgress : levelingProgress // ignore: cast_nullable_to_non_nullable | ||||||
| as double,picture: freezed == picture ? _self.picture : picture // ignore: cast_nullable_to_non_nullable | 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?,background: freezed == background ? _self.background : background // ignore: cast_nullable_to_non_nullable | ||||||
| @@ -817,10 +822,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, @ProfileLinkConverter()  List<ProfileLink> links,  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; | @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, @ProfileLinkConverter()  List<ProfileLink> links,  DateTime? lastSeenAt,  SnAccountBadge? activeBadge,  int experience,  int level,  double socialCredits,  int socialCreditsLevel,  double levelingProgress,  SnCloudFile? picture,  SnCloudFile? background,  SnVerificationMark? verification,  DateTime createdAt,  DateTime updatedAt,  DateTime? deletedAt)?  $default,{required TResult orElse(),}) {final _that = this; | ||||||
| switch (_that) { | switch (_that) { | ||||||
| case _SnAccountProfile() when $default != null: | 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.links,_that.lastSeenAt,_that.activeBadge,_that.experience,_that.level,_that.levelingProgress,_that.picture,_that.background,_that.verification,_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.links,_that.lastSeenAt,_that.activeBadge,_that.experience,_that.level,_that.socialCredits,_that.socialCreditsLevel,_that.levelingProgress,_that.picture,_that.background,_that.verification,_that.createdAt,_that.updatedAt,_that.deletedAt);case _: | ||||||
|   return orElse(); |   return orElse(); | ||||||
|  |  | ||||||
| } | } | ||||||
| @@ -838,10 +843,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, @ProfileLinkConverter()  List<ProfileLink> links,  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; | @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, @ProfileLinkConverter()  List<ProfileLink> links,  DateTime? lastSeenAt,  SnAccountBadge? activeBadge,  int experience,  int level,  double socialCredits,  int socialCreditsLevel,  double levelingProgress,  SnCloudFile? picture,  SnCloudFile? background,  SnVerificationMark? verification,  DateTime createdAt,  DateTime updatedAt,  DateTime? deletedAt)  $default,) {final _that = this; | ||||||
| switch (_that) { | switch (_that) { | ||||||
| case _SnAccountProfile(): | 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.links,_that.lastSeenAt,_that.activeBadge,_that.experience,_that.level,_that.levelingProgress,_that.picture,_that.background,_that.verification,_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.links,_that.lastSeenAt,_that.activeBadge,_that.experience,_that.level,_that.socialCredits,_that.socialCreditsLevel,_that.levelingProgress,_that.picture,_that.background,_that.verification,_that.createdAt,_that.updatedAt,_that.deletedAt);} | ||||||
| } | } | ||||||
| /// A variant of `when` that fallback to returning `null` | /// A variant of `when` that fallback to returning `null` | ||||||
| /// | /// | ||||||
| @@ -855,10 +860,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, @ProfileLinkConverter()  List<ProfileLink> links,  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; | @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, @ProfileLinkConverter()  List<ProfileLink> links,  DateTime? lastSeenAt,  SnAccountBadge? activeBadge,  int experience,  int level,  double socialCredits,  int socialCreditsLevel,  double levelingProgress,  SnCloudFile? picture,  SnCloudFile? background,  SnVerificationMark? verification,  DateTime createdAt,  DateTime updatedAt,  DateTime? deletedAt)?  $default,) {final _that = this; | ||||||
| switch (_that) { | switch (_that) { | ||||||
| case _SnAccountProfile() when $default != null: | 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.links,_that.lastSeenAt,_that.activeBadge,_that.experience,_that.level,_that.levelingProgress,_that.picture,_that.background,_that.verification,_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.links,_that.lastSeenAt,_that.activeBadge,_that.experience,_that.level,_that.socialCredits,_that.socialCreditsLevel,_that.levelingProgress,_that.picture,_that.background,_that.verification,_that.createdAt,_that.updatedAt,_that.deletedAt);case _: | ||||||
|   return null; |   return null; | ||||||
|  |  | ||||||
| } | } | ||||||
| @@ -870,7 +875,7 @@ return $default(_that.id,_that.firstName,_that.middleName,_that.lastName,_that.b | |||||||
| @JsonSerializable() | @JsonSerializable() | ||||||
|  |  | ||||||
| class _SnAccountProfile implements SnAccountProfile { | 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, @ProfileLinkConverter() final  List<ProfileLink> links = const [], 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}): _links = links; |   const _SnAccountProfile({required this.id, this.firstName = '', this.middleName = '', this.lastName = '', this.bio = '', this.gender = '', this.pronouns = '', this.location = '', this.timeZone = '', this.birthday, @ProfileLinkConverter() final  List<ProfileLink> links = const [], this.lastSeenAt, this.activeBadge, required this.experience, required this.level, this.socialCredits = 100, this.socialCreditsLevel = 0, required this.levelingProgress, required this.picture, required this.background, required this.verification, required this.createdAt, required this.updatedAt, required this.deletedAt}): _links = links; | ||||||
|   factory _SnAccountProfile.fromJson(Map<String, dynamic> json) => _$SnAccountProfileFromJson(json); |   factory _SnAccountProfile.fromJson(Map<String, dynamic> json) => _$SnAccountProfileFromJson(json); | ||||||
|  |  | ||||||
| @override final  String id; | @override final  String id; | ||||||
| @@ -894,6 +899,8 @@ class _SnAccountProfile implements SnAccountProfile { | |||||||
| @override final  SnAccountBadge? activeBadge; | @override final  SnAccountBadge? activeBadge; | ||||||
| @override final  int experience; | @override final  int experience; | ||||||
| @override final  int level; | @override final  int level; | ||||||
|  | @override@JsonKey() final  double socialCredits; | ||||||
|  | @override@JsonKey() final  int socialCreditsLevel; | ||||||
| @override final  double levelingProgress; | @override final  double levelingProgress; | ||||||
| @override final  SnCloudFile? picture; | @override final  SnCloudFile? picture; | ||||||
| @override final  SnCloudFile? background; | @override final  SnCloudFile? background; | ||||||
| @@ -915,16 +922,16 @@ Map<String, dynamic> toJson() { | |||||||
|  |  | ||||||
| @override | @override | ||||||
| bool operator ==(Object other) { | 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)&&const DeepCollectionEquality().equals(other._links, _links)&&(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)); |   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)&&const DeepCollectionEquality().equals(other._links, _links)&&(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.socialCredits, socialCredits) || other.socialCredits == socialCredits)&&(identical(other.socialCreditsLevel, socialCreditsLevel) || other.socialCreditsLevel == socialCreditsLevel)&&(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) | @JsonKey(includeFromJson: false, includeToJson: false) | ||||||
| @override | @override | ||||||
| int get hashCode => Object.hashAll([runtimeType,id,firstName,middleName,lastName,bio,gender,pronouns,location,timeZone,birthday,const DeepCollectionEquality().hash(_links),lastSeenAt,activeBadge,experience,level,levelingProgress,picture,background,verification,createdAt,updatedAt,deletedAt]); | int get hashCode => Object.hashAll([runtimeType,id,firstName,middleName,lastName,bio,gender,pronouns,location,timeZone,birthday,const DeepCollectionEquality().hash(_links),lastSeenAt,activeBadge,experience,level,socialCredits,socialCreditsLevel,levelingProgress,picture,background,verification,createdAt,updatedAt,deletedAt]); | ||||||
|  |  | ||||||
| @override | @override | ||||||
| String toString() { | String toString() { | ||||||
|   return 'SnAccountProfile(id: $id, firstName: $firstName, middleName: $middleName, lastName: $lastName, bio: $bio, gender: $gender, pronouns: $pronouns, location: $location, timeZone: $timeZone, birthday: $birthday, links: $links, lastSeenAt: $lastSeenAt, activeBadge: $activeBadge, experience: $experience, level: $level, levelingProgress: $levelingProgress, picture: $picture, background: $background, verification: $verification, 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, links: $links, lastSeenAt: $lastSeenAt, activeBadge: $activeBadge, experience: $experience, level: $level, socialCredits: $socialCredits, socialCreditsLevel: $socialCreditsLevel, levelingProgress: $levelingProgress, picture: $picture, background: $background, verification: $verification, createdAt: $createdAt, updatedAt: $updatedAt, deletedAt: $deletedAt)'; | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -935,7 +942,7 @@ abstract mixin class _$SnAccountProfileCopyWith<$Res> implements $SnAccountProfi | |||||||
|   factory _$SnAccountProfileCopyWith(_SnAccountProfile value, $Res Function(_SnAccountProfile) _then) = __$SnAccountProfileCopyWithImpl; |   factory _$SnAccountProfileCopyWith(_SnAccountProfile value, $Res Function(_SnAccountProfile) _then) = __$SnAccountProfileCopyWithImpl; | ||||||
| @override @useResult | @override @useResult | ||||||
| $Res call({ | $Res call({ | ||||||
|  String id, String firstName, String middleName, String lastName, String bio, String gender, String pronouns, String location, String timeZone, DateTime? birthday,@ProfileLinkConverter() List<ProfileLink> links, DateTime? lastSeenAt, SnAccountBadge? activeBadge, int experience, int level, double levelingProgress, SnCloudFile? picture, SnCloudFile? background, SnVerificationMark? verification, 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,@ProfileLinkConverter() List<ProfileLink> links, DateTime? lastSeenAt, SnAccountBadge? activeBadge, int experience, int level, double socialCredits, int socialCreditsLevel, double levelingProgress, SnCloudFile? picture, SnCloudFile? background, SnVerificationMark? verification, DateTime createdAt, DateTime updatedAt, DateTime? deletedAt | ||||||
| }); | }); | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -952,7 +959,7 @@ class __$SnAccountProfileCopyWithImpl<$Res> | |||||||
|  |  | ||||||
| /// Create a copy of SnAccountProfile | /// Create a copy of SnAccountProfile | ||||||
| /// with the given fields replaced by the non-null parameter values. | /// 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? links = null,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,}) { | @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? links = null,Object? lastSeenAt = freezed,Object? activeBadge = freezed,Object? experience = null,Object? level = null,Object? socialCredits = null,Object? socialCreditsLevel = 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( |   return _then(_SnAccountProfile( | ||||||
| id: null == id ? _self.id : id // ignore: cast_nullable_to_non_nullable | 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 | as String,firstName: null == firstName ? _self.firstName : firstName // ignore: cast_nullable_to_non_nullable | ||||||
| @@ -969,6 +976,8 @@ as List<ProfileLink>,lastSeenAt: freezed == lastSeenAt ? _self.lastSeenAt : last | |||||||
| as DateTime?,activeBadge: freezed == activeBadge ? _self.activeBadge : activeBadge // ignore: cast_nullable_to_non_nullable | as DateTime?,activeBadge: freezed == activeBadge ? _self.activeBadge : activeBadge // ignore: cast_nullable_to_non_nullable | ||||||
| as SnAccountBadge?,experience: null == experience ? _self.experience : experience // ignore: cast_nullable_to_non_nullable | as SnAccountBadge?,experience: null == experience ? _self.experience : experience // ignore: cast_nullable_to_non_nullable | ||||||
| as int,level: null == level ? _self.level : level // ignore: cast_nullable_to_non_nullable | as int,level: null == level ? _self.level : level // ignore: cast_nullable_to_non_nullable | ||||||
|  | as int,socialCredits: null == socialCredits ? _self.socialCredits : socialCredits // ignore: cast_nullable_to_non_nullable | ||||||
|  | as double,socialCreditsLevel: null == socialCreditsLevel ? _self.socialCreditsLevel : socialCreditsLevel // ignore: cast_nullable_to_non_nullable | ||||||
| as int,levelingProgress: null == levelingProgress ? _self.levelingProgress : levelingProgress // ignore: cast_nullable_to_non_nullable | as int,levelingProgress: null == levelingProgress ? _self.levelingProgress : levelingProgress // ignore: cast_nullable_to_non_nullable | ||||||
| as double,picture: freezed == picture ? _self.picture : picture // ignore: cast_nullable_to_non_nullable | 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?,background: freezed == background ? _self.background : background // ignore: cast_nullable_to_non_nullable | ||||||
|   | |||||||
| @@ -11,6 +11,7 @@ _SnAccount _$SnAccountFromJson(Map<String, dynamic> json) => _SnAccount( | |||||||
|   name: json['name'] as String, |   name: json['name'] as String, | ||||||
|   nick: json['nick'] as String, |   nick: json['nick'] as String, | ||||||
|   language: json['language'] as String, |   language: json['language'] as String, | ||||||
|  |   region: json['region'] as String? ?? "", | ||||||
|   isSuperuser: json['is_superuser'] as bool, |   isSuperuser: json['is_superuser'] as bool, | ||||||
|   automatedId: json['automated_id'] as String?, |   automatedId: json['automated_id'] as String?, | ||||||
|   profile: SnAccountProfile.fromJson(json['profile'] as Map<String, dynamic>), |   profile: SnAccountProfile.fromJson(json['profile'] as Map<String, dynamic>), | ||||||
| @@ -39,6 +40,7 @@ Map<String, dynamic> _$SnAccountToJson(_SnAccount instance) => | |||||||
|       'name': instance.name, |       'name': instance.name, | ||||||
|       'nick': instance.nick, |       'nick': instance.nick, | ||||||
|       'language': instance.language, |       'language': instance.language, | ||||||
|  |       'region': instance.region, | ||||||
|       'is_superuser': instance.isSuperuser, |       'is_superuser': instance.isSuperuser, | ||||||
|       'automated_id': instance.automatedId, |       'automated_id': instance.automatedId, | ||||||
|       'profile': instance.profile.toJson(), |       'profile': instance.profile.toJson(), | ||||||
| @@ -86,6 +88,8 @@ _SnAccountProfile _$SnAccountProfileFromJson(Map<String, dynamic> json) => | |||||||
|               ), |               ), | ||||||
|       experience: (json['experience'] as num).toInt(), |       experience: (json['experience'] as num).toInt(), | ||||||
|       level: (json['level'] as num).toInt(), |       level: (json['level'] as num).toInt(), | ||||||
|  |       socialCredits: (json['social_credits'] as num?)?.toDouble() ?? 100, | ||||||
|  |       socialCreditsLevel: (json['social_credits_level'] as num?)?.toInt() ?? 0, | ||||||
|       levelingProgress: (json['leveling_progress'] as num).toDouble(), |       levelingProgress: (json['leveling_progress'] as num).toDouble(), | ||||||
|       picture: |       picture: | ||||||
|           json['picture'] == null |           json['picture'] == null | ||||||
| @@ -128,6 +132,8 @@ Map<String, dynamic> _$SnAccountProfileToJson(_SnAccountProfile instance) => | |||||||
|       'active_badge': instance.activeBadge?.toJson(), |       'active_badge': instance.activeBadge?.toJson(), | ||||||
|       'experience': instance.experience, |       'experience': instance.experience, | ||||||
|       'level': instance.level, |       'level': instance.level, | ||||||
|  |       'social_credits': instance.socialCredits, | ||||||
|  |       'social_credits_level': instance.socialCreditsLevel, | ||||||
|       'leveling_progress': instance.levelingProgress, |       'leveling_progress': instance.levelingProgress, | ||||||
|       'picture': instance.picture?.toJson(), |       'picture': instance.picture?.toJson(), | ||||||
|       'background': instance.background?.toJson(), |       'background': instance.background?.toJson(), | ||||||
|   | |||||||
| @@ -4,6 +4,20 @@ import 'package:island/models/account.dart'; | |||||||
| part 'activity.freezed.dart'; | part 'activity.freezed.dart'; | ||||||
| part 'activity.g.dart'; | part 'activity.g.dart'; | ||||||
|  |  | ||||||
|  | @freezed | ||||||
|  | sealed class SnNotableDay with _$SnNotableDay { | ||||||
|  |   const factory SnNotableDay({ | ||||||
|  |     required DateTime date, | ||||||
|  |     required String localName, | ||||||
|  |     required String globalName, | ||||||
|  |     required String countryCode, | ||||||
|  |     required List<int> holidays, | ||||||
|  |   }) = _SnNotableDay; | ||||||
|  |  | ||||||
|  |   factory SnNotableDay.fromJson(Map<String, dynamic> json) => | ||||||
|  |       _$SnNotableDayFromJson(json); | ||||||
|  | } | ||||||
|  |  | ||||||
| @freezed | @freezed | ||||||
| sealed class SnActivity with _$SnActivity { | sealed class SnActivity with _$SnActivity { | ||||||
|   const factory SnActivity({ |   const factory SnActivity({ | ||||||
| @@ -54,7 +68,7 @@ sealed class SnEventCalendarEntry with _$SnEventCalendarEntry { | |||||||
|   const factory SnEventCalendarEntry({ |   const factory SnEventCalendarEntry({ | ||||||
|     required DateTime date, |     required DateTime date, | ||||||
|     required SnCheckInResult? checkInResult, |     required SnCheckInResult? checkInResult, | ||||||
|     required List<dynamic> statuses, |     required List<SnAccountStatus> statuses, | ||||||
|   }) = _SnEventCalendarEntry; |   }) = _SnEventCalendarEntry; | ||||||
|  |  | ||||||
|   factory SnEventCalendarEntry.fromJson(Map<String, dynamic> json) => |   factory SnEventCalendarEntry.fromJson(Map<String, dynamic> json) => | ||||||
|   | |||||||
| @@ -12,6 +12,281 @@ part of 'activity.dart'; | |||||||
| // dart format off | // dart format off | ||||||
| T _$identity<T>(T value) => value; | T _$identity<T>(T value) => value; | ||||||
|  |  | ||||||
|  | /// @nodoc | ||||||
|  | mixin _$SnNotableDay { | ||||||
|  |  | ||||||
|  |  DateTime get date; String get localName; String get globalName; String get countryCode; List<int> get holidays; | ||||||
|  | /// Create a copy of SnNotableDay | ||||||
|  | /// with the given fields replaced by the non-null parameter values. | ||||||
|  | @JsonKey(includeFromJson: false, includeToJson: false) | ||||||
|  | @pragma('vm:prefer-inline') | ||||||
|  | $SnNotableDayCopyWith<SnNotableDay> get copyWith => _$SnNotableDayCopyWithImpl<SnNotableDay>(this as SnNotableDay, _$identity); | ||||||
|  |  | ||||||
|  |   /// Serializes this SnNotableDay to a JSON map. | ||||||
|  |   Map<String, dynamic> toJson(); | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @override | ||||||
|  | bool operator ==(Object other) { | ||||||
|  |   return identical(this, other) || (other.runtimeType == runtimeType&&other is SnNotableDay&&(identical(other.date, date) || other.date == date)&&(identical(other.localName, localName) || other.localName == localName)&&(identical(other.globalName, globalName) || other.globalName == globalName)&&(identical(other.countryCode, countryCode) || other.countryCode == countryCode)&&const DeepCollectionEquality().equals(other.holidays, holidays)); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @JsonKey(includeFromJson: false, includeToJson: false) | ||||||
|  | @override | ||||||
|  | int get hashCode => Object.hash(runtimeType,date,localName,globalName,countryCode,const DeepCollectionEquality().hash(holidays)); | ||||||
|  |  | ||||||
|  | @override | ||||||
|  | String toString() { | ||||||
|  |   return 'SnNotableDay(date: $date, localName: $localName, globalName: $globalName, countryCode: $countryCode, holidays: $holidays)'; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /// @nodoc | ||||||
|  | abstract mixin class $SnNotableDayCopyWith<$Res>  { | ||||||
|  |   factory $SnNotableDayCopyWith(SnNotableDay value, $Res Function(SnNotableDay) _then) = _$SnNotableDayCopyWithImpl; | ||||||
|  | @useResult | ||||||
|  | $Res call({ | ||||||
|  |  DateTime date, String localName, String globalName, String countryCode, List<int> holidays | ||||||
|  | }); | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | } | ||||||
|  | /// @nodoc | ||||||
|  | class _$SnNotableDayCopyWithImpl<$Res> | ||||||
|  |     implements $SnNotableDayCopyWith<$Res> { | ||||||
|  |   _$SnNotableDayCopyWithImpl(this._self, this._then); | ||||||
|  |  | ||||||
|  |   final SnNotableDay _self; | ||||||
|  |   final $Res Function(SnNotableDay) _then; | ||||||
|  |  | ||||||
|  | /// Create a copy of SnNotableDay | ||||||
|  | /// with the given fields replaced by the non-null parameter values. | ||||||
|  | @pragma('vm:prefer-inline') @override $Res call({Object? date = null,Object? localName = null,Object? globalName = null,Object? countryCode = null,Object? holidays = null,}) { | ||||||
|  |   return _then(_self.copyWith( | ||||||
|  | date: null == date ? _self.date : date // ignore: cast_nullable_to_non_nullable | ||||||
|  | as DateTime,localName: null == localName ? _self.localName : localName // ignore: cast_nullable_to_non_nullable | ||||||
|  | as String,globalName: null == globalName ? _self.globalName : globalName // ignore: cast_nullable_to_non_nullable | ||||||
|  | as String,countryCode: null == countryCode ? _self.countryCode : countryCode // ignore: cast_nullable_to_non_nullable | ||||||
|  | as String,holidays: null == holidays ? _self.holidays : holidays // ignore: cast_nullable_to_non_nullable | ||||||
|  | as List<int>, | ||||||
|  |   )); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | /// Adds pattern-matching-related methods to [SnNotableDay]. | ||||||
|  | extension SnNotableDayPatterns on SnNotableDay { | ||||||
|  | /// 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( _SnNotableDay value)?  $default,{required TResult orElse(),}){ | ||||||
|  | final _that = this; | ||||||
|  | switch (_that) { | ||||||
|  | case _SnNotableDay() 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( _SnNotableDay value)  $default,){ | ||||||
|  | final _that = this; | ||||||
|  | switch (_that) { | ||||||
|  | case _SnNotableDay(): | ||||||
|  | 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( _SnNotableDay value)?  $default,){ | ||||||
|  | final _that = this; | ||||||
|  | switch (_that) { | ||||||
|  | case _SnNotableDay() 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( DateTime date,  String localName,  String globalName,  String countryCode,  List<int> holidays)?  $default,{required TResult orElse(),}) {final _that = this; | ||||||
|  | switch (_that) { | ||||||
|  | case _SnNotableDay() when $default != null: | ||||||
|  | return $default(_that.date,_that.localName,_that.globalName,_that.countryCode,_that.holidays);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( DateTime date,  String localName,  String globalName,  String countryCode,  List<int> holidays)  $default,) {final _that = this; | ||||||
|  | switch (_that) { | ||||||
|  | case _SnNotableDay(): | ||||||
|  | return $default(_that.date,_that.localName,_that.globalName,_that.countryCode,_that.holidays);} | ||||||
|  | } | ||||||
|  | /// 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( DateTime date,  String localName,  String globalName,  String countryCode,  List<int> holidays)?  $default,) {final _that = this; | ||||||
|  | switch (_that) { | ||||||
|  | case _SnNotableDay() when $default != null: | ||||||
|  | return $default(_that.date,_that.localName,_that.globalName,_that.countryCode,_that.holidays);case _: | ||||||
|  |   return null; | ||||||
|  |  | ||||||
|  | } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /// @nodoc | ||||||
|  | @JsonSerializable() | ||||||
|  |  | ||||||
|  | class _SnNotableDay implements SnNotableDay { | ||||||
|  |   const _SnNotableDay({required this.date, required this.localName, required this.globalName, required this.countryCode, required final  List<int> holidays}): _holidays = holidays; | ||||||
|  |   factory _SnNotableDay.fromJson(Map<String, dynamic> json) => _$SnNotableDayFromJson(json); | ||||||
|  |  | ||||||
|  | @override final  DateTime date; | ||||||
|  | @override final  String localName; | ||||||
|  | @override final  String globalName; | ||||||
|  | @override final  String countryCode; | ||||||
|  |  final  List<int> _holidays; | ||||||
|  | @override List<int> get holidays { | ||||||
|  |   if (_holidays is EqualUnmodifiableListView) return _holidays; | ||||||
|  |   // ignore: implicit_dynamic_type | ||||||
|  |   return EqualUnmodifiableListView(_holidays); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | /// Create a copy of SnNotableDay | ||||||
|  | /// with the given fields replaced by the non-null parameter values. | ||||||
|  | @override @JsonKey(includeFromJson: false, includeToJson: false) | ||||||
|  | @pragma('vm:prefer-inline') | ||||||
|  | _$SnNotableDayCopyWith<_SnNotableDay> get copyWith => __$SnNotableDayCopyWithImpl<_SnNotableDay>(this, _$identity); | ||||||
|  |  | ||||||
|  | @override | ||||||
|  | Map<String, dynamic> toJson() { | ||||||
|  |   return _$SnNotableDayToJson(this, ); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @override | ||||||
|  | bool operator ==(Object other) { | ||||||
|  |   return identical(this, other) || (other.runtimeType == runtimeType&&other is _SnNotableDay&&(identical(other.date, date) || other.date == date)&&(identical(other.localName, localName) || other.localName == localName)&&(identical(other.globalName, globalName) || other.globalName == globalName)&&(identical(other.countryCode, countryCode) || other.countryCode == countryCode)&&const DeepCollectionEquality().equals(other._holidays, _holidays)); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @JsonKey(includeFromJson: false, includeToJson: false) | ||||||
|  | @override | ||||||
|  | int get hashCode => Object.hash(runtimeType,date,localName,globalName,countryCode,const DeepCollectionEquality().hash(_holidays)); | ||||||
|  |  | ||||||
|  | @override | ||||||
|  | String toString() { | ||||||
|  |   return 'SnNotableDay(date: $date, localName: $localName, globalName: $globalName, countryCode: $countryCode, holidays: $holidays)'; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /// @nodoc | ||||||
|  | abstract mixin class _$SnNotableDayCopyWith<$Res> implements $SnNotableDayCopyWith<$Res> { | ||||||
|  |   factory _$SnNotableDayCopyWith(_SnNotableDay value, $Res Function(_SnNotableDay) _then) = __$SnNotableDayCopyWithImpl; | ||||||
|  | @override @useResult | ||||||
|  | $Res call({ | ||||||
|  |  DateTime date, String localName, String globalName, String countryCode, List<int> holidays | ||||||
|  | }); | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | } | ||||||
|  | /// @nodoc | ||||||
|  | class __$SnNotableDayCopyWithImpl<$Res> | ||||||
|  |     implements _$SnNotableDayCopyWith<$Res> { | ||||||
|  |   __$SnNotableDayCopyWithImpl(this._self, this._then); | ||||||
|  |  | ||||||
|  |   final _SnNotableDay _self; | ||||||
|  |   final $Res Function(_SnNotableDay) _then; | ||||||
|  |  | ||||||
|  | /// Create a copy of SnNotableDay | ||||||
|  | /// with the given fields replaced by the non-null parameter values. | ||||||
|  | @override @pragma('vm:prefer-inline') $Res call({Object? date = null,Object? localName = null,Object? globalName = null,Object? countryCode = null,Object? holidays = null,}) { | ||||||
|  |   return _then(_SnNotableDay( | ||||||
|  | date: null == date ? _self.date : date // ignore: cast_nullable_to_non_nullable | ||||||
|  | as DateTime,localName: null == localName ? _self.localName : localName // ignore: cast_nullable_to_non_nullable | ||||||
|  | as String,globalName: null == globalName ? _self.globalName : globalName // ignore: cast_nullable_to_non_nullable | ||||||
|  | as String,countryCode: null == countryCode ? _self.countryCode : countryCode // ignore: cast_nullable_to_non_nullable | ||||||
|  | as String,holidays: null == holidays ? _self._holidays : holidays // ignore: cast_nullable_to_non_nullable | ||||||
|  | as List<int>, | ||||||
|  |   )); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
| /// @nodoc | /// @nodoc | ||||||
| mixin _$SnActivity { | mixin _$SnActivity { | ||||||
|  |  | ||||||
| @@ -861,7 +1136,7 @@ as String, | |||||||
| /// @nodoc | /// @nodoc | ||||||
| mixin _$SnEventCalendarEntry { | mixin _$SnEventCalendarEntry { | ||||||
|  |  | ||||||
|  DateTime get date; SnCheckInResult? get checkInResult; List<dynamic> get statuses; |  DateTime get date; SnCheckInResult? get checkInResult; List<SnAccountStatus> get statuses; | ||||||
| /// Create a copy of SnEventCalendarEntry | /// Create a copy of SnEventCalendarEntry | ||||||
| /// with the given fields replaced by the non-null parameter values. | /// with the given fields replaced by the non-null parameter values. | ||||||
| @JsonKey(includeFromJson: false, includeToJson: false) | @JsonKey(includeFromJson: false, includeToJson: false) | ||||||
| @@ -894,7 +1169,7 @@ abstract mixin class $SnEventCalendarEntryCopyWith<$Res>  { | |||||||
|   factory $SnEventCalendarEntryCopyWith(SnEventCalendarEntry value, $Res Function(SnEventCalendarEntry) _then) = _$SnEventCalendarEntryCopyWithImpl; |   factory $SnEventCalendarEntryCopyWith(SnEventCalendarEntry value, $Res Function(SnEventCalendarEntry) _then) = _$SnEventCalendarEntryCopyWithImpl; | ||||||
| @useResult | @useResult | ||||||
| $Res call({ | $Res call({ | ||||||
|  DateTime date, SnCheckInResult? checkInResult, List<dynamic> statuses |  DateTime date, SnCheckInResult? checkInResult, List<SnAccountStatus> statuses | ||||||
| }); | }); | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -916,7 +1191,7 @@ class _$SnEventCalendarEntryCopyWithImpl<$Res> | |||||||
| date: null == date ? _self.date : date // ignore: cast_nullable_to_non_nullable | date: null == date ? _self.date : date // ignore: cast_nullable_to_non_nullable | ||||||
| as DateTime,checkInResult: freezed == checkInResult ? _self.checkInResult : checkInResult // ignore: cast_nullable_to_non_nullable | as DateTime,checkInResult: freezed == checkInResult ? _self.checkInResult : checkInResult // ignore: cast_nullable_to_non_nullable | ||||||
| as SnCheckInResult?,statuses: null == statuses ? _self.statuses : statuses // ignore: cast_nullable_to_non_nullable | as SnCheckInResult?,statuses: null == statuses ? _self.statuses : statuses // ignore: cast_nullable_to_non_nullable | ||||||
| as List<dynamic>, | as List<SnAccountStatus>, | ||||||
|   )); |   )); | ||||||
| } | } | ||||||
| /// Create a copy of SnEventCalendarEntry | /// Create a copy of SnEventCalendarEntry | ||||||
| @@ -1010,7 +1285,7 @@ return $default(_that);case _: | |||||||
| /// } | /// } | ||||||
| /// ``` | /// ``` | ||||||
|  |  | ||||||
| @optionalTypeArgs TResult maybeWhen<TResult extends Object?>(TResult Function( DateTime date,  SnCheckInResult? checkInResult,  List<dynamic> statuses)?  $default,{required TResult orElse(),}) {final _that = this; | @optionalTypeArgs TResult maybeWhen<TResult extends Object?>(TResult Function( DateTime date,  SnCheckInResult? checkInResult,  List<SnAccountStatus> statuses)?  $default,{required TResult orElse(),}) {final _that = this; | ||||||
| switch (_that) { | switch (_that) { | ||||||
| case _SnEventCalendarEntry() when $default != null: | case _SnEventCalendarEntry() when $default != null: | ||||||
| return $default(_that.date,_that.checkInResult,_that.statuses);case _: | return $default(_that.date,_that.checkInResult,_that.statuses);case _: | ||||||
| @@ -1031,7 +1306,7 @@ return $default(_that.date,_that.checkInResult,_that.statuses);case _: | |||||||
| /// } | /// } | ||||||
| /// ``` | /// ``` | ||||||
|  |  | ||||||
| @optionalTypeArgs TResult when<TResult extends Object?>(TResult Function( DateTime date,  SnCheckInResult? checkInResult,  List<dynamic> statuses)  $default,) {final _that = this; | @optionalTypeArgs TResult when<TResult extends Object?>(TResult Function( DateTime date,  SnCheckInResult? checkInResult,  List<SnAccountStatus> statuses)  $default,) {final _that = this; | ||||||
| switch (_that) { | switch (_that) { | ||||||
| case _SnEventCalendarEntry(): | case _SnEventCalendarEntry(): | ||||||
| return $default(_that.date,_that.checkInResult,_that.statuses);} | return $default(_that.date,_that.checkInResult,_that.statuses);} | ||||||
| @@ -1048,7 +1323,7 @@ return $default(_that.date,_that.checkInResult,_that.statuses);} | |||||||
| /// } | /// } | ||||||
| /// ``` | /// ``` | ||||||
|  |  | ||||||
| @optionalTypeArgs TResult? whenOrNull<TResult extends Object?>(TResult? Function( DateTime date,  SnCheckInResult? checkInResult,  List<dynamic> statuses)?  $default,) {final _that = this; | @optionalTypeArgs TResult? whenOrNull<TResult extends Object?>(TResult? Function( DateTime date,  SnCheckInResult? checkInResult,  List<SnAccountStatus> statuses)?  $default,) {final _that = this; | ||||||
| switch (_that) { | switch (_that) { | ||||||
| case _SnEventCalendarEntry() when $default != null: | case _SnEventCalendarEntry() when $default != null: | ||||||
| return $default(_that.date,_that.checkInResult,_that.statuses);case _: | return $default(_that.date,_that.checkInResult,_that.statuses);case _: | ||||||
| @@ -1063,13 +1338,13 @@ return $default(_that.date,_that.checkInResult,_that.statuses);case _: | |||||||
| @JsonSerializable() | @JsonSerializable() | ||||||
|  |  | ||||||
| class _SnEventCalendarEntry implements SnEventCalendarEntry { | class _SnEventCalendarEntry implements SnEventCalendarEntry { | ||||||
|   const _SnEventCalendarEntry({required this.date, required this.checkInResult, required final  List<dynamic> statuses}): _statuses = statuses; |   const _SnEventCalendarEntry({required this.date, required this.checkInResult, required final  List<SnAccountStatus> statuses}): _statuses = statuses; | ||||||
|   factory _SnEventCalendarEntry.fromJson(Map<String, dynamic> json) => _$SnEventCalendarEntryFromJson(json); |   factory _SnEventCalendarEntry.fromJson(Map<String, dynamic> json) => _$SnEventCalendarEntryFromJson(json); | ||||||
|  |  | ||||||
| @override final  DateTime date; | @override final  DateTime date; | ||||||
| @override final  SnCheckInResult? checkInResult; | @override final  SnCheckInResult? checkInResult; | ||||||
|  final  List<dynamic> _statuses; |  final  List<SnAccountStatus> _statuses; | ||||||
| @override List<dynamic> get statuses { | @override List<SnAccountStatus> get statuses { | ||||||
|   if (_statuses is EqualUnmodifiableListView) return _statuses; |   if (_statuses is EqualUnmodifiableListView) return _statuses; | ||||||
|   // ignore: implicit_dynamic_type |   // ignore: implicit_dynamic_type | ||||||
|   return EqualUnmodifiableListView(_statuses); |   return EqualUnmodifiableListView(_statuses); | ||||||
| @@ -1109,7 +1384,7 @@ abstract mixin class _$SnEventCalendarEntryCopyWith<$Res> implements $SnEventCal | |||||||
|   factory _$SnEventCalendarEntryCopyWith(_SnEventCalendarEntry value, $Res Function(_SnEventCalendarEntry) _then) = __$SnEventCalendarEntryCopyWithImpl; |   factory _$SnEventCalendarEntryCopyWith(_SnEventCalendarEntry value, $Res Function(_SnEventCalendarEntry) _then) = __$SnEventCalendarEntryCopyWithImpl; | ||||||
| @override @useResult | @override @useResult | ||||||
| $Res call({ | $Res call({ | ||||||
|  DateTime date, SnCheckInResult? checkInResult, List<dynamic> statuses |  DateTime date, SnCheckInResult? checkInResult, List<SnAccountStatus> statuses | ||||||
| }); | }); | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -1131,7 +1406,7 @@ class __$SnEventCalendarEntryCopyWithImpl<$Res> | |||||||
| date: null == date ? _self.date : date // ignore: cast_nullable_to_non_nullable | date: null == date ? _self.date : date // ignore: cast_nullable_to_non_nullable | ||||||
| as DateTime,checkInResult: freezed == checkInResult ? _self.checkInResult : checkInResult // ignore: cast_nullable_to_non_nullable | as DateTime,checkInResult: freezed == checkInResult ? _self.checkInResult : checkInResult // ignore: cast_nullable_to_non_nullable | ||||||
| as SnCheckInResult?,statuses: null == statuses ? _self._statuses : statuses // ignore: cast_nullable_to_non_nullable | as SnCheckInResult?,statuses: null == statuses ? _self._statuses : statuses // ignore: cast_nullable_to_non_nullable | ||||||
| as List<dynamic>, | as List<SnAccountStatus>, | ||||||
|   )); |   )); | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -6,6 +6,27 @@ part of 'activity.dart'; | |||||||
| // JsonSerializableGenerator | // JsonSerializableGenerator | ||||||
| // ************************************************************************** | // ************************************************************************** | ||||||
|  |  | ||||||
|  | _SnNotableDay _$SnNotableDayFromJson(Map<String, dynamic> json) => | ||||||
|  |     _SnNotableDay( | ||||||
|  |       date: DateTime.parse(json['date'] as String), | ||||||
|  |       localName: json['local_name'] as String, | ||||||
|  |       globalName: json['global_name'] as String, | ||||||
|  |       countryCode: json['country_code'] as String, | ||||||
|  |       holidays: | ||||||
|  |           (json['holidays'] as List<dynamic>) | ||||||
|  |               .map((e) => (e as num).toInt()) | ||||||
|  |               .toList(), | ||||||
|  |     ); | ||||||
|  |  | ||||||
|  | Map<String, dynamic> _$SnNotableDayToJson(_SnNotableDay instance) => | ||||||
|  |     <String, dynamic>{ | ||||||
|  |       'date': instance.date.toIso8601String(), | ||||||
|  |       'local_name': instance.localName, | ||||||
|  |       'global_name': instance.globalName, | ||||||
|  |       'country_code': instance.countryCode, | ||||||
|  |       'holidays': instance.holidays, | ||||||
|  |     }; | ||||||
|  |  | ||||||
| _SnActivity _$SnActivityFromJson(Map<String, dynamic> json) => _SnActivity( | _SnActivity _$SnActivityFromJson(Map<String, dynamic> json) => _SnActivity( | ||||||
|   id: json['id'] as String, |   id: json['id'] as String, | ||||||
|   type: json['type'] as String, |   type: json['type'] as String, | ||||||
| @@ -87,7 +108,10 @@ _SnEventCalendarEntry _$SnEventCalendarEntryFromJson( | |||||||
|           : SnCheckInResult.fromJson( |           : SnCheckInResult.fromJson( | ||||||
|             json['check_in_result'] as Map<String, dynamic>, |             json['check_in_result'] as Map<String, dynamic>, | ||||||
|           ), |           ), | ||||||
|   statuses: json['statuses'] as List<dynamic>, |   statuses: | ||||||
|  |       (json['statuses'] as List<dynamic>) | ||||||
|  |           .map((e) => SnAccountStatus.fromJson(e as Map<String, dynamic>)) | ||||||
|  |           .toList(), | ||||||
| ); | ); | ||||||
|  |  | ||||||
| Map<String, dynamic> _$SnEventCalendarEntryToJson( | Map<String, dynamic> _$SnEventCalendarEntryToJson( | ||||||
| @@ -95,5 +119,5 @@ Map<String, dynamic> _$SnEventCalendarEntryToJson( | |||||||
| ) => <String, dynamic>{ | ) => <String, dynamic>{ | ||||||
|   'date': instance.date.toIso8601String(), |   'date': instance.date.toIso8601String(), | ||||||
|   'check_in_result': instance.checkInResult?.toJson(), |   'check_in_result': instance.checkInResult?.toJson(), | ||||||
|   'statuses': instance.statuses, |   'statuses': instance.statuses.map((e) => e.toJson()).toList(), | ||||||
| }; | }; | ||||||
|   | |||||||
| @@ -11,6 +11,20 @@ sealed class AppToken with _$AppToken { | |||||||
|       _$AppTokenFromJson(json); |       _$AppTokenFromJson(json); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @freezed | ||||||
|  | sealed class GeoIpLocation with _$GeoIpLocation { | ||||||
|  |   const factory GeoIpLocation({ | ||||||
|  |     required double latitude, | ||||||
|  |     required double longitude, | ||||||
|  |     required String countryCode, | ||||||
|  |     required String country, | ||||||
|  |     required String city, | ||||||
|  |   }) = _GeoIpLocation; | ||||||
|  |  | ||||||
|  |   factory GeoIpLocation.fromJson(Map<String, dynamic> json) => | ||||||
|  |       _$GeoIpLocationFromJson(json); | ||||||
|  | } | ||||||
|  |  | ||||||
| @freezed | @freezed | ||||||
| sealed class SnAuthChallenge with _$SnAuthChallenge { | sealed class SnAuthChallenge with _$SnAuthChallenge { | ||||||
|   const factory SnAuthChallenge({ |   const factory SnAuthChallenge({ | ||||||
| @@ -26,7 +40,7 @@ sealed class SnAuthChallenge with _$SnAuthChallenge { | |||||||
|     required String ipAddress, |     required String ipAddress, | ||||||
|     required String userAgent, |     required String userAgent, | ||||||
|     required String? nonce, |     required String? nonce, | ||||||
|     required String? location, |     required GeoIpLocation? location, | ||||||
|     required String accountId, |     required String accountId, | ||||||
|     required DateTime createdAt, |     required DateTime createdAt, | ||||||
|     required DateTime updatedAt, |     required DateTime updatedAt, | ||||||
|   | |||||||
| @@ -269,10 +269,279 @@ as String, | |||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | /// @nodoc | ||||||
|  | mixin _$GeoIpLocation { | ||||||
|  |  | ||||||
|  |  double get latitude; double get longitude; String get countryCode; String get country; String get city; | ||||||
|  | /// Create a copy of GeoIpLocation | ||||||
|  | /// with the given fields replaced by the non-null parameter values. | ||||||
|  | @JsonKey(includeFromJson: false, includeToJson: false) | ||||||
|  | @pragma('vm:prefer-inline') | ||||||
|  | $GeoIpLocationCopyWith<GeoIpLocation> get copyWith => _$GeoIpLocationCopyWithImpl<GeoIpLocation>(this as GeoIpLocation, _$identity); | ||||||
|  |  | ||||||
|  |   /// Serializes this GeoIpLocation to a JSON map. | ||||||
|  |   Map<String, dynamic> toJson(); | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @override | ||||||
|  | bool operator ==(Object other) { | ||||||
|  |   return identical(this, other) || (other.runtimeType == runtimeType&&other is GeoIpLocation&&(identical(other.latitude, latitude) || other.latitude == latitude)&&(identical(other.longitude, longitude) || other.longitude == longitude)&&(identical(other.countryCode, countryCode) || other.countryCode == countryCode)&&(identical(other.country, country) || other.country == country)&&(identical(other.city, city) || other.city == city)); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @JsonKey(includeFromJson: false, includeToJson: false) | ||||||
|  | @override | ||||||
|  | int get hashCode => Object.hash(runtimeType,latitude,longitude,countryCode,country,city); | ||||||
|  |  | ||||||
|  | @override | ||||||
|  | String toString() { | ||||||
|  |   return 'GeoIpLocation(latitude: $latitude, longitude: $longitude, countryCode: $countryCode, country: $country, city: $city)'; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /// @nodoc | ||||||
|  | abstract mixin class $GeoIpLocationCopyWith<$Res>  { | ||||||
|  |   factory $GeoIpLocationCopyWith(GeoIpLocation value, $Res Function(GeoIpLocation) _then) = _$GeoIpLocationCopyWithImpl; | ||||||
|  | @useResult | ||||||
|  | $Res call({ | ||||||
|  |  double latitude, double longitude, String countryCode, String country, String city | ||||||
|  | }); | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | } | ||||||
|  | /// @nodoc | ||||||
|  | class _$GeoIpLocationCopyWithImpl<$Res> | ||||||
|  |     implements $GeoIpLocationCopyWith<$Res> { | ||||||
|  |   _$GeoIpLocationCopyWithImpl(this._self, this._then); | ||||||
|  |  | ||||||
|  |   final GeoIpLocation _self; | ||||||
|  |   final $Res Function(GeoIpLocation) _then; | ||||||
|  |  | ||||||
|  | /// Create a copy of GeoIpLocation | ||||||
|  | /// with the given fields replaced by the non-null parameter values. | ||||||
|  | @pragma('vm:prefer-inline') @override $Res call({Object? latitude = null,Object? longitude = null,Object? countryCode = null,Object? country = null,Object? city = null,}) { | ||||||
|  |   return _then(_self.copyWith( | ||||||
|  | latitude: null == latitude ? _self.latitude : latitude // ignore: cast_nullable_to_non_nullable | ||||||
|  | as double,longitude: null == longitude ? _self.longitude : longitude // ignore: cast_nullable_to_non_nullable | ||||||
|  | as double,countryCode: null == countryCode ? _self.countryCode : countryCode // ignore: cast_nullable_to_non_nullable | ||||||
|  | as String,country: null == country ? _self.country : country // ignore: cast_nullable_to_non_nullable | ||||||
|  | as String,city: null == city ? _self.city : city // ignore: cast_nullable_to_non_nullable | ||||||
|  | as String, | ||||||
|  |   )); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | /// Adds pattern-matching-related methods to [GeoIpLocation]. | ||||||
|  | extension GeoIpLocationPatterns on GeoIpLocation { | ||||||
|  | /// 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( _GeoIpLocation value)?  $default,{required TResult orElse(),}){ | ||||||
|  | final _that = this; | ||||||
|  | switch (_that) { | ||||||
|  | case _GeoIpLocation() 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( _GeoIpLocation value)  $default,){ | ||||||
|  | final _that = this; | ||||||
|  | switch (_that) { | ||||||
|  | case _GeoIpLocation(): | ||||||
|  | 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( _GeoIpLocation value)?  $default,){ | ||||||
|  | final _that = this; | ||||||
|  | switch (_that) { | ||||||
|  | case _GeoIpLocation() 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( double latitude,  double longitude,  String countryCode,  String country,  String city)?  $default,{required TResult orElse(),}) {final _that = this; | ||||||
|  | switch (_that) { | ||||||
|  | case _GeoIpLocation() when $default != null: | ||||||
|  | return $default(_that.latitude,_that.longitude,_that.countryCode,_that.country,_that.city);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( double latitude,  double longitude,  String countryCode,  String country,  String city)  $default,) {final _that = this; | ||||||
|  | switch (_that) { | ||||||
|  | case _GeoIpLocation(): | ||||||
|  | return $default(_that.latitude,_that.longitude,_that.countryCode,_that.country,_that.city);} | ||||||
|  | } | ||||||
|  | /// 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( double latitude,  double longitude,  String countryCode,  String country,  String city)?  $default,) {final _that = this; | ||||||
|  | switch (_that) { | ||||||
|  | case _GeoIpLocation() when $default != null: | ||||||
|  | return $default(_that.latitude,_that.longitude,_that.countryCode,_that.country,_that.city);case _: | ||||||
|  |   return null; | ||||||
|  |  | ||||||
|  | } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /// @nodoc | ||||||
|  | @JsonSerializable() | ||||||
|  |  | ||||||
|  | class _GeoIpLocation implements GeoIpLocation { | ||||||
|  |   const _GeoIpLocation({required this.latitude, required this.longitude, required this.countryCode, required this.country, required this.city}); | ||||||
|  |   factory _GeoIpLocation.fromJson(Map<String, dynamic> json) => _$GeoIpLocationFromJson(json); | ||||||
|  |  | ||||||
|  | @override final  double latitude; | ||||||
|  | @override final  double longitude; | ||||||
|  | @override final  String countryCode; | ||||||
|  | @override final  String country; | ||||||
|  | @override final  String city; | ||||||
|  |  | ||||||
|  | /// Create a copy of GeoIpLocation | ||||||
|  | /// with the given fields replaced by the non-null parameter values. | ||||||
|  | @override @JsonKey(includeFromJson: false, includeToJson: false) | ||||||
|  | @pragma('vm:prefer-inline') | ||||||
|  | _$GeoIpLocationCopyWith<_GeoIpLocation> get copyWith => __$GeoIpLocationCopyWithImpl<_GeoIpLocation>(this, _$identity); | ||||||
|  |  | ||||||
|  | @override | ||||||
|  | Map<String, dynamic> toJson() { | ||||||
|  |   return _$GeoIpLocationToJson(this, ); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @override | ||||||
|  | bool operator ==(Object other) { | ||||||
|  |   return identical(this, other) || (other.runtimeType == runtimeType&&other is _GeoIpLocation&&(identical(other.latitude, latitude) || other.latitude == latitude)&&(identical(other.longitude, longitude) || other.longitude == longitude)&&(identical(other.countryCode, countryCode) || other.countryCode == countryCode)&&(identical(other.country, country) || other.country == country)&&(identical(other.city, city) || other.city == city)); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @JsonKey(includeFromJson: false, includeToJson: false) | ||||||
|  | @override | ||||||
|  | int get hashCode => Object.hash(runtimeType,latitude,longitude,countryCode,country,city); | ||||||
|  |  | ||||||
|  | @override | ||||||
|  | String toString() { | ||||||
|  |   return 'GeoIpLocation(latitude: $latitude, longitude: $longitude, countryCode: $countryCode, country: $country, city: $city)'; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /// @nodoc | ||||||
|  | abstract mixin class _$GeoIpLocationCopyWith<$Res> implements $GeoIpLocationCopyWith<$Res> { | ||||||
|  |   factory _$GeoIpLocationCopyWith(_GeoIpLocation value, $Res Function(_GeoIpLocation) _then) = __$GeoIpLocationCopyWithImpl; | ||||||
|  | @override @useResult | ||||||
|  | $Res call({ | ||||||
|  |  double latitude, double longitude, String countryCode, String country, String city | ||||||
|  | }); | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | } | ||||||
|  | /// @nodoc | ||||||
|  | class __$GeoIpLocationCopyWithImpl<$Res> | ||||||
|  |     implements _$GeoIpLocationCopyWith<$Res> { | ||||||
|  |   __$GeoIpLocationCopyWithImpl(this._self, this._then); | ||||||
|  |  | ||||||
|  |   final _GeoIpLocation _self; | ||||||
|  |   final $Res Function(_GeoIpLocation) _then; | ||||||
|  |  | ||||||
|  | /// Create a copy of GeoIpLocation | ||||||
|  | /// with the given fields replaced by the non-null parameter values. | ||||||
|  | @override @pragma('vm:prefer-inline') $Res call({Object? latitude = null,Object? longitude = null,Object? countryCode = null,Object? country = null,Object? city = null,}) { | ||||||
|  |   return _then(_GeoIpLocation( | ||||||
|  | latitude: null == latitude ? _self.latitude : latitude // ignore: cast_nullable_to_non_nullable | ||||||
|  | as double,longitude: null == longitude ? _self.longitude : longitude // ignore: cast_nullable_to_non_nullable | ||||||
|  | as double,countryCode: null == countryCode ? _self.countryCode : countryCode // ignore: cast_nullable_to_non_nullable | ||||||
|  | as String,country: null == country ? _self.country : country // ignore: cast_nullable_to_non_nullable | ||||||
|  | as String,city: null == city ? _self.city : city // ignore: cast_nullable_to_non_nullable | ||||||
|  | as String, | ||||||
|  |   )); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
| /// @nodoc | /// @nodoc | ||||||
| mixin _$SnAuthChallenge { | mixin _$SnAuthChallenge { | ||||||
|  |  | ||||||
|  String get id; DateTime get expiredAt; int get stepRemain; int get stepTotal; int get failedAttempts; int get type; List<String> get blacklistFactors; List<dynamic> get audiences; List<dynamic> get scopes; String get ipAddress; String get userAgent; String? get nonce; String? get location; String get accountId; DateTime get createdAt; DateTime get updatedAt; DateTime? get deletedAt; |  String get id; DateTime get expiredAt; int get stepRemain; int get stepTotal; int get failedAttempts; int get type; List<String> get blacklistFactors; List<dynamic> get audiences; List<dynamic> get scopes; String get ipAddress; String get userAgent; String? get nonce; GeoIpLocation? get location; String get accountId; DateTime get createdAt; DateTime get updatedAt; DateTime? get deletedAt; | ||||||
| /// Create a copy of SnAuthChallenge | /// Create a copy of SnAuthChallenge | ||||||
| /// with the given fields replaced by the non-null parameter values. | /// with the given fields replaced by the non-null parameter values. | ||||||
| @JsonKey(includeFromJson: false, includeToJson: false) | @JsonKey(includeFromJson: false, includeToJson: false) | ||||||
| @@ -305,11 +574,11 @@ abstract mixin class $SnAuthChallengeCopyWith<$Res>  { | |||||||
|   factory $SnAuthChallengeCopyWith(SnAuthChallenge value, $Res Function(SnAuthChallenge) _then) = _$SnAuthChallengeCopyWithImpl; |   factory $SnAuthChallengeCopyWith(SnAuthChallenge value, $Res Function(SnAuthChallenge) _then) = _$SnAuthChallengeCopyWithImpl; | ||||||
| @useResult | @useResult | ||||||
| $Res call({ | $Res call({ | ||||||
|  String id, DateTime expiredAt, int stepRemain, int stepTotal, int failedAttempts, int type, List<String> blacklistFactors, List<dynamic> audiences, List<dynamic> scopes, String ipAddress, String userAgent, String? nonce, String? location, String accountId, DateTime createdAt, DateTime updatedAt, DateTime? deletedAt |  String id, DateTime expiredAt, int stepRemain, int stepTotal, int failedAttempts, int type, List<String> blacklistFactors, List<dynamic> audiences, List<dynamic> scopes, String ipAddress, String userAgent, String? nonce, GeoIpLocation? location, String accountId, DateTime createdAt, DateTime updatedAt, DateTime? deletedAt | ||||||
| }); | }); | ||||||
|  |  | ||||||
|  |  | ||||||
|  | $GeoIpLocationCopyWith<$Res>? get location; | ||||||
|  |  | ||||||
| } | } | ||||||
| /// @nodoc | /// @nodoc | ||||||
| @@ -337,14 +606,26 @@ as List<dynamic>,ipAddress: null == ipAddress ? _self.ipAddress : ipAddress // i | |||||||
| as String,userAgent: null == userAgent ? _self.userAgent : userAgent // ignore: cast_nullable_to_non_nullable | as String,userAgent: null == userAgent ? _self.userAgent : userAgent // ignore: cast_nullable_to_non_nullable | ||||||
| as String,nonce: freezed == nonce ? _self.nonce : nonce // ignore: cast_nullable_to_non_nullable | as String,nonce: freezed == nonce ? _self.nonce : nonce // ignore: cast_nullable_to_non_nullable | ||||||
| as String?,location: freezed == location ? _self.location : location // ignore: cast_nullable_to_non_nullable | as String?,location: freezed == location ? _self.location : location // ignore: cast_nullable_to_non_nullable | ||||||
| as String?,accountId: null == accountId ? _self.accountId : accountId // ignore: cast_nullable_to_non_nullable | as GeoIpLocation?,accountId: null == accountId ? _self.accountId : accountId // ignore: cast_nullable_to_non_nullable | ||||||
| as String,createdAt: null == createdAt ? _self.createdAt : createdAt // ignore: cast_nullable_to_non_nullable | as String,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,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,deletedAt: freezed == deletedAt ? _self.deletedAt : deletedAt // ignore: cast_nullable_to_non_nullable | ||||||
| as DateTime?, | as DateTime?, | ||||||
|   )); |   )); | ||||||
| } | } | ||||||
|  | /// Create a copy of SnAuthChallenge | ||||||
|  | /// with the given fields replaced by the non-null parameter values. | ||||||
|  | @override | ||||||
|  | @pragma('vm:prefer-inline') | ||||||
|  | $GeoIpLocationCopyWith<$Res>? get location { | ||||||
|  |     if (_self.location == null) { | ||||||
|  |     return null; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   return $GeoIpLocationCopyWith<$Res>(_self.location!, (value) { | ||||||
|  |     return _then(_self.copyWith(location: value)); | ||||||
|  |   }); | ||||||
|  | } | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -423,7 +704,7 @@ return $default(_that);case _: | |||||||
| /// } | /// } | ||||||
| /// ``` | /// ``` | ||||||
|  |  | ||||||
| @optionalTypeArgs TResult maybeWhen<TResult extends Object?>(TResult Function( String id,  DateTime expiredAt,  int stepRemain,  int stepTotal,  int failedAttempts,  int type,  List<String> blacklistFactors,  List<dynamic> audiences,  List<dynamic> scopes,  String ipAddress,  String userAgent,  String? nonce,  String? location,  String accountId,  DateTime createdAt,  DateTime updatedAt,  DateTime? deletedAt)?  $default,{required TResult orElse(),}) {final _that = this; | @optionalTypeArgs TResult maybeWhen<TResult extends Object?>(TResult Function( String id,  DateTime expiredAt,  int stepRemain,  int stepTotal,  int failedAttempts,  int type,  List<String> blacklistFactors,  List<dynamic> audiences,  List<dynamic> scopes,  String ipAddress,  String userAgent,  String? nonce,  GeoIpLocation? location,  String accountId,  DateTime createdAt,  DateTime updatedAt,  DateTime? deletedAt)?  $default,{required TResult orElse(),}) {final _that = this; | ||||||
| switch (_that) { | switch (_that) { | ||||||
| case _SnAuthChallenge() when $default != null: | case _SnAuthChallenge() when $default != null: | ||||||
| return $default(_that.id,_that.expiredAt,_that.stepRemain,_that.stepTotal,_that.failedAttempts,_that.type,_that.blacklistFactors,_that.audiences,_that.scopes,_that.ipAddress,_that.userAgent,_that.nonce,_that.location,_that.accountId,_that.createdAt,_that.updatedAt,_that.deletedAt);case _: | return $default(_that.id,_that.expiredAt,_that.stepRemain,_that.stepTotal,_that.failedAttempts,_that.type,_that.blacklistFactors,_that.audiences,_that.scopes,_that.ipAddress,_that.userAgent,_that.nonce,_that.location,_that.accountId,_that.createdAt,_that.updatedAt,_that.deletedAt);case _: | ||||||
| @@ -444,7 +725,7 @@ return $default(_that.id,_that.expiredAt,_that.stepRemain,_that.stepTotal,_that. | |||||||
| /// } | /// } | ||||||
| /// ``` | /// ``` | ||||||
|  |  | ||||||
| @optionalTypeArgs TResult when<TResult extends Object?>(TResult Function( String id,  DateTime expiredAt,  int stepRemain,  int stepTotal,  int failedAttempts,  int type,  List<String> blacklistFactors,  List<dynamic> audiences,  List<dynamic> scopes,  String ipAddress,  String userAgent,  String? nonce,  String? location,  String accountId,  DateTime createdAt,  DateTime updatedAt,  DateTime? deletedAt)  $default,) {final _that = this; | @optionalTypeArgs TResult when<TResult extends Object?>(TResult Function( String id,  DateTime expiredAt,  int stepRemain,  int stepTotal,  int failedAttempts,  int type,  List<String> blacklistFactors,  List<dynamic> audiences,  List<dynamic> scopes,  String ipAddress,  String userAgent,  String? nonce,  GeoIpLocation? location,  String accountId,  DateTime createdAt,  DateTime updatedAt,  DateTime? deletedAt)  $default,) {final _that = this; | ||||||
| switch (_that) { | switch (_that) { | ||||||
| case _SnAuthChallenge(): | case _SnAuthChallenge(): | ||||||
| return $default(_that.id,_that.expiredAt,_that.stepRemain,_that.stepTotal,_that.failedAttempts,_that.type,_that.blacklistFactors,_that.audiences,_that.scopes,_that.ipAddress,_that.userAgent,_that.nonce,_that.location,_that.accountId,_that.createdAt,_that.updatedAt,_that.deletedAt);} | return $default(_that.id,_that.expiredAt,_that.stepRemain,_that.stepTotal,_that.failedAttempts,_that.type,_that.blacklistFactors,_that.audiences,_that.scopes,_that.ipAddress,_that.userAgent,_that.nonce,_that.location,_that.accountId,_that.createdAt,_that.updatedAt,_that.deletedAt);} | ||||||
| @@ -461,7 +742,7 @@ return $default(_that.id,_that.expiredAt,_that.stepRemain,_that.stepTotal,_that. | |||||||
| /// } | /// } | ||||||
| /// ``` | /// ``` | ||||||
|  |  | ||||||
| @optionalTypeArgs TResult? whenOrNull<TResult extends Object?>(TResult? Function( String id,  DateTime expiredAt,  int stepRemain,  int stepTotal,  int failedAttempts,  int type,  List<String> blacklistFactors,  List<dynamic> audiences,  List<dynamic> scopes,  String ipAddress,  String userAgent,  String? nonce,  String? location,  String accountId,  DateTime createdAt,  DateTime updatedAt,  DateTime? deletedAt)?  $default,) {final _that = this; | @optionalTypeArgs TResult? whenOrNull<TResult extends Object?>(TResult? Function( String id,  DateTime expiredAt,  int stepRemain,  int stepTotal,  int failedAttempts,  int type,  List<String> blacklistFactors,  List<dynamic> audiences,  List<dynamic> scopes,  String ipAddress,  String userAgent,  String? nonce,  GeoIpLocation? location,  String accountId,  DateTime createdAt,  DateTime updatedAt,  DateTime? deletedAt)?  $default,) {final _that = this; | ||||||
| switch (_that) { | switch (_that) { | ||||||
| case _SnAuthChallenge() when $default != null: | case _SnAuthChallenge() when $default != null: | ||||||
| return $default(_that.id,_that.expiredAt,_that.stepRemain,_that.stepTotal,_that.failedAttempts,_that.type,_that.blacklistFactors,_that.audiences,_that.scopes,_that.ipAddress,_that.userAgent,_that.nonce,_that.location,_that.accountId,_that.createdAt,_that.updatedAt,_that.deletedAt);case _: | return $default(_that.id,_that.expiredAt,_that.stepRemain,_that.stepTotal,_that.failedAttempts,_that.type,_that.blacklistFactors,_that.audiences,_that.scopes,_that.ipAddress,_that.userAgent,_that.nonce,_that.location,_that.accountId,_that.createdAt,_that.updatedAt,_that.deletedAt);case _: | ||||||
| @@ -509,7 +790,7 @@ class _SnAuthChallenge implements SnAuthChallenge { | |||||||
| @override final  String ipAddress; | @override final  String ipAddress; | ||||||
| @override final  String userAgent; | @override final  String userAgent; | ||||||
| @override final  String? nonce; | @override final  String? nonce; | ||||||
| @override final  String? location; | @override final  GeoIpLocation? location; | ||||||
| @override final  String accountId; | @override final  String accountId; | ||||||
| @override final  DateTime createdAt; | @override final  DateTime createdAt; | ||||||
| @override final  DateTime updatedAt; | @override final  DateTime updatedAt; | ||||||
| @@ -548,11 +829,11 @@ abstract mixin class _$SnAuthChallengeCopyWith<$Res> implements $SnAuthChallenge | |||||||
|   factory _$SnAuthChallengeCopyWith(_SnAuthChallenge value, $Res Function(_SnAuthChallenge) _then) = __$SnAuthChallengeCopyWithImpl; |   factory _$SnAuthChallengeCopyWith(_SnAuthChallenge value, $Res Function(_SnAuthChallenge) _then) = __$SnAuthChallengeCopyWithImpl; | ||||||
| @override @useResult | @override @useResult | ||||||
| $Res call({ | $Res call({ | ||||||
|  String id, DateTime expiredAt, int stepRemain, int stepTotal, int failedAttempts, int type, List<String> blacklistFactors, List<dynamic> audiences, List<dynamic> scopes, String ipAddress, String userAgent, String? nonce, String? location, String accountId, DateTime createdAt, DateTime updatedAt, DateTime? deletedAt |  String id, DateTime expiredAt, int stepRemain, int stepTotal, int failedAttempts, int type, List<String> blacklistFactors, List<dynamic> audiences, List<dynamic> scopes, String ipAddress, String userAgent, String? nonce, GeoIpLocation? location, String accountId, DateTime createdAt, DateTime updatedAt, DateTime? deletedAt | ||||||
| }); | }); | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @override $GeoIpLocationCopyWith<$Res>? get location; | ||||||
|  |  | ||||||
| } | } | ||||||
| /// @nodoc | /// @nodoc | ||||||
| @@ -580,7 +861,7 @@ as List<dynamic>,ipAddress: null == ipAddress ? _self.ipAddress : ipAddress // i | |||||||
| as String,userAgent: null == userAgent ? _self.userAgent : userAgent // ignore: cast_nullable_to_non_nullable | as String,userAgent: null == userAgent ? _self.userAgent : userAgent // ignore: cast_nullable_to_non_nullable | ||||||
| as String,nonce: freezed == nonce ? _self.nonce : nonce // ignore: cast_nullable_to_non_nullable | as String,nonce: freezed == nonce ? _self.nonce : nonce // ignore: cast_nullable_to_non_nullable | ||||||
| as String?,location: freezed == location ? _self.location : location // ignore: cast_nullable_to_non_nullable | as String?,location: freezed == location ? _self.location : location // ignore: cast_nullable_to_non_nullable | ||||||
| as String?,accountId: null == accountId ? _self.accountId : accountId // ignore: cast_nullable_to_non_nullable | as GeoIpLocation?,accountId: null == accountId ? _self.accountId : accountId // ignore: cast_nullable_to_non_nullable | ||||||
| as String,createdAt: null == createdAt ? _self.createdAt : createdAt // ignore: cast_nullable_to_non_nullable | as String,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,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,deletedAt: freezed == deletedAt ? _self.deletedAt : deletedAt // ignore: cast_nullable_to_non_nullable | ||||||
| @@ -588,7 +869,19 @@ as DateTime?, | |||||||
|   )); |   )); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | /// Create a copy of SnAuthChallenge | ||||||
|  | /// with the given fields replaced by the non-null parameter values. | ||||||
|  | @override | ||||||
|  | @pragma('vm:prefer-inline') | ||||||
|  | $GeoIpLocationCopyWith<$Res>? get location { | ||||||
|  |     if (_self.location == null) { | ||||||
|  |     return null; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   return $GeoIpLocationCopyWith<$Res>(_self.location!, (value) { | ||||||
|  |     return _then(_self.copyWith(location: value)); | ||||||
|  |   }); | ||||||
|  | } | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
|   | |||||||
| @@ -13,6 +13,24 @@ Map<String, dynamic> _$AppTokenToJson(_AppToken instance) => <String, dynamic>{ | |||||||
|   'token': instance.token, |   'token': instance.token, | ||||||
| }; | }; | ||||||
|  |  | ||||||
|  | _GeoIpLocation _$GeoIpLocationFromJson(Map<String, dynamic> json) => | ||||||
|  |     _GeoIpLocation( | ||||||
|  |       latitude: (json['latitude'] as num).toDouble(), | ||||||
|  |       longitude: (json['longitude'] as num).toDouble(), | ||||||
|  |       countryCode: json['country_code'] as String, | ||||||
|  |       country: json['country'] as String, | ||||||
|  |       city: json['city'] as String, | ||||||
|  |     ); | ||||||
|  |  | ||||||
|  | Map<String, dynamic> _$GeoIpLocationToJson(_GeoIpLocation instance) => | ||||||
|  |     <String, dynamic>{ | ||||||
|  |       'latitude': instance.latitude, | ||||||
|  |       'longitude': instance.longitude, | ||||||
|  |       'country_code': instance.countryCode, | ||||||
|  |       'country': instance.country, | ||||||
|  |       'city': instance.city, | ||||||
|  |     }; | ||||||
|  |  | ||||||
| _SnAuthChallenge _$SnAuthChallengeFromJson(Map<String, dynamic> json) => | _SnAuthChallenge _$SnAuthChallengeFromJson(Map<String, dynamic> json) => | ||||||
|     _SnAuthChallenge( |     _SnAuthChallenge( | ||||||
|       id: json['id'] as String, |       id: json['id'] as String, | ||||||
| @@ -30,7 +48,12 @@ _SnAuthChallenge _$SnAuthChallengeFromJson(Map<String, dynamic> json) => | |||||||
|       ipAddress: json['ip_address'] as String, |       ipAddress: json['ip_address'] as String, | ||||||
|       userAgent: json['user_agent'] as String, |       userAgent: json['user_agent'] as String, | ||||||
|       nonce: json['nonce'] as String?, |       nonce: json['nonce'] as String?, | ||||||
|       location: json['location'] as String?, |       location: | ||||||
|  |           json['location'] == null | ||||||
|  |               ? null | ||||||
|  |               : GeoIpLocation.fromJson( | ||||||
|  |                 json['location'] as Map<String, dynamic>, | ||||||
|  |               ), | ||||||
|       accountId: json['account_id'] as String, |       accountId: json['account_id'] as String, | ||||||
|       createdAt: DateTime.parse(json['created_at'] as String), |       createdAt: DateTime.parse(json['created_at'] as String), | ||||||
|       updatedAt: DateTime.parse(json['updated_at'] as String), |       updatedAt: DateTime.parse(json['updated_at'] as String), | ||||||
| @@ -54,7 +77,7 @@ Map<String, dynamic> _$SnAuthChallengeToJson(_SnAuthChallenge instance) => | |||||||
|       'ip_address': instance.ipAddress, |       'ip_address': instance.ipAddress, | ||||||
|       'user_agent': instance.userAgent, |       'user_agent': instance.userAgent, | ||||||
|       'nonce': instance.nonce, |       'nonce': instance.nonce, | ||||||
|       'location': instance.location, |       'location': instance.location?.toJson(), | ||||||
|       'account_id': instance.accountId, |       'account_id': instance.accountId, | ||||||
|       'created_at': instance.createdAt.toIso8601String(), |       'created_at': instance.createdAt.toIso8601String(), | ||||||
|       'updated_at': instance.updatedAt.toIso8601String(), |       'updated_at': instance.updatedAt.toIso8601String(), | ||||||
|   | |||||||
| @@ -27,6 +27,7 @@ sealed class SnPost with _$SnPost { | |||||||
|     @Default(0) int upvotes, |     @Default(0) int upvotes, | ||||||
|     @Default(0) int downvotes, |     @Default(0) int downvotes, | ||||||
|     @Default(0) int repliesCount, |     @Default(0) int repliesCount, | ||||||
|  |     int? pinMode, | ||||||
|     String? threadedPostId, |     String? threadedPostId, | ||||||
|     SnPost? threadedPost, |     SnPost? threadedPost, | ||||||
|     String? repliedPostId, |     String? repliedPostId, | ||||||
|   | |||||||
| @@ -15,7 +15,7 @@ T _$identity<T>(T value) => value; | |||||||
| /// @nodoc | /// @nodoc | ||||||
| mixin _$SnPost { | 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; String? get slug; 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; String? get realmId; SnRealm? get realm; List<SnCloudFile> get attachments; SnPublisher get publisher; Map<String, int> get reactionsCount; Map<String, bool> get reactionsMade; List<dynamic> get reactions; List<SnPostTag> get tags; List<SnPostCategory> 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; String? get slug; int get type; Map<String, dynamic>? get meta; int get viewsUnique; int get viewsTotal; int get upvotes; int get downvotes; int get repliesCount; int? get pinMode; String? get threadedPostId; SnPost? get threadedPost; String? get repliedPostId; SnPost? get repliedPost; String? get forwardedPostId; SnPost? get forwardedPost; String? get realmId; SnRealm? get realm; List<SnCloudFile> get attachments; SnPublisher get publisher; Map<String, int> get reactionsCount; Map<String, bool> get reactionsMade; List<dynamic> get reactions; List<SnPostTag> get tags; List<SnPostCategory> get categories; List<dynamic> get collections; DateTime? get createdAt; DateTime? get updatedAt; DateTime? get deletedAt; bool get isTruncated; | ||||||
| /// Create a copy of SnPost | /// Create a copy of SnPost | ||||||
| /// with the given fields replaced by the non-null parameter values. | /// with the given fields replaced by the non-null parameter values. | ||||||
| @JsonKey(includeFromJson: false, includeToJson: false) | @JsonKey(includeFromJson: false, includeToJson: false) | ||||||
| @@ -28,16 +28,16 @@ $SnPostCopyWith<SnPost> get copyWith => _$SnPostCopyWithImpl<SnPost>(this as SnP | |||||||
|  |  | ||||||
| @override | @override | ||||||
| bool operator ==(Object other) { | 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.slug, slug) || other.slug == slug)&&(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)&&(identical(other.realmId, realmId) || other.realmId == realmId)&&(identical(other.realm, realm) || other.realm == realm)&&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)); |   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.slug, slug) || other.slug == slug)&&(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.pinMode, pinMode) || other.pinMode == pinMode)&&(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)&&(identical(other.realmId, realmId) || other.realmId == realmId)&&(identical(other.realm, realm) || other.realm == realm)&&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) | @JsonKey(includeFromJson: false, includeToJson: false) | ||||||
| @override | @override | ||||||
| int get hashCode => Object.hashAll([runtimeType,id,title,description,language,editedAt,publishedAt,visibility,content,slug,type,const DeepCollectionEquality().hash(meta),viewsUnique,viewsTotal,upvotes,downvotes,repliesCount,threadedPostId,threadedPost,repliedPostId,repliedPost,forwardedPostId,forwardedPost,realmId,realm,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]); | int get hashCode => Object.hashAll([runtimeType,id,title,description,language,editedAt,publishedAt,visibility,content,slug,type,const DeepCollectionEquality().hash(meta),viewsUnique,viewsTotal,upvotes,downvotes,repliesCount,pinMode,threadedPostId,threadedPost,repliedPostId,repliedPost,forwardedPostId,forwardedPost,realmId,realm,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 | @override | ||||||
| String toString() { | String toString() { | ||||||
|   return 'SnPost(id: $id, title: $title, description: $description, language: $language, editedAt: $editedAt, publishedAt: $publishedAt, visibility: $visibility, content: $content, slug: $slug, 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, realmId: $realmId, realm: $realm, attachments: $attachments, publisher: $publisher, reactionsCount: $reactionsCount, reactionsMade: $reactionsMade, 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, slug: $slug, type: $type, meta: $meta, viewsUnique: $viewsUnique, viewsTotal: $viewsTotal, upvotes: $upvotes, downvotes: $downvotes, repliesCount: $repliesCount, pinMode: $pinMode, threadedPostId: $threadedPostId, threadedPost: $threadedPost, repliedPostId: $repliedPostId, repliedPost: $repliedPost, forwardedPostId: $forwardedPostId, forwardedPost: $forwardedPost, realmId: $realmId, realm: $realm, 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; |   factory $SnPostCopyWith(SnPost value, $Res Function(SnPost) _then) = _$SnPostCopyWithImpl; | ||||||
| @useResult | @useResult | ||||||
| $Res call({ | $Res call({ | ||||||
|  String id, String? title, String? description, String? language, DateTime? editedAt, DateTime? publishedAt, int visibility, String? content, String? slug, 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, String? realmId, SnRealm? realm, List<SnCloudFile> attachments, SnPublisher publisher, Map<String, int> reactionsCount, Map<String, bool> reactionsMade, List<dynamic> reactions, List<SnPostTag> tags, List<SnPostCategory> 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, String? slug, int type, Map<String, dynamic>? meta, int viewsUnique, int viewsTotal, int upvotes, int downvotes, int repliesCount, int? pinMode, String? threadedPostId, SnPost? threadedPost, String? repliedPostId, SnPost? repliedPost, String? forwardedPostId, SnPost? forwardedPost, String? realmId, SnRealm? realm, List<SnCloudFile> attachments, SnPublisher publisher, Map<String, int> reactionsCount, Map<String, bool> reactionsMade, List<dynamic> reactions, List<SnPostTag> tags, List<SnPostCategory> categories, List<dynamic> collections, DateTime? createdAt, DateTime? updatedAt, DateTime? deletedAt, bool isTruncated | ||||||
| }); | }); | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -65,7 +65,7 @@ class _$SnPostCopyWithImpl<$Res> | |||||||
|  |  | ||||||
| /// Create a copy of SnPost | /// Create a copy of SnPost | ||||||
| /// with the given fields replaced by the non-null parameter values. | /// 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? slug = 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? realmId = freezed,Object? realm = 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,}) { | @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? slug = freezed,Object? type = null,Object? meta = freezed,Object? viewsUnique = null,Object? viewsTotal = null,Object? upvotes = null,Object? downvotes = null,Object? repliesCount = null,Object? pinMode = freezed,Object? threadedPostId = freezed,Object? threadedPost = freezed,Object? repliedPostId = freezed,Object? repliedPost = freezed,Object? forwardedPostId = freezed,Object? forwardedPost = freezed,Object? realmId = freezed,Object? realm = 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( |   return _then(_self.copyWith( | ||||||
| id: null == id ? _self.id : id // ignore: cast_nullable_to_non_nullable | 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 | as String,title: freezed == title ? _self.title : title // ignore: cast_nullable_to_non_nullable | ||||||
| @@ -83,7 +83,8 @@ as int,viewsTotal: null == viewsTotal ? _self.viewsTotal : viewsTotal // ignore: | |||||||
| as int,upvotes: null == upvotes ? _self.upvotes : upvotes // ignore: cast_nullable_to_non_nullable | as int,upvotes: null == upvotes ? _self.upvotes : upvotes // ignore: cast_nullable_to_non_nullable | ||||||
| as int,downvotes: null == downvotes ? _self.downvotes : downvotes // ignore: cast_nullable_to_non_nullable | as int,downvotes: null == downvotes ? _self.downvotes : downvotes // ignore: cast_nullable_to_non_nullable | ||||||
| as int,repliesCount: null == repliesCount ? _self.repliesCount : repliesCount // ignore: cast_nullable_to_non_nullable | as int,repliesCount: null == repliesCount ? _self.repliesCount : repliesCount // ignore: cast_nullable_to_non_nullable | ||||||
| as int,threadedPostId: freezed == threadedPostId ? _self.threadedPostId : threadedPostId // ignore: cast_nullable_to_non_nullable | as int,pinMode: freezed == pinMode ? _self.pinMode : pinMode // ignore: cast_nullable_to_non_nullable | ||||||
|  | as int?,threadedPostId: freezed == threadedPostId ? _self.threadedPostId : threadedPostId // ignore: cast_nullable_to_non_nullable | ||||||
| as String?,threadedPost: freezed == threadedPost ? _self.threadedPost : threadedPost // ignore: cast_nullable_to_non_nullable | as String?,threadedPost: freezed == threadedPost ? _self.threadedPost : threadedPost // ignore: cast_nullable_to_non_nullable | ||||||
| as SnPost?,repliedPostId: freezed == repliedPostId ? _self.repliedPostId : repliedPostId // ignore: cast_nullable_to_non_nullable | as SnPost?,repliedPostId: freezed == repliedPostId ? _self.repliedPostId : repliedPostId // ignore: cast_nullable_to_non_nullable | ||||||
| as String?,repliedPost: freezed == repliedPost ? _self.repliedPost : repliedPost // ignore: cast_nullable_to_non_nullable | as String?,repliedPost: freezed == repliedPost ? _self.repliedPost : repliedPost // ignore: cast_nullable_to_non_nullable | ||||||
| @@ -242,10 +243,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,  String? slug,  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,  String? realmId,  SnRealm? realm,  List<SnCloudFile> attachments,  SnPublisher publisher,  Map<String, int> reactionsCount,  Map<String, bool> reactionsMade,  List<dynamic> reactions,  List<SnPostTag> tags,  List<SnPostCategory> 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,  String? slug,  int type,  Map<String, dynamic>? meta,  int viewsUnique,  int viewsTotal,  int upvotes,  int downvotes,  int repliesCount,  int? pinMode,  String? threadedPostId,  SnPost? threadedPost,  String? repliedPostId,  SnPost? repliedPost,  String? forwardedPostId,  SnPost? forwardedPost,  String? realmId,  SnRealm? realm,  List<SnCloudFile> attachments,  SnPublisher publisher,  Map<String, int> reactionsCount,  Map<String, bool> reactionsMade,  List<dynamic> reactions,  List<SnPostTag> tags,  List<SnPostCategory> categories,  List<dynamic> collections,  DateTime? createdAt,  DateTime? updatedAt,  DateTime? deletedAt,  bool isTruncated)?  $default,{required TResult orElse(),}) {final _that = this; | ||||||
| switch (_that) { | switch (_that) { | ||||||
| case _SnPost() when $default != null: | case _SnPost() when $default != null: | ||||||
| return $default(_that.id,_that.title,_that.description,_that.language,_that.editedAt,_that.publishedAt,_that.visibility,_that.content,_that.slug,_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.realmId,_that.realm,_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 $default(_that.id,_that.title,_that.description,_that.language,_that.editedAt,_that.publishedAt,_that.visibility,_that.content,_that.slug,_that.type,_that.meta,_that.viewsUnique,_that.viewsTotal,_that.upvotes,_that.downvotes,_that.repliesCount,_that.pinMode,_that.threadedPostId,_that.threadedPost,_that.repliedPostId,_that.repliedPost,_that.forwardedPostId,_that.forwardedPost,_that.realmId,_that.realm,_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(); |   return orElse(); | ||||||
|  |  | ||||||
| } | } | ||||||
| @@ -263,10 +264,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,  String? slug,  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,  String? realmId,  SnRealm? realm,  List<SnCloudFile> attachments,  SnPublisher publisher,  Map<String, int> reactionsCount,  Map<String, bool> reactionsMade,  List<dynamic> reactions,  List<SnPostTag> tags,  List<SnPostCategory> 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,  String? slug,  int type,  Map<String, dynamic>? meta,  int viewsUnique,  int viewsTotal,  int upvotes,  int downvotes,  int repliesCount,  int? pinMode,  String? threadedPostId,  SnPost? threadedPost,  String? repliedPostId,  SnPost? repliedPost,  String? forwardedPostId,  SnPost? forwardedPost,  String? realmId,  SnRealm? realm,  List<SnCloudFile> attachments,  SnPublisher publisher,  Map<String, int> reactionsCount,  Map<String, bool> reactionsMade,  List<dynamic> reactions,  List<SnPostTag> tags,  List<SnPostCategory> categories,  List<dynamic> collections,  DateTime? createdAt,  DateTime? updatedAt,  DateTime? deletedAt,  bool isTruncated)  $default,) {final _that = this; | ||||||
| switch (_that) { | switch (_that) { | ||||||
| case _SnPost(): | case _SnPost(): | ||||||
| return $default(_that.id,_that.title,_that.description,_that.language,_that.editedAt,_that.publishedAt,_that.visibility,_that.content,_that.slug,_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.realmId,_that.realm,_that.attachments,_that.publisher,_that.reactionsCount,_that.reactionsMade,_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.slug,_that.type,_that.meta,_that.viewsUnique,_that.viewsTotal,_that.upvotes,_that.downvotes,_that.repliesCount,_that.pinMode,_that.threadedPostId,_that.threadedPost,_that.repliedPostId,_that.repliedPost,_that.forwardedPostId,_that.forwardedPost,_that.realmId,_that.realm,_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` | /// A variant of `when` that fallback to returning `null` | ||||||
| /// | /// | ||||||
| @@ -280,10 +281,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,  String? slug,  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,  String? realmId,  SnRealm? realm,  List<SnCloudFile> attachments,  SnPublisher publisher,  Map<String, int> reactionsCount,  Map<String, bool> reactionsMade,  List<dynamic> reactions,  List<SnPostTag> tags,  List<SnPostCategory> 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,  String? slug,  int type,  Map<String, dynamic>? meta,  int viewsUnique,  int viewsTotal,  int upvotes,  int downvotes,  int repliesCount,  int? pinMode,  String? threadedPostId,  SnPost? threadedPost,  String? repliedPostId,  SnPost? repliedPost,  String? forwardedPostId,  SnPost? forwardedPost,  String? realmId,  SnRealm? realm,  List<SnCloudFile> attachments,  SnPublisher publisher,  Map<String, int> reactionsCount,  Map<String, bool> reactionsMade,  List<dynamic> reactions,  List<SnPostTag> tags,  List<SnPostCategory> categories,  List<dynamic> collections,  DateTime? createdAt,  DateTime? updatedAt,  DateTime? deletedAt,  bool isTruncated)?  $default,) {final _that = this; | ||||||
| switch (_that) { | switch (_that) { | ||||||
| case _SnPost() when $default != null: | case _SnPost() when $default != null: | ||||||
| return $default(_that.id,_that.title,_that.description,_that.language,_that.editedAt,_that.publishedAt,_that.visibility,_that.content,_that.slug,_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.realmId,_that.realm,_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 $default(_that.id,_that.title,_that.description,_that.language,_that.editedAt,_that.publishedAt,_that.visibility,_that.content,_that.slug,_that.type,_that.meta,_that.viewsUnique,_that.viewsTotal,_that.upvotes,_that.downvotes,_that.repliesCount,_that.pinMode,_that.threadedPostId,_that.threadedPost,_that.repliedPostId,_that.repliedPost,_that.forwardedPostId,_that.forwardedPost,_that.realmId,_that.realm,_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; |   return null; | ||||||
|  |  | ||||||
| } | } | ||||||
| @@ -295,7 +296,7 @@ return $default(_that.id,_that.title,_that.description,_that.language,_that.edit | |||||||
| @JsonSerializable() | @JsonSerializable() | ||||||
|  |  | ||||||
| class _SnPost implements SnPost { | 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.slug, 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, this.realmId, this.realm, 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<SnPostTag> tags = const [], final  List<SnPostCategory> 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; |   const _SnPost({required this.id, this.title, this.description, this.language, this.editedAt, this.publishedAt = null, this.visibility = 0, this.content, this.slug, this.type = 0, final  Map<String, dynamic>? meta, this.viewsUnique = 0, this.viewsTotal = 0, this.upvotes = 0, this.downvotes = 0, this.repliesCount = 0, this.pinMode, this.threadedPostId, this.threadedPost, this.repliedPostId, this.repliedPost, this.forwardedPostId, this.forwardedPost, this.realmId, this.realm, 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<SnPostTag> tags = const [], final  List<SnPostCategory> 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); |   factory _SnPost.fromJson(Map<String, dynamic> json) => _$SnPostFromJson(json); | ||||||
|  |  | ||||||
| @override final  String id; | @override final  String id; | ||||||
| @@ -322,6 +323,7 @@ class _SnPost implements SnPost { | |||||||
| @override@JsonKey() final  int upvotes; | @override@JsonKey() final  int upvotes; | ||||||
| @override@JsonKey() final  int downvotes; | @override@JsonKey() final  int downvotes; | ||||||
| @override@JsonKey() final  int repliesCount; | @override@JsonKey() final  int repliesCount; | ||||||
|  | @override final  int? pinMode; | ||||||
| @override final  String? threadedPostId; | @override final  String? threadedPostId; | ||||||
| @override final  SnPost? threadedPost; | @override final  SnPost? threadedPost; | ||||||
| @override final  String? repliedPostId; | @override final  String? repliedPostId; | ||||||
| @@ -398,16 +400,16 @@ Map<String, dynamic> toJson() { | |||||||
|  |  | ||||||
| @override | @override | ||||||
| bool operator ==(Object other) { | 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.slug, slug) || other.slug == slug)&&(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)&&(identical(other.realmId, realmId) || other.realmId == realmId)&&(identical(other.realm, realm) || other.realm == realm)&&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)); |   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.slug, slug) || other.slug == slug)&&(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.pinMode, pinMode) || other.pinMode == pinMode)&&(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)&&(identical(other.realmId, realmId) || other.realmId == realmId)&&(identical(other.realm, realm) || other.realm == realm)&&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) | @JsonKey(includeFromJson: false, includeToJson: false) | ||||||
| @override | @override | ||||||
| int get hashCode => Object.hashAll([runtimeType,id,title,description,language,editedAt,publishedAt,visibility,content,slug,type,const DeepCollectionEquality().hash(_meta),viewsUnique,viewsTotal,upvotes,downvotes,repliesCount,threadedPostId,threadedPost,repliedPostId,repliedPost,forwardedPostId,forwardedPost,realmId,realm,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]); | int get hashCode => Object.hashAll([runtimeType,id,title,description,language,editedAt,publishedAt,visibility,content,slug,type,const DeepCollectionEquality().hash(_meta),viewsUnique,viewsTotal,upvotes,downvotes,repliesCount,pinMode,threadedPostId,threadedPost,repliedPostId,repliedPost,forwardedPostId,forwardedPost,realmId,realm,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 | @override | ||||||
| String toString() { | String toString() { | ||||||
|   return 'SnPost(id: $id, title: $title, description: $description, language: $language, editedAt: $editedAt, publishedAt: $publishedAt, visibility: $visibility, content: $content, slug: $slug, 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, realmId: $realmId, realm: $realm, attachments: $attachments, publisher: $publisher, reactionsCount: $reactionsCount, reactionsMade: $reactionsMade, 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, slug: $slug, type: $type, meta: $meta, viewsUnique: $viewsUnique, viewsTotal: $viewsTotal, upvotes: $upvotes, downvotes: $downvotes, repliesCount: $repliesCount, pinMode: $pinMode, threadedPostId: $threadedPostId, threadedPost: $threadedPost, repliedPostId: $repliedPostId, repliedPost: $repliedPost, forwardedPostId: $forwardedPostId, forwardedPost: $forwardedPost, realmId: $realmId, realm: $realm, attachments: $attachments, publisher: $publisher, reactionsCount: $reactionsCount, reactionsMade: $reactionsMade, reactions: $reactions, tags: $tags, categories: $categories, collections: $collections, createdAt: $createdAt, updatedAt: $updatedAt, deletedAt: $deletedAt, isTruncated: $isTruncated)'; | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -418,7 +420,7 @@ abstract mixin class _$SnPostCopyWith<$Res> implements $SnPostCopyWith<$Res> { | |||||||
|   factory _$SnPostCopyWith(_SnPost value, $Res Function(_SnPost) _then) = __$SnPostCopyWithImpl; |   factory _$SnPostCopyWith(_SnPost value, $Res Function(_SnPost) _then) = __$SnPostCopyWithImpl; | ||||||
| @override @useResult | @override @useResult | ||||||
| $Res call({ | $Res call({ | ||||||
|  String id, String? title, String? description, String? language, DateTime? editedAt, DateTime? publishedAt, int visibility, String? content, String? slug, 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, String? realmId, SnRealm? realm, List<SnCloudFile> attachments, SnPublisher publisher, Map<String, int> reactionsCount, Map<String, bool> reactionsMade, List<dynamic> reactions, List<SnPostTag> tags, List<SnPostCategory> 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, String? slug, int type, Map<String, dynamic>? meta, int viewsUnique, int viewsTotal, int upvotes, int downvotes, int repliesCount, int? pinMode, String? threadedPostId, SnPost? threadedPost, String? repliedPostId, SnPost? repliedPost, String? forwardedPostId, SnPost? forwardedPost, String? realmId, SnRealm? realm, List<SnCloudFile> attachments, SnPublisher publisher, Map<String, int> reactionsCount, Map<String, bool> reactionsMade, List<dynamic> reactions, List<SnPostTag> tags, List<SnPostCategory> categories, List<dynamic> collections, DateTime? createdAt, DateTime? updatedAt, DateTime? deletedAt, bool isTruncated | ||||||
| }); | }); | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -435,7 +437,7 @@ class __$SnPostCopyWithImpl<$Res> | |||||||
|  |  | ||||||
| /// Create a copy of SnPost | /// Create a copy of SnPost | ||||||
| /// with the given fields replaced by the non-null parameter values. | /// 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? slug = 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? realmId = freezed,Object? realm = 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,}) { | @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? slug = freezed,Object? type = null,Object? meta = freezed,Object? viewsUnique = null,Object? viewsTotal = null,Object? upvotes = null,Object? downvotes = null,Object? repliesCount = null,Object? pinMode = freezed,Object? threadedPostId = freezed,Object? threadedPost = freezed,Object? repliedPostId = freezed,Object? repliedPost = freezed,Object? forwardedPostId = freezed,Object? forwardedPost = freezed,Object? realmId = freezed,Object? realm = 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( |   return _then(_SnPost( | ||||||
| id: null == id ? _self.id : id // ignore: cast_nullable_to_non_nullable | 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 | as String,title: freezed == title ? _self.title : title // ignore: cast_nullable_to_non_nullable | ||||||
| @@ -453,7 +455,8 @@ as int,viewsTotal: null == viewsTotal ? _self.viewsTotal : viewsTotal // ignore: | |||||||
| as int,upvotes: null == upvotes ? _self.upvotes : upvotes // ignore: cast_nullable_to_non_nullable | as int,upvotes: null == upvotes ? _self.upvotes : upvotes // ignore: cast_nullable_to_non_nullable | ||||||
| as int,downvotes: null == downvotes ? _self.downvotes : downvotes // ignore: cast_nullable_to_non_nullable | as int,downvotes: null == downvotes ? _self.downvotes : downvotes // ignore: cast_nullable_to_non_nullable | ||||||
| as int,repliesCount: null == repliesCount ? _self.repliesCount : repliesCount // ignore: cast_nullable_to_non_nullable | as int,repliesCount: null == repliesCount ? _self.repliesCount : repliesCount // ignore: cast_nullable_to_non_nullable | ||||||
| as int,threadedPostId: freezed == threadedPostId ? _self.threadedPostId : threadedPostId // ignore: cast_nullable_to_non_nullable | as int,pinMode: freezed == pinMode ? _self.pinMode : pinMode // ignore: cast_nullable_to_non_nullable | ||||||
|  | as int?,threadedPostId: freezed == threadedPostId ? _self.threadedPostId : threadedPostId // ignore: cast_nullable_to_non_nullable | ||||||
| as String?,threadedPost: freezed == threadedPost ? _self.threadedPost : threadedPost // ignore: cast_nullable_to_non_nullable | as String?,threadedPost: freezed == threadedPost ? _self.threadedPost : threadedPost // ignore: cast_nullable_to_non_nullable | ||||||
| as SnPost?,repliedPostId: freezed == repliedPostId ? _self.repliedPostId : repliedPostId // ignore: cast_nullable_to_non_nullable | as SnPost?,repliedPostId: freezed == repliedPostId ? _self.repliedPostId : repliedPostId // ignore: cast_nullable_to_non_nullable | ||||||
| as String?,repliedPost: freezed == repliedPost ? _self.repliedPost : repliedPost // ignore: cast_nullable_to_non_nullable | as String?,repliedPost: freezed == repliedPost ? _self.repliedPost : repliedPost // ignore: cast_nullable_to_non_nullable | ||||||
|   | |||||||
| @@ -29,6 +29,7 @@ _SnPost _$SnPostFromJson(Map<String, dynamic> json) => _SnPost( | |||||||
|   upvotes: (json['upvotes'] as num?)?.toInt() ?? 0, |   upvotes: (json['upvotes'] as num?)?.toInt() ?? 0, | ||||||
|   downvotes: (json['downvotes'] as num?)?.toInt() ?? 0, |   downvotes: (json['downvotes'] as num?)?.toInt() ?? 0, | ||||||
|   repliesCount: (json['replies_count'] as num?)?.toInt() ?? 0, |   repliesCount: (json['replies_count'] as num?)?.toInt() ?? 0, | ||||||
|  |   pinMode: (json['pin_mode'] as num?)?.toInt(), | ||||||
|   threadedPostId: json['threaded_post_id'] as String?, |   threadedPostId: json['threaded_post_id'] as String?, | ||||||
|   threadedPost: |   threadedPost: | ||||||
|       json['threaded_post'] == null |       json['threaded_post'] == null | ||||||
| @@ -109,6 +110,7 @@ Map<String, dynamic> _$SnPostToJson(_SnPost instance) => <String, dynamic>{ | |||||||
|   'upvotes': instance.upvotes, |   'upvotes': instance.upvotes, | ||||||
|   'downvotes': instance.downvotes, |   'downvotes': instance.downvotes, | ||||||
|   'replies_count': instance.repliesCount, |   'replies_count': instance.repliesCount, | ||||||
|  |   'pin_mode': instance.pinMode, | ||||||
|   'threaded_post_id': instance.threadedPostId, |   'threaded_post_id': instance.threadedPostId, | ||||||
|   'threaded_post': instance.threadedPost?.toJson(), |   'threaded_post': instance.threadedPost?.toJson(), | ||||||
|   'replied_post_id': instance.repliedPostId, |   'replied_post_id': instance.repliedPostId, | ||||||
|   | |||||||
| @@ -1,7 +1,7 @@ | |||||||
| import 'package:easy_localization/easy_localization.dart'; | import 'package:easy_localization/easy_localization.dart'; | ||||||
| import 'package:freezed_annotation/freezed_annotation.dart'; | import 'package:freezed_annotation/freezed_annotation.dart'; | ||||||
| import 'package:island/models/post.dart'; | import 'package:island/models/post.dart'; | ||||||
| import 'package:island/services/text.dart'; | import 'package:island/utils/text.dart'; | ||||||
|  |  | ||||||
| part 'post_category.freezed.dart'; | part 'post_category.freezed.dart'; | ||||||
| part 'post_category.g.dart'; | part 'post_category.g.dart'; | ||||||
|   | |||||||
| @@ -9,6 +9,7 @@ import 'package:freezed_annotation/freezed_annotation.dart'; | |||||||
| import 'package:riverpod_annotation/riverpod_annotation.dart'; | import 'package:riverpod_annotation/riverpod_annotation.dart'; | ||||||
| import 'package:island/pods/network.dart'; | import 'package:island/pods/network.dart'; | ||||||
| import 'package:island/models/chat.dart'; | import 'package:island/models/chat.dart'; | ||||||
|  | import 'package:wakelock_plus/wakelock_plus.dart'; | ||||||
|  |  | ||||||
| part 'call.g.dart'; | part 'call.g.dart'; | ||||||
| part 'call.freezed.dart'; | part 'call.freezed.dart'; | ||||||
| @@ -54,7 +55,7 @@ sealed class CallParticipantLive with _$CallParticipantLive { | |||||||
|   bool get hasAudio => remoteParticipant.hasAudio; |   bool get hasAudio => remoteParticipant.hasAudio; | ||||||
| } | } | ||||||
|  |  | ||||||
| @riverpod | @Riverpod(keepAlive: true) | ||||||
| class CallNotifier extends _$CallNotifier { | class CallNotifier extends _$CallNotifier { | ||||||
|   Room? _room; |   Room? _room; | ||||||
|   LocalParticipant? _localParticipant; |   LocalParticipant? _localParticipant; | ||||||
| @@ -277,14 +278,27 @@ class CallNotifier extends _$CallNotifier { | |||||||
|  |  | ||||||
|         // Listen for connection updates |         // Listen for connection updates | ||||||
|         _room!.addListener(() { |         _room!.addListener(() { | ||||||
|  |           final wasConnected = state.isConnected; | ||||||
|  |           final isNowConnected = | ||||||
|  |               _room!.connectionState == ConnectionState.connected; | ||||||
|           state = state.copyWith( |           state = state.copyWith( | ||||||
|             isConnected: _room!.connectionState == ConnectionState.connected, |             isConnected: isNowConnected, | ||||||
|             isMicrophoneEnabled: _localParticipant!.isMicrophoneEnabled(), |             isMicrophoneEnabled: _localParticipant!.isMicrophoneEnabled(), | ||||||
|             isCameraEnabled: _localParticipant!.isCameraEnabled(), |             isCameraEnabled: _localParticipant!.isCameraEnabled(), | ||||||
|             isScreenSharing: _localParticipant!.isScreenShareEnabled(), |             isScreenSharing: _localParticipant!.isScreenShareEnabled(), | ||||||
|           ); |           ); | ||||||
|  |           // Enable wakelock when call connects | ||||||
|  |           if (!wasConnected && isNowConnected) { | ||||||
|  |             WakelockPlus.enable(); | ||||||
|  |           } | ||||||
|  |           // Disable wakelock when call disconnects | ||||||
|  |           else if (wasConnected && !isNowConnected) { | ||||||
|  |             WakelockPlus.disable(); | ||||||
|  |           } | ||||||
|         }); |         }); | ||||||
|         state = state.copyWith(isConnected: true); |         state = state.copyWith(isConnected: true); | ||||||
|  |         // Enable wakelock when call connects | ||||||
|  |         WakelockPlus.enable(); | ||||||
|       } else { |       } else { | ||||||
|         state = state.copyWith(error: 'Failed to join room'); |         state = state.copyWith(error: 'Failed to join room'); | ||||||
|       } |       } | ||||||
| @@ -344,6 +358,8 @@ class CallNotifier extends _$CallNotifier { | |||||||
|         isCameraEnabled: false, |         isCameraEnabled: false, | ||||||
|         isScreenSharing: false, |         isScreenSharing: false, | ||||||
|       ); |       ); | ||||||
|  |       // Disable wakelock when call disconnects | ||||||
|  |       WakelockPlus.disable(); | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
| @@ -381,5 +397,7 @@ class CallNotifier extends _$CallNotifier { | |||||||
|     _durationTimer?.cancel(); |     _durationTimer?.cancel(); | ||||||
|     _roomId = null; |     _roomId = null; | ||||||
|     participantsVolumes = {}; |     participantsVolumes = {}; | ||||||
|  |     // Disable wakelock when disposing | ||||||
|  |     WakelockPlus.disable(); | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -6,22 +6,19 @@ part of 'call.dart'; | |||||||
| // RiverpodGenerator | // RiverpodGenerator | ||||||
| // ************************************************************************** | // ************************************************************************** | ||||||
|  |  | ||||||
| String _$callNotifierHash() => r'18fb807f067eecd3ea42631c1426c3e5f1fb4280'; | String _$callNotifierHash() => r'eb9bd41b97e9b5e9d54007c8327edb6567458846'; | ||||||
|  |  | ||||||
| /// See also [CallNotifier]. | /// See also [CallNotifier]. | ||||||
| @ProviderFor(CallNotifier) | @ProviderFor(CallNotifier) | ||||||
| final callNotifierProvider = | final callNotifierProvider = NotifierProvider<CallNotifier, CallState>.internal( | ||||||
|     AutoDisposeNotifierProvider<CallNotifier, CallState>.internal( |   CallNotifier.new, | ||||||
|       CallNotifier.new, |   name: r'callNotifierProvider', | ||||||
|       name: r'callNotifierProvider', |   debugGetCreateSourceHash: | ||||||
|       debugGetCreateSourceHash: |       const bool.fromEnvironment('dart.vm.product') ? null : _$callNotifierHash, | ||||||
|           const bool.fromEnvironment('dart.vm.product') |   dependencies: null, | ||||||
|               ? null |   allTransitiveDependencies: null, | ||||||
|               : _$callNotifierHash, | ); | ||||||
|       dependencies: null, |  | ||||||
|       allTransitiveDependencies: null, |  | ||||||
|     ); |  | ||||||
|  |  | ||||||
| typedef _$CallNotifier = AutoDisposeNotifier<CallState>; | typedef _$CallNotifier = Notifier<CallState>; | ||||||
| // ignore_for_file: type=lint | // 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 | // 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 | ||||||
|   | |||||||
| @@ -15,10 +15,12 @@ const kNetworkServerStoreKey = 'app_server_url'; | |||||||
|  |  | ||||||
| const kAppbarTransparentStoreKey = 'app_bar_transparent'; | const kAppbarTransparentStoreKey = 'app_bar_transparent'; | ||||||
| const kAppBackgroundStoreKey = 'app_has_background'; | const kAppBackgroundStoreKey = 'app_has_background'; | ||||||
|  | const kAppShowBackgroundImage = 'app_show_background_image'; | ||||||
| const kAppColorSchemeStoreKey = 'app_color_scheme'; | const kAppColorSchemeStoreKey = 'app_color_scheme'; | ||||||
| const kAppNotifyWithHaptic = 'app_notify_with_haptic'; | const kAppNotifyWithHaptic = 'app_notify_with_haptic'; | ||||||
| const kAppCustomFonts = 'app_custom_fonts'; | const kAppCustomFonts = 'app_custom_fonts'; | ||||||
| const kAppAutoTranslate = 'app_auto_translate'; | const kAppAutoTranslate = 'app_auto_translate'; | ||||||
|  | const kAppDataSavingMode = 'app_data_saving_mode'; | ||||||
| const kAppSoundEffects = 'app_sound_effects'; | const kAppSoundEffects = 'app_sound_effects'; | ||||||
| const kAppAprilFoolFeatures = 'app_april_fool_features'; | const kAppAprilFoolFeatures = 'app_april_fool_features'; | ||||||
| const kAppWindowSize = 'app_window_size'; | const kAppWindowSize = 'app_window_size'; | ||||||
| @@ -54,10 +56,12 @@ final serverUrlProvider = Provider<String>((ref) { | |||||||
| sealed class AppSettings with _$AppSettings { | sealed class AppSettings with _$AppSettings { | ||||||
|   const factory AppSettings({ |   const factory AppSettings({ | ||||||
|     required bool autoTranslate, |     required bool autoTranslate, | ||||||
|  |     required bool dataSavingMode, | ||||||
|     required bool soundEffects, |     required bool soundEffects, | ||||||
|     required bool aprilFoolFeatures, |     required bool aprilFoolFeatures, | ||||||
|     required bool enterToSend, |     required bool enterToSend, | ||||||
|     required bool appBarTransparent, |     required bool appBarTransparent, | ||||||
|  |     required bool showBackgroundImage, | ||||||
|     required String? customFonts, |     required String? customFonts, | ||||||
|     required int? appColorScheme, // The color stored via the int type |     required int? appColorScheme, // The color stored via the int type | ||||||
|     required Size? windowSize, // The window size for desktop platforms |     required Size? windowSize, // The window size for desktop platforms | ||||||
| @@ -71,10 +75,12 @@ class AppSettingsNotifier extends _$AppSettingsNotifier { | |||||||
|     final prefs = ref.watch(sharedPreferencesProvider); |     final prefs = ref.watch(sharedPreferencesProvider); | ||||||
|     return AppSettings( |     return AppSettings( | ||||||
|       autoTranslate: prefs.getBool(kAppAutoTranslate) ?? false, |       autoTranslate: prefs.getBool(kAppAutoTranslate) ?? false, | ||||||
|  |       dataSavingMode: prefs.getBool(kAppDataSavingMode) ?? false, | ||||||
|       soundEffects: prefs.getBool(kAppSoundEffects) ?? true, |       soundEffects: prefs.getBool(kAppSoundEffects) ?? true, | ||||||
|       aprilFoolFeatures: prefs.getBool(kAppAprilFoolFeatures) ?? true, |       aprilFoolFeatures: prefs.getBool(kAppAprilFoolFeatures) ?? true, | ||||||
|       enterToSend: prefs.getBool(kAppEnterToSend) ?? true, |       enterToSend: prefs.getBool(kAppEnterToSend) ?? true, | ||||||
|       appBarTransparent: prefs.getBool(kAppbarTransparentStoreKey) ?? false, |       appBarTransparent: prefs.getBool(kAppbarTransparentStoreKey) ?? false, | ||||||
|  |       showBackgroundImage: prefs.getBool(kAppShowBackgroundImage) ?? true, | ||||||
|       customFonts: prefs.getString(kAppCustomFonts), |       customFonts: prefs.getString(kAppCustomFonts), | ||||||
|       appColorScheme: prefs.getInt(kAppColorSchemeStoreKey), |       appColorScheme: prefs.getInt(kAppColorSchemeStoreKey), | ||||||
|       windowSize: _getWindowSizeFromPrefs(prefs), |       windowSize: _getWindowSizeFromPrefs(prefs), | ||||||
| @@ -104,6 +110,12 @@ class AppSettingsNotifier extends _$AppSettingsNotifier { | |||||||
|     state = state.copyWith(autoTranslate: value); |     state = state.copyWith(autoTranslate: value); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   void setDataSavingMode(bool value){ | ||||||
|  |     final prefs = ref.read(sharedPreferencesProvider); | ||||||
|  |     prefs.setBool(kAppDataSavingMode, value); | ||||||
|  |     state = state.copyWith(dataSavingMode: value); | ||||||
|  |   } | ||||||
|  |  | ||||||
|   void setSoundEffects(bool value) { |   void setSoundEffects(bool value) { | ||||||
|     final prefs = ref.read(sharedPreferencesProvider); |     final prefs = ref.read(sharedPreferencesProvider); | ||||||
|     prefs.setBool(kAppSoundEffects, value); |     prefs.setBool(kAppSoundEffects, value); | ||||||
| @@ -129,6 +141,12 @@ class AppSettingsNotifier extends _$AppSettingsNotifier { | |||||||
|     ref.read(themeProvider.notifier).reloadTheme(); |     ref.read(themeProvider.notifier).reloadTheme(); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   void setShowBackgroundImage(bool value) { | ||||||
|  |     final prefs = ref.read(sharedPreferencesProvider); | ||||||
|  |     prefs.setBool(kAppShowBackgroundImage, value); | ||||||
|  |     state = state.copyWith(showBackgroundImage: value); | ||||||
|  |   } | ||||||
|  |  | ||||||
|   void setCustomFonts(String? value) { |   void setCustomFonts(String? value) { | ||||||
|     final prefs = ref.read(sharedPreferencesProvider); |     final prefs = ref.read(sharedPreferencesProvider); | ||||||
|     prefs.setString(kAppCustomFonts, value ?? ''); |     prefs.setString(kAppCustomFonts, value ?? ''); | ||||||
|   | |||||||
| @@ -14,7 +14,7 @@ T _$identity<T>(T value) => value; | |||||||
| /// @nodoc | /// @nodoc | ||||||
| mixin _$AppSettings { | mixin _$AppSettings { | ||||||
|  |  | ||||||
|  bool get autoTranslate; bool get soundEffects; bool get aprilFoolFeatures; bool get enterToSend; bool get appBarTransparent; String? get customFonts; int? get appColorScheme;// The color stored via the int type |  bool get autoTranslate; bool get dataSavingMode; bool get soundEffects; bool get aprilFoolFeatures; bool get enterToSend; bool get appBarTransparent; bool get showBackgroundImage; String? get customFonts; int? get appColorScheme;// The color stored via the int type | ||||||
|  Size? get windowSize; |  Size? get windowSize; | ||||||
| /// Create a copy of AppSettings | /// Create a copy of AppSettings | ||||||
| /// with the given fields replaced by the non-null parameter values. | /// with the given fields replaced by the non-null parameter values. | ||||||
| @@ -26,16 +26,16 @@ $AppSettingsCopyWith<AppSettings> get copyWith => _$AppSettingsCopyWithImpl<AppS | |||||||
|  |  | ||||||
| @override | @override | ||||||
| bool operator ==(Object other) { | bool operator ==(Object other) { | ||||||
|   return identical(this, other) || (other.runtimeType == runtimeType&&other is AppSettings&&(identical(other.autoTranslate, autoTranslate) || other.autoTranslate == autoTranslate)&&(identical(other.soundEffects, soundEffects) || other.soundEffects == soundEffects)&&(identical(other.aprilFoolFeatures, aprilFoolFeatures) || other.aprilFoolFeatures == aprilFoolFeatures)&&(identical(other.enterToSend, enterToSend) || other.enterToSend == enterToSend)&&(identical(other.appBarTransparent, appBarTransparent) || other.appBarTransparent == appBarTransparent)&&(identical(other.customFonts, customFonts) || other.customFonts == customFonts)&&(identical(other.appColorScheme, appColorScheme) || other.appColorScheme == appColorScheme)&&(identical(other.windowSize, windowSize) || other.windowSize == windowSize)); |   return identical(this, other) || (other.runtimeType == runtimeType&&other is AppSettings&&(identical(other.autoTranslate, autoTranslate) || other.autoTranslate == autoTranslate)&&(identical(other.dataSavingMode, dataSavingMode) || other.dataSavingMode == dataSavingMode)&&(identical(other.soundEffects, soundEffects) || other.soundEffects == soundEffects)&&(identical(other.aprilFoolFeatures, aprilFoolFeatures) || other.aprilFoolFeatures == aprilFoolFeatures)&&(identical(other.enterToSend, enterToSend) || other.enterToSend == enterToSend)&&(identical(other.appBarTransparent, appBarTransparent) || other.appBarTransparent == appBarTransparent)&&(identical(other.showBackgroundImage, showBackgroundImage) || other.showBackgroundImage == showBackgroundImage)&&(identical(other.customFonts, customFonts) || other.customFonts == customFonts)&&(identical(other.appColorScheme, appColorScheme) || other.appColorScheme == appColorScheme)&&(identical(other.windowSize, windowSize) || other.windowSize == windowSize)); | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| @override | @override | ||||||
| int get hashCode => Object.hash(runtimeType,autoTranslate,soundEffects,aprilFoolFeatures,enterToSend,appBarTransparent,customFonts,appColorScheme,windowSize); | int get hashCode => Object.hash(runtimeType,autoTranslate,dataSavingMode,soundEffects,aprilFoolFeatures,enterToSend,appBarTransparent,showBackgroundImage,customFonts,appColorScheme,windowSize); | ||||||
|  |  | ||||||
| @override | @override | ||||||
| String toString() { | String toString() { | ||||||
|   return 'AppSettings(autoTranslate: $autoTranslate, soundEffects: $soundEffects, aprilFoolFeatures: $aprilFoolFeatures, enterToSend: $enterToSend, appBarTransparent: $appBarTransparent, customFonts: $customFonts, appColorScheme: $appColorScheme, windowSize: $windowSize)'; |   return 'AppSettings(autoTranslate: $autoTranslate, dataSavingMode: $dataSavingMode, soundEffects: $soundEffects, aprilFoolFeatures: $aprilFoolFeatures, enterToSend: $enterToSend, appBarTransparent: $appBarTransparent, showBackgroundImage: $showBackgroundImage, customFonts: $customFonts, appColorScheme: $appColorScheme, windowSize: $windowSize)'; | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -46,7 +46,7 @@ abstract mixin class $AppSettingsCopyWith<$Res>  { | |||||||
|   factory $AppSettingsCopyWith(AppSettings value, $Res Function(AppSettings) _then) = _$AppSettingsCopyWithImpl; |   factory $AppSettingsCopyWith(AppSettings value, $Res Function(AppSettings) _then) = _$AppSettingsCopyWithImpl; | ||||||
| @useResult | @useResult | ||||||
| $Res call({ | $Res call({ | ||||||
|  bool autoTranslate, bool soundEffects, bool aprilFoolFeatures, bool enterToSend, bool appBarTransparent, String? customFonts, int? appColorScheme, Size? windowSize |  bool autoTranslate, bool dataSavingMode, bool soundEffects, bool aprilFoolFeatures, bool enterToSend, bool appBarTransparent, bool showBackgroundImage, String? customFonts, int? appColorScheme, Size? windowSize | ||||||
| }); | }); | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -63,13 +63,15 @@ class _$AppSettingsCopyWithImpl<$Res> | |||||||
|  |  | ||||||
| /// Create a copy of AppSettings | /// Create a copy of AppSettings | ||||||
| /// with the given fields replaced by the non-null parameter values. | /// with the given fields replaced by the non-null parameter values. | ||||||
| @pragma('vm:prefer-inline') @override $Res call({Object? autoTranslate = null,Object? soundEffects = null,Object? aprilFoolFeatures = null,Object? enterToSend = null,Object? appBarTransparent = null,Object? customFonts = freezed,Object? appColorScheme = freezed,Object? windowSize = freezed,}) { | @pragma('vm:prefer-inline') @override $Res call({Object? autoTranslate = null,Object? dataSavingMode = null,Object? soundEffects = null,Object? aprilFoolFeatures = null,Object? enterToSend = null,Object? appBarTransparent = null,Object? showBackgroundImage = null,Object? customFonts = freezed,Object? appColorScheme = freezed,Object? windowSize = freezed,}) { | ||||||
|   return _then(_self.copyWith( |   return _then(_self.copyWith( | ||||||
| autoTranslate: null == autoTranslate ? _self.autoTranslate : autoTranslate // ignore: cast_nullable_to_non_nullable | autoTranslate: null == autoTranslate ? _self.autoTranslate : autoTranslate // ignore: cast_nullable_to_non_nullable | ||||||
|  | as bool,dataSavingMode: null == dataSavingMode ? _self.dataSavingMode : dataSavingMode // ignore: cast_nullable_to_non_nullable | ||||||
| as bool,soundEffects: null == soundEffects ? _self.soundEffects : soundEffects // ignore: cast_nullable_to_non_nullable | as bool,soundEffects: null == soundEffects ? _self.soundEffects : soundEffects // ignore: cast_nullable_to_non_nullable | ||||||
| as bool,aprilFoolFeatures: null == aprilFoolFeatures ? _self.aprilFoolFeatures : aprilFoolFeatures // ignore: cast_nullable_to_non_nullable | as bool,aprilFoolFeatures: null == aprilFoolFeatures ? _self.aprilFoolFeatures : aprilFoolFeatures // ignore: cast_nullable_to_non_nullable | ||||||
| as bool,enterToSend: null == enterToSend ? _self.enterToSend : enterToSend // ignore: cast_nullable_to_non_nullable | as bool,enterToSend: null == enterToSend ? _self.enterToSend : enterToSend // ignore: cast_nullable_to_non_nullable | ||||||
| as bool,appBarTransparent: null == appBarTransparent ? _self.appBarTransparent : appBarTransparent // ignore: cast_nullable_to_non_nullable | as bool,appBarTransparent: null == appBarTransparent ? _self.appBarTransparent : appBarTransparent // ignore: cast_nullable_to_non_nullable | ||||||
|  | as bool,showBackgroundImage: null == showBackgroundImage ? _self.showBackgroundImage : showBackgroundImage // ignore: cast_nullable_to_non_nullable | ||||||
| as bool,customFonts: freezed == customFonts ? _self.customFonts : customFonts // ignore: cast_nullable_to_non_nullable | as bool,customFonts: freezed == customFonts ? _self.customFonts : customFonts // ignore: cast_nullable_to_non_nullable | ||||||
| as String?,appColorScheme: freezed == appColorScheme ? _self.appColorScheme : appColorScheme // ignore: cast_nullable_to_non_nullable | as String?,appColorScheme: freezed == appColorScheme ? _self.appColorScheme : appColorScheme // ignore: cast_nullable_to_non_nullable | ||||||
| as int?,windowSize: freezed == windowSize ? _self.windowSize : windowSize // ignore: cast_nullable_to_non_nullable | as int?,windowSize: freezed == windowSize ? _self.windowSize : windowSize // ignore: cast_nullable_to_non_nullable | ||||||
| @@ -155,10 +157,10 @@ return $default(_that);case _: | |||||||
| /// } | /// } | ||||||
| /// ``` | /// ``` | ||||||
|  |  | ||||||
| @optionalTypeArgs TResult maybeWhen<TResult extends Object?>(TResult Function( bool autoTranslate,  bool soundEffects,  bool aprilFoolFeatures,  bool enterToSend,  bool appBarTransparent,  String? customFonts,  int? appColorScheme,  Size? windowSize)?  $default,{required TResult orElse(),}) {final _that = this; | @optionalTypeArgs TResult maybeWhen<TResult extends Object?>(TResult Function( bool autoTranslate,  bool dataSavingMode,  bool soundEffects,  bool aprilFoolFeatures,  bool enterToSend,  bool appBarTransparent,  bool showBackgroundImage,  String? customFonts,  int? appColorScheme,  Size? windowSize)?  $default,{required TResult orElse(),}) {final _that = this; | ||||||
| switch (_that) { | switch (_that) { | ||||||
| case _AppSettings() when $default != null: | case _AppSettings() when $default != null: | ||||||
| return $default(_that.autoTranslate,_that.soundEffects,_that.aprilFoolFeatures,_that.enterToSend,_that.appBarTransparent,_that.customFonts,_that.appColorScheme,_that.windowSize);case _: | return $default(_that.autoTranslate,_that.dataSavingMode,_that.soundEffects,_that.aprilFoolFeatures,_that.enterToSend,_that.appBarTransparent,_that.showBackgroundImage,_that.customFonts,_that.appColorScheme,_that.windowSize);case _: | ||||||
|   return orElse(); |   return orElse(); | ||||||
|  |  | ||||||
| } | } | ||||||
| @@ -176,10 +178,10 @@ return $default(_that.autoTranslate,_that.soundEffects,_that.aprilFoolFeatures,_ | |||||||
| /// } | /// } | ||||||
| /// ``` | /// ``` | ||||||
|  |  | ||||||
| @optionalTypeArgs TResult when<TResult extends Object?>(TResult Function( bool autoTranslate,  bool soundEffects,  bool aprilFoolFeatures,  bool enterToSend,  bool appBarTransparent,  String? customFonts,  int? appColorScheme,  Size? windowSize)  $default,) {final _that = this; | @optionalTypeArgs TResult when<TResult extends Object?>(TResult Function( bool autoTranslate,  bool dataSavingMode,  bool soundEffects,  bool aprilFoolFeatures,  bool enterToSend,  bool appBarTransparent,  bool showBackgroundImage,  String? customFonts,  int? appColorScheme,  Size? windowSize)  $default,) {final _that = this; | ||||||
| switch (_that) { | switch (_that) { | ||||||
| case _AppSettings(): | case _AppSettings(): | ||||||
| return $default(_that.autoTranslate,_that.soundEffects,_that.aprilFoolFeatures,_that.enterToSend,_that.appBarTransparent,_that.customFonts,_that.appColorScheme,_that.windowSize);} | return $default(_that.autoTranslate,_that.dataSavingMode,_that.soundEffects,_that.aprilFoolFeatures,_that.enterToSend,_that.appBarTransparent,_that.showBackgroundImage,_that.customFonts,_that.appColorScheme,_that.windowSize);} | ||||||
| } | } | ||||||
| /// A variant of `when` that fallback to returning `null` | /// A variant of `when` that fallback to returning `null` | ||||||
| /// | /// | ||||||
| @@ -193,10 +195,10 @@ return $default(_that.autoTranslate,_that.soundEffects,_that.aprilFoolFeatures,_ | |||||||
| /// } | /// } | ||||||
| /// ``` | /// ``` | ||||||
|  |  | ||||||
| @optionalTypeArgs TResult? whenOrNull<TResult extends Object?>(TResult? Function( bool autoTranslate,  bool soundEffects,  bool aprilFoolFeatures,  bool enterToSend,  bool appBarTransparent,  String? customFonts,  int? appColorScheme,  Size? windowSize)?  $default,) {final _that = this; | @optionalTypeArgs TResult? whenOrNull<TResult extends Object?>(TResult? Function( bool autoTranslate,  bool dataSavingMode,  bool soundEffects,  bool aprilFoolFeatures,  bool enterToSend,  bool appBarTransparent,  bool showBackgroundImage,  String? customFonts,  int? appColorScheme,  Size? windowSize)?  $default,) {final _that = this; | ||||||
| switch (_that) { | switch (_that) { | ||||||
| case _AppSettings() when $default != null: | case _AppSettings() when $default != null: | ||||||
| return $default(_that.autoTranslate,_that.soundEffects,_that.aprilFoolFeatures,_that.enterToSend,_that.appBarTransparent,_that.customFonts,_that.appColorScheme,_that.windowSize);case _: | return $default(_that.autoTranslate,_that.dataSavingMode,_that.soundEffects,_that.aprilFoolFeatures,_that.enterToSend,_that.appBarTransparent,_that.showBackgroundImage,_that.customFonts,_that.appColorScheme,_that.windowSize);case _: | ||||||
|   return null; |   return null; | ||||||
|  |  | ||||||
| } | } | ||||||
| @@ -208,14 +210,16 @@ return $default(_that.autoTranslate,_that.soundEffects,_that.aprilFoolFeatures,_ | |||||||
|  |  | ||||||
|  |  | ||||||
| class _AppSettings implements AppSettings { | class _AppSettings implements AppSettings { | ||||||
|   const _AppSettings({required this.autoTranslate, required this.soundEffects, required this.aprilFoolFeatures, required this.enterToSend, required this.appBarTransparent, required this.customFonts, required this.appColorScheme, required this.windowSize}); |   const _AppSettings({required this.autoTranslate, required this.dataSavingMode, required this.soundEffects, required this.aprilFoolFeatures, required this.enterToSend, required this.appBarTransparent, required this.showBackgroundImage, required this.customFonts, required this.appColorScheme, required this.windowSize}); | ||||||
|    |    | ||||||
|  |  | ||||||
| @override final  bool autoTranslate; | @override final  bool autoTranslate; | ||||||
|  | @override final  bool dataSavingMode; | ||||||
| @override final  bool soundEffects; | @override final  bool soundEffects; | ||||||
| @override final  bool aprilFoolFeatures; | @override final  bool aprilFoolFeatures; | ||||||
| @override final  bool enterToSend; | @override final  bool enterToSend; | ||||||
| @override final  bool appBarTransparent; | @override final  bool appBarTransparent; | ||||||
|  | @override final  bool showBackgroundImage; | ||||||
| @override final  String? customFonts; | @override final  String? customFonts; | ||||||
| @override final  int? appColorScheme; | @override final  int? appColorScheme; | ||||||
| // The color stored via the int type | // The color stored via the int type | ||||||
| @@ -231,16 +235,16 @@ _$AppSettingsCopyWith<_AppSettings> get copyWith => __$AppSettingsCopyWithImpl<_ | |||||||
|  |  | ||||||
| @override | @override | ||||||
| bool operator ==(Object other) { | bool operator ==(Object other) { | ||||||
|   return identical(this, other) || (other.runtimeType == runtimeType&&other is _AppSettings&&(identical(other.autoTranslate, autoTranslate) || other.autoTranslate == autoTranslate)&&(identical(other.soundEffects, soundEffects) || other.soundEffects == soundEffects)&&(identical(other.aprilFoolFeatures, aprilFoolFeatures) || other.aprilFoolFeatures == aprilFoolFeatures)&&(identical(other.enterToSend, enterToSend) || other.enterToSend == enterToSend)&&(identical(other.appBarTransparent, appBarTransparent) || other.appBarTransparent == appBarTransparent)&&(identical(other.customFonts, customFonts) || other.customFonts == customFonts)&&(identical(other.appColorScheme, appColorScheme) || other.appColorScheme == appColorScheme)&&(identical(other.windowSize, windowSize) || other.windowSize == windowSize)); |   return identical(this, other) || (other.runtimeType == runtimeType&&other is _AppSettings&&(identical(other.autoTranslate, autoTranslate) || other.autoTranslate == autoTranslate)&&(identical(other.dataSavingMode, dataSavingMode) || other.dataSavingMode == dataSavingMode)&&(identical(other.soundEffects, soundEffects) || other.soundEffects == soundEffects)&&(identical(other.aprilFoolFeatures, aprilFoolFeatures) || other.aprilFoolFeatures == aprilFoolFeatures)&&(identical(other.enterToSend, enterToSend) || other.enterToSend == enterToSend)&&(identical(other.appBarTransparent, appBarTransparent) || other.appBarTransparent == appBarTransparent)&&(identical(other.showBackgroundImage, showBackgroundImage) || other.showBackgroundImage == showBackgroundImage)&&(identical(other.customFonts, customFonts) || other.customFonts == customFonts)&&(identical(other.appColorScheme, appColorScheme) || other.appColorScheme == appColorScheme)&&(identical(other.windowSize, windowSize) || other.windowSize == windowSize)); | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| @override | @override | ||||||
| int get hashCode => Object.hash(runtimeType,autoTranslate,soundEffects,aprilFoolFeatures,enterToSend,appBarTransparent,customFonts,appColorScheme,windowSize); | int get hashCode => Object.hash(runtimeType,autoTranslate,dataSavingMode,soundEffects,aprilFoolFeatures,enterToSend,appBarTransparent,showBackgroundImage,customFonts,appColorScheme,windowSize); | ||||||
|  |  | ||||||
| @override | @override | ||||||
| String toString() { | String toString() { | ||||||
|   return 'AppSettings(autoTranslate: $autoTranslate, soundEffects: $soundEffects, aprilFoolFeatures: $aprilFoolFeatures, enterToSend: $enterToSend, appBarTransparent: $appBarTransparent, customFonts: $customFonts, appColorScheme: $appColorScheme, windowSize: $windowSize)'; |   return 'AppSettings(autoTranslate: $autoTranslate, dataSavingMode: $dataSavingMode, soundEffects: $soundEffects, aprilFoolFeatures: $aprilFoolFeatures, enterToSend: $enterToSend, appBarTransparent: $appBarTransparent, showBackgroundImage: $showBackgroundImage, customFonts: $customFonts, appColorScheme: $appColorScheme, windowSize: $windowSize)'; | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -251,7 +255,7 @@ abstract mixin class _$AppSettingsCopyWith<$Res> implements $AppSettingsCopyWith | |||||||
|   factory _$AppSettingsCopyWith(_AppSettings value, $Res Function(_AppSettings) _then) = __$AppSettingsCopyWithImpl; |   factory _$AppSettingsCopyWith(_AppSettings value, $Res Function(_AppSettings) _then) = __$AppSettingsCopyWithImpl; | ||||||
| @override @useResult | @override @useResult | ||||||
| $Res call({ | $Res call({ | ||||||
|  bool autoTranslate, bool soundEffects, bool aprilFoolFeatures, bool enterToSend, bool appBarTransparent, String? customFonts, int? appColorScheme, Size? windowSize |  bool autoTranslate, bool dataSavingMode, bool soundEffects, bool aprilFoolFeatures, bool enterToSend, bool appBarTransparent, bool showBackgroundImage, String? customFonts, int? appColorScheme, Size? windowSize | ||||||
| }); | }); | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -268,13 +272,15 @@ class __$AppSettingsCopyWithImpl<$Res> | |||||||
|  |  | ||||||
| /// Create a copy of AppSettings | /// Create a copy of AppSettings | ||||||
| /// with the given fields replaced by the non-null parameter values. | /// with the given fields replaced by the non-null parameter values. | ||||||
| @override @pragma('vm:prefer-inline') $Res call({Object? autoTranslate = null,Object? soundEffects = null,Object? aprilFoolFeatures = null,Object? enterToSend = null,Object? appBarTransparent = null,Object? customFonts = freezed,Object? appColorScheme = freezed,Object? windowSize = freezed,}) { | @override @pragma('vm:prefer-inline') $Res call({Object? autoTranslate = null,Object? dataSavingMode = null,Object? soundEffects = null,Object? aprilFoolFeatures = null,Object? enterToSend = null,Object? appBarTransparent = null,Object? showBackgroundImage = null,Object? customFonts = freezed,Object? appColorScheme = freezed,Object? windowSize = freezed,}) { | ||||||
|   return _then(_AppSettings( |   return _then(_AppSettings( | ||||||
| autoTranslate: null == autoTranslate ? _self.autoTranslate : autoTranslate // ignore: cast_nullable_to_non_nullable | autoTranslate: null == autoTranslate ? _self.autoTranslate : autoTranslate // ignore: cast_nullable_to_non_nullable | ||||||
|  | as bool,dataSavingMode: null == dataSavingMode ? _self.dataSavingMode : dataSavingMode // ignore: cast_nullable_to_non_nullable | ||||||
| as bool,soundEffects: null == soundEffects ? _self.soundEffects : soundEffects // ignore: cast_nullable_to_non_nullable | as bool,soundEffects: null == soundEffects ? _self.soundEffects : soundEffects // ignore: cast_nullable_to_non_nullable | ||||||
| as bool,aprilFoolFeatures: null == aprilFoolFeatures ? _self.aprilFoolFeatures : aprilFoolFeatures // ignore: cast_nullable_to_non_nullable | as bool,aprilFoolFeatures: null == aprilFoolFeatures ? _self.aprilFoolFeatures : aprilFoolFeatures // ignore: cast_nullable_to_non_nullable | ||||||
| as bool,enterToSend: null == enterToSend ? _self.enterToSend : enterToSend // ignore: cast_nullable_to_non_nullable | as bool,enterToSend: null == enterToSend ? _self.enterToSend : enterToSend // ignore: cast_nullable_to_non_nullable | ||||||
| as bool,appBarTransparent: null == appBarTransparent ? _self.appBarTransparent : appBarTransparent // ignore: cast_nullable_to_non_nullable | as bool,appBarTransparent: null == appBarTransparent ? _self.appBarTransparent : appBarTransparent // ignore: cast_nullable_to_non_nullable | ||||||
|  | as bool,showBackgroundImage: null == showBackgroundImage ? _self.showBackgroundImage : showBackgroundImage // ignore: cast_nullable_to_non_nullable | ||||||
| as bool,customFonts: freezed == customFonts ? _self.customFonts : customFonts // ignore: cast_nullable_to_non_nullable | as bool,customFonts: freezed == customFonts ? _self.customFonts : customFonts // ignore: cast_nullable_to_non_nullable | ||||||
| as String?,appColorScheme: freezed == appColorScheme ? _self.appColorScheme : appColorScheme // ignore: cast_nullable_to_non_nullable | as String?,appColorScheme: freezed == appColorScheme ? _self.appColorScheme : appColorScheme // ignore: cast_nullable_to_non_nullable | ||||||
| as int?,windowSize: freezed == windowSize ? _self.windowSize : windowSize // ignore: cast_nullable_to_non_nullable | as int?,windowSize: freezed == windowSize ? _self.windowSize : windowSize // ignore: cast_nullable_to_non_nullable | ||||||
|   | |||||||
| @@ -7,7 +7,7 @@ part of 'config.dart'; | |||||||
| // ************************************************************************** | // ************************************************************************** | ||||||
|  |  | ||||||
| String _$appSettingsNotifierHash() => | String _$appSettingsNotifierHash() => | ||||||
|     r'c4f40a3bc4311c6360c2b5e44f8df5e5d7c1bd75'; |     r'cd18bff2614a94e3523634e6c577cefad0367eba'; | ||||||
|  |  | ||||||
| /// See also [AppSettingsNotifier]. | /// See also [AppSettingsNotifier]. | ||||||
| @ProviderFor(AppSettingsNotifier) | @ProviderFor(AppSettingsNotifier) | ||||||
|   | |||||||
| @@ -1,11 +1,9 @@ | |||||||
| import 'dart:convert'; | import 'dart:convert'; | ||||||
| import 'dart:developer'; |  | ||||||
|  |  | ||||||
| import 'package:freezed_annotation/freezed_annotation.dart'; | import 'package:freezed_annotation/freezed_annotation.dart'; | ||||||
| import 'package:hooks_riverpod/hooks_riverpod.dart'; | import 'package:hooks_riverpod/hooks_riverpod.dart'; | ||||||
| import 'package:island/pods/network.dart'; | import 'package:island/pods/network.dart'; | ||||||
| import 'package:riverpod_annotation/riverpod_annotation.dart'; | import 'package:riverpod_annotation/riverpod_annotation.dart'; | ||||||
| import 'package:flutter_langdetect/flutter_langdetect.dart' as langdetect; |  | ||||||
|  |  | ||||||
| part 'translate.freezed.dart'; | part 'translate.freezed.dart'; | ||||||
| part 'translate.g.dart'; | part 'translate.g.dart'; | ||||||
| @@ -29,10 +27,17 @@ Future<String> translateString(Ref ref, TranslateQuery query) async { | |||||||
|  |  | ||||||
| @riverpod | @riverpod | ||||||
| String? detectStringLanguage(Ref ref, String text) { | String? detectStringLanguage(Ref ref, String text) { | ||||||
|   try { |   bool isChinese(String text) { | ||||||
|     return langdetect.detectLangs(text).firstOrNull?.lang; |     final chineseRegex = RegExp(r'[\u4e00-\u9fff]'); | ||||||
|   } catch (err) { |     return chineseRegex.hasMatch(text); | ||||||
|     log('[Language] Unable to detect text\'s language: $text'); |  | ||||||
|     return null; |  | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   bool isEnglish(String text) { | ||||||
|  |     final englishRegex = RegExp(r'[a-zA-Z]'); | ||||||
|  |     return englishRegex.hasMatch(text) && !isChinese(text); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   if (isChinese(text)) return "zh"; | ||||||
|  |   if (isEnglish(text)) return "en"; | ||||||
|  |   return null; | ||||||
| } | } | ||||||
|   | |||||||
| @@ -149,7 +149,7 @@ class _TranslateStringProviderElement | |||||||
| } | } | ||||||
|  |  | ||||||
| String _$detectStringLanguageHash() => | String _$detectStringLanguageHash() => | ||||||
|     r'697b68464b3d00927cc43ccc1ba8ba93f2a470ed'; |     r'24fbf52edbbffcc8dc4f09f7206f82d69728e703'; | ||||||
|  |  | ||||||
| /// See also [detectStringLanguage]. | /// See also [detectStringLanguage]. | ||||||
| @ProviderFor(detectStringLanguage) | @ProviderFor(detectStringLanguage) | ||||||
|   | |||||||
| @@ -38,6 +38,7 @@ import 'package:island/screens/chat/chat.dart'; | |||||||
| import 'package:island/screens/chat/room.dart'; | import 'package:island/screens/chat/room.dart'; | ||||||
| import 'package:island/screens/chat/room_detail.dart'; | import 'package:island/screens/chat/room_detail.dart'; | ||||||
| import 'package:island/screens/chat/call.dart'; | import 'package:island/screens/chat/call.dart'; | ||||||
|  | import 'package:island/screens/chat/search_messages_screen.dart'; | ||||||
| import 'package:island/screens/creators/hub.dart'; | import 'package:island/screens/creators/hub.dart'; | ||||||
| import 'package:island/screens/creators/posts/post_manage_list.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/stickers.dart'; | ||||||
| @@ -555,6 +556,14 @@ final routerProvider = Provider<GoRouter>((ref) { | |||||||
|                       return ChatDetailScreen(id: id); |                       return ChatDetailScreen(id: id); | ||||||
|                     }, |                     }, | ||||||
|                   ), |                   ), | ||||||
|  |                   GoRoute( | ||||||
|  |                     name: 'searchMessages', | ||||||
|  |                     path: '/chat/:id/search', | ||||||
|  |                     builder: (context, state) { | ||||||
|  |                       final id = state.pathParameters['id']!; | ||||||
|  |                       return SearchMessagesScreen(roomId: id); | ||||||
|  |                     }, | ||||||
|  |                   ), | ||||||
|                 ], |                 ], | ||||||
|               ), |               ), | ||||||
|  |  | ||||||
|   | |||||||
| @@ -68,6 +68,7 @@ class AccountScreen extends HookConsumerWidget { | |||||||
|       body: SingleChildScrollView( |       body: SingleChildScrollView( | ||||||
|         padding: getTabbedPadding(context), |         padding: getTabbedPadding(context), | ||||||
|         child: Column( |         child: Column( | ||||||
|  |           spacing: 4, | ||||||
|           children: <Widget>[ |           children: <Widget>[ | ||||||
|             Card( |             Card( | ||||||
|               child: Column( |               child: Column( | ||||||
| @@ -112,20 +113,22 @@ class AccountScreen extends HookConsumerWidget { | |||||||
|                               crossAxisAlignment: CrossAxisAlignment.baseline, |                               crossAxisAlignment: CrossAxisAlignment.baseline, | ||||||
|                               textBaseline: TextBaseline.alphabetic, |                               textBaseline: TextBaseline.alphabetic, | ||||||
|                               children: [ |                               children: [ | ||||||
|                                 AccountName( |                                 Flexible( | ||||||
|                                   account: user.value!, |                                   child: AccountName( | ||||||
|                                   style: TextStyle( |                                     account: user.value!, | ||||||
|                                     fontSize: 16, |                                     style: TextStyle( | ||||||
|                                     fontWeight: FontWeight.bold, |                                       fontSize: 16, | ||||||
|  |                                       fontWeight: FontWeight.bold, | ||||||
|  |                                     ), | ||||||
|                                   ), |                                   ), | ||||||
|                                 ), |                                 ), | ||||||
|                                 Text('@${user.value!.name}'), |                                 Flexible(child: Text('@${user.value!.name}')), | ||||||
|                               ], |                               ], | ||||||
|                             ), |                             ), | ||||||
|                             Text( |                             Text( | ||||||
|                               (user.value!.profile.bio.isNotEmpty) |                               (user.value!.profile.bio.isNotEmpty) | ||||||
|                                   ? user.value!.profile.bio |                                   ? user.value!.profile.bio | ||||||
|                                   : 'No description yet.', |                                   : 'descriptionNone'.tr(), | ||||||
|                               maxLines: 1, |                               maxLines: 1, | ||||||
|                               overflow: TextOverflow.ellipsis, |                               overflow: TextOverflow.ellipsis, | ||||||
|                             ), |                             ), | ||||||
| @@ -158,8 +161,16 @@ class AccountScreen extends HookConsumerWidget { | |||||||
|                         crossAxisAlignment: CrossAxisAlignment.start, |                         crossAxisAlignment: CrossAxisAlignment.start, | ||||||
|                         children: [ |                         children: [ | ||||||
|                           Icon(Symbols.draw, size: 28).padding(bottom: 8), |                           Icon(Symbols.draw, size: 28).padding(bottom: 8), | ||||||
|                           Text('creatorHub').tr().fontSize(16).bold(), |                           Text( | ||||||
|                           Text('creatorHubDescription').tr(), |                             'creatorHub', | ||||||
|  |                             maxLines: 1, | ||||||
|  |                             overflow: TextOverflow.ellipsis, | ||||||
|  |                           ).tr().fontSize(16).bold(), | ||||||
|  |                           Text( | ||||||
|  |                             'creatorHubDescription', | ||||||
|  |                             maxLines: 2, | ||||||
|  |                             overflow: TextOverflow.ellipsis, | ||||||
|  |                           ).tr(), | ||||||
|                         ], |                         ], | ||||||
|                       ).padding(horizontal: 16, vertical: 12), |                       ).padding(horizontal: 16, vertical: 12), | ||||||
|                       onTap: () { |                       onTap: () { | ||||||
| @@ -176,8 +187,16 @@ class AccountScreen extends HookConsumerWidget { | |||||||
|                         crossAxisAlignment: CrossAxisAlignment.start, |                         crossAxisAlignment: CrossAxisAlignment.start, | ||||||
|                         children: [ |                         children: [ | ||||||
|                           Icon(Symbols.code, size: 28).padding(bottom: 8), |                           Icon(Symbols.code, size: 28).padding(bottom: 8), | ||||||
|                           Text('developerPortal').tr().fontSize(16).bold(), |                           Text( | ||||||
|                           Text('developerPortalDescription').tr(), |                             'developerPortal', | ||||||
|  |                             maxLines: 1, | ||||||
|  |                             overflow: TextOverflow.ellipsis, | ||||||
|  |                           ).tr().fontSize(16).bold(), | ||||||
|  |                           Text( | ||||||
|  |                             'developerPortalDescription', | ||||||
|  |                             maxLines: 2, | ||||||
|  |                             overflow: TextOverflow.ellipsis, | ||||||
|  |                           ).tr(), | ||||||
|                         ], |                         ], | ||||||
|                       ).padding(horizontal: 16, vertical: 12), |                       ).padding(horizontal: 16, vertical: 12), | ||||||
|                       onTap: () { |                       onTap: () { | ||||||
|   | |||||||
| @@ -95,8 +95,24 @@ class LevelingScreen extends HookConsumerWidget { | |||||||
|           title: Text('levelingProgress'.tr()), |           title: Text('levelingProgress'.tr()), | ||||||
|           bottom: TabBar( |           bottom: TabBar( | ||||||
|             tabs: [ |             tabs: [ | ||||||
|               Tab(text: 'leveling'.tr()), |               Tab( | ||||||
|               Tab(text: 'stellarProgram'.tr()), |                 child: Text( | ||||||
|  |                   'leveling'.tr(), | ||||||
|  |                   textAlign: TextAlign.center, | ||||||
|  |                   style: TextStyle( | ||||||
|  |                     color: Theme.of(context).appBarTheme.foregroundColor!, | ||||||
|  |                   ), | ||||||
|  |                 ), | ||||||
|  |               ), | ||||||
|  |               Tab( | ||||||
|  |                 child: Text( | ||||||
|  |                   'stellarProgram'.tr(), | ||||||
|  |                   textAlign: TextAlign.center, | ||||||
|  |                   style: TextStyle( | ||||||
|  |                     color: Theme.of(context).appBarTheme.foregroundColor!, | ||||||
|  |                   ), | ||||||
|  |                 ), | ||||||
|  |               ), | ||||||
|             ], |             ], | ||||||
|           ), |           ), | ||||||
|         ), |         ), | ||||||
|   | |||||||
| @@ -14,7 +14,6 @@ import 'package:island/screens/account/me/settings_connections.dart'; | |||||||
| import 'package:island/screens/account/me/settings_contacts.dart'; | import 'package:island/screens/account/me/settings_contacts.dart'; | ||||||
| import 'package:island/screens/auth/captcha.dart'; | import 'package:island/screens/auth/captcha.dart'; | ||||||
| import 'package:island/screens/auth/login.dart'; | import 'package:island/screens/auth/login.dart'; | ||||||
| import 'package:island/services/responsive.dart'; |  | ||||||
| import 'package:island/widgets/account/account_devices.dart'; | import 'package:island/widgets/account/account_devices.dart'; | ||||||
| import 'package:island/widgets/alert.dart'; | import 'package:island/widgets/alert.dart'; | ||||||
| import 'package:island/widgets/app_scaffold.dart'; | import 'package:island/widgets/app_scaffold.dart'; | ||||||
| @@ -57,7 +56,6 @@ class AccountSettingsScreen extends HookConsumerWidget { | |||||||
|   Widget build(BuildContext context, WidgetRef ref) { |   Widget build(BuildContext context, WidgetRef ref) { | ||||||
|     final isDesktop = |     final isDesktop = | ||||||
|         !kIsWeb && (Platform.isWindows || Platform.isMacOS || Platform.isLinux); |         !kIsWeb && (Platform.isWindows || Platform.isMacOS || Platform.isLinux); | ||||||
|     final isWide = isWideScreen(context); |  | ||||||
|  |  | ||||||
|     Future<void> requestAccountDeletion() async { |     Future<void> requestAccountDeletion() async { | ||||||
|       final confirm = await showConfirmAlert( |       final confirm = await showConfirmAlert( | ||||||
| @@ -440,51 +438,19 @@ class AccountSettingsScreen extends HookConsumerWidget { | |||||||
|  |  | ||||||
|     // Create a responsive layout based on screen width |     // Create a responsive layout based on screen width | ||||||
|     Widget buildSettingsList() { |     Widget buildSettingsList() { | ||||||
|       if (isWide) { |       return Column( | ||||||
|         // Two-column layout for wide screens |         crossAxisAlignment: CrossAxisAlignment.start, | ||||||
|         return Row( |         children: [ | ||||||
|           crossAxisAlignment: CrossAxisAlignment.start, |           _SettingsSection( | ||||||
|           children: [ |             title: 'accountSecurityTitle', | ||||||
|             Expanded( |             children: securitySettings, | ||||||
|               child: Column( |           ), | ||||||
|                 crossAxisAlignment: CrossAxisAlignment.start, |           _SettingsSection( | ||||||
|                 children: [ |             title: 'accountDangerZoneTitle', | ||||||
|                   _SettingsSection( |             children: dangerZoneSettings, | ||||||
|                     title: 'accountSecurityTitle', |           ), | ||||||
|                     children: securitySettings, |         ], | ||||||
|                   ), |       ); | ||||||
|                 ], |  | ||||||
|               ), |  | ||||||
|             ), |  | ||||||
|             Expanded( |  | ||||||
|               child: Column( |  | ||||||
|                 crossAxisAlignment: CrossAxisAlignment.start, |  | ||||||
|                 children: [ |  | ||||||
|                   _SettingsSection( |  | ||||||
|                     title: 'accountDangerZoneTitle', |  | ||||||
|                     children: dangerZoneSettings, |  | ||||||
|                   ), |  | ||||||
|                 ], |  | ||||||
|               ), |  | ||||||
|             ), |  | ||||||
|           ], |  | ||||||
|         ); |  | ||||||
|       } else { |  | ||||||
|         // Single column layout for narrow screens |  | ||||||
|         return Column( |  | ||||||
|           crossAxisAlignment: CrossAxisAlignment.start, |  | ||||||
|           children: [ |  | ||||||
|             _SettingsSection( |  | ||||||
|               title: 'accountSecurityTitle', |  | ||||||
|               children: securitySettings, |  | ||||||
|             ), |  | ||||||
|             _SettingsSection( |  | ||||||
|               title: 'accountDangerZoneTitle', |  | ||||||
|               children: dangerZoneSettings, |  | ||||||
|             ), |  | ||||||
|           ], |  | ||||||
|         ); |  | ||||||
|       } |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     return AppScaffold( |     return AppScaffold( | ||||||
|   | |||||||
| @@ -21,6 +21,7 @@ import 'package:material_symbols_icons/symbols.dart'; | |||||||
| import 'package:styled_widget/styled_widget.dart'; | import 'package:styled_widget/styled_widget.dart'; | ||||||
|  |  | ||||||
| const kServerSupportedLanguages = {'en-US': 'en-us', 'zh-CN': 'zh-hans'}; | const kServerSupportedLanguages = {'en-US': 'en-us', 'zh-CN': 'zh-hans'}; | ||||||
|  | const kServerSupportedRegions = ['US', 'JP', 'CN']; | ||||||
|  |  | ||||||
| class UpdateProfileScreen extends HookConsumerWidget { | class UpdateProfileScreen extends HookConsumerWidget { | ||||||
|   const UpdateProfileScreen({super.key}); |   const UpdateProfileScreen({super.key}); | ||||||
| @@ -97,6 +98,7 @@ class UpdateProfileScreen extends HookConsumerWidget { | |||||||
|     final usernameController = useTextEditingController(text: user.value!.name); |     final usernameController = useTextEditingController(text: user.value!.name); | ||||||
|     final nicknameController = useTextEditingController(text: user.value!.nick); |     final nicknameController = useTextEditingController(text: user.value!.nick); | ||||||
|     final language = useState(user.value!.language); |     final language = useState(user.value!.language); | ||||||
|  |     final region = useState(user.value!.region); | ||||||
|     final links = useState<List<ProfileLink>>(user.value!.profile.links); |     final links = useState<List<ProfileLink>>(user.value!.profile.links); | ||||||
|  |  | ||||||
|     void updateBasicInfo() async { |     void updateBasicInfo() async { | ||||||
| @@ -111,6 +113,7 @@ class UpdateProfileScreen extends HookConsumerWidget { | |||||||
|             'name': usernameController.text, |             'name': usernameController.text, | ||||||
|             'nick': nicknameController.text, |             'nick': nicknameController.text, | ||||||
|             'language': language.value, |             'language': language.value, | ||||||
|  |             'region': region.value, | ||||||
|           }, |           }, | ||||||
|         ); |         ); | ||||||
|         final userNotifier = ref.read(userInfoProvider.notifier); |         final userNotifier = ref.read(userInfoProvider.notifier); | ||||||
| @@ -291,6 +294,32 @@ class UpdateProfileScreen extends HookConsumerWidget { | |||||||
|                       ], |                       ], | ||||||
|                     ), |                     ), | ||||||
|                   ), |                   ), | ||||||
|  |                   DropdownButtonFormField2<String>( | ||||||
|  |                     decoration: InputDecoration( | ||||||
|  |                       labelText: 'region'.tr(), | ||||||
|  |                       helperText: 'accountRegionHint'.tr(), | ||||||
|  |                     ), | ||||||
|  |                     items: [ | ||||||
|  |                       ...kServerSupportedRegions.map( | ||||||
|  |                         (e) => DropdownMenuItem(value: e, child: Text(e)), | ||||||
|  |                       ), | ||||||
|  |                       if (!kServerSupportedRegions.contains(region.value)) | ||||||
|  |                         DropdownMenuItem( | ||||||
|  |                           value: region.value, | ||||||
|  |                           child: Text(region.value), | ||||||
|  |                         ), | ||||||
|  |                     ], | ||||||
|  |                     value: region.value, | ||||||
|  |                     onChanged: (value) { | ||||||
|  |                       region.value = value ?? region.value; | ||||||
|  |                     }, | ||||||
|  |                     customButton: Row( | ||||||
|  |                       children: [ | ||||||
|  |                         Expanded(child: Text(region.value)), | ||||||
|  |                         Icon(Symbols.arrow_drop_down), | ||||||
|  |                       ], | ||||||
|  |                     ), | ||||||
|  |                   ), | ||||||
|                   Align( |                   Align( | ||||||
|                     alignment: Alignment.centerRight, |                     alignment: Alignment.centerRight, | ||||||
|                     child: TextButton.icon( |                     child: TextButton.icon( | ||||||
|   | |||||||
| @@ -1,14 +1,16 @@ | |||||||
| import 'package:easy_localization/easy_localization.dart'; | import 'package:easy_localization/easy_localization.dart'; | ||||||
|  | import 'package:flutter/foundation.dart'; | ||||||
| import 'package:flutter/material.dart'; | import 'package:flutter/material.dart'; | ||||||
| import 'package:flutter_hooks/flutter_hooks.dart'; | import 'package:flutter_hooks/flutter_hooks.dart'; | ||||||
| import 'package:flutter_svg/flutter_svg.dart'; | import 'package:flutter_svg/flutter_svg.dart'; | ||||||
| import 'package:gap/gap.dart'; | import 'package:gap/gap.dart'; | ||||||
| import 'package:hooks_riverpod/hooks_riverpod.dart'; | import 'package:hooks_riverpod/hooks_riverpod.dart'; | ||||||
| import 'package:island/models/auth.dart'; | import 'package:island/models/auth.dart'; | ||||||
|  | import 'package:island/pods/config.dart'; | ||||||
| import 'package:island/pods/network.dart'; | import 'package:island/pods/network.dart'; | ||||||
| import 'package:island/screens/account/me/account_settings.dart'; | import 'package:island/screens/account/me/account_settings.dart'; | ||||||
| import 'package:island/screens/auth/oidc.native.dart'; | import 'package:island/screens/auth/oidc.native.dart'; | ||||||
| import 'package:island/services/text.dart'; | import 'package:island/utils/text.dart'; | ||||||
| import 'package:island/services/time.dart'; | import 'package:island/services/time.dart'; | ||||||
| import 'package:island/widgets/alert.dart'; | import 'package:island/widgets/alert.dart'; | ||||||
| import 'package:island/widgets/content/sheet.dart'; | import 'package:island/widgets/content/sheet.dart'; | ||||||
| @@ -16,6 +18,7 @@ import 'package:island/widgets/response.dart'; | |||||||
| import 'package:material_symbols_icons/symbols.dart'; | import 'package:material_symbols_icons/symbols.dart'; | ||||||
| import 'package:sign_in_with_apple/sign_in_with_apple.dart'; | import 'package:sign_in_with_apple/sign_in_with_apple.dart'; | ||||||
| import 'package:styled_widget/styled_widget.dart'; | import 'package:styled_widget/styled_widget.dart'; | ||||||
|  | import 'package:url_launcher/url_launcher_string.dart'; | ||||||
|  |  | ||||||
| // Helper function to get provider icon and localized name | // Helper function to get provider icon and localized name | ||||||
| Widget getProviderIcon(String provider, {double size = 24, Color? color}) { | Widget getProviderIcon(String provider, {double size = 24, Color? color}) { | ||||||
| @@ -165,9 +168,7 @@ class AccountConnectionNewSheet extends HookConsumerWidget { | |||||||
|               scopes: [AppleIDAuthorizationScopes.email], |               scopes: [AppleIDAuthorizationScopes.email], | ||||||
|               webAuthenticationOptions: WebAuthenticationOptions( |               webAuthenticationOptions: WebAuthenticationOptions( | ||||||
|                 clientId: 'dev.solsynth.solarpass', |                 clientId: 'dev.solsynth.solarpass', | ||||||
|                 redirectUri: Uri.parse( |                 redirectUri: Uri.parse('https://id.solian.app/auth/callback'), | ||||||
|                   'https://id.solian.app/auth/callback/apple', |  | ||||||
|                 ), |  | ||||||
|               ), |               ), | ||||||
|             ); |             ); | ||||||
|  |  | ||||||
| @@ -195,17 +196,25 @@ class AccountConnectionNewSheet extends HookConsumerWidget { | |||||||
|         case 'github': |         case 'github': | ||||||
|         case 'discord': |         case 'discord': | ||||||
|         case 'afdian': |         case 'afdian': | ||||||
|           await Navigator.of(context, rootNavigator: true).push( |           if (kIsWeb) { | ||||||
|             MaterialPageRoute( |             final serverUrl = ref.watch(serverUrlProvider); | ||||||
|               builder: |             final accessToken = ref.watch(tokenProvider); | ||||||
|                   (context) => OidcScreen( |             launchUrlString( | ||||||
|                     provider: selectedProvider.value.toLowerCase(), |               '$serverUrl/id/auth/login/${selectedProvider.value}?tk=${accessToken!.token}', | ||||||
|                     title: |             ); | ||||||
|                         'Connect with ${selectedProvider.value.capitalizeEachWord()}', |           } else { | ||||||
|                   ), |             await Navigator.of(context, rootNavigator: true).push( | ||||||
|             ), |               MaterialPageRoute( | ||||||
|           ); |                 builder: | ||||||
|           if (context.mounted) Navigator.pop(context, true); |                     (context) => OidcScreen( | ||||||
|  |                       provider: selectedProvider.value.toLowerCase(), | ||||||
|  |                       title: | ||||||
|  |                           'Connect with ${selectedProvider.value.capitalizeEachWord()}', | ||||||
|  |                     ), | ||||||
|  |               ), | ||||||
|  |             ); | ||||||
|  |             if (context.mounted) Navigator.pop(context, true); | ||||||
|  |           } | ||||||
|           break; |           break; | ||||||
|         default: |         default: | ||||||
|           showSnackBar('accountConnectionAddError'.tr()); |           showSnackBar('accountConnectionAddError'.tr()); | ||||||
|   | |||||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -762,5 +762,127 @@ class _AccountBotDeveloperProviderElement | |||||||
|   String get uname => (origin as AccountBotDeveloperProvider).uname; |   String get uname => (origin as AccountBotDeveloperProvider).uname; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | String _$accountPublishersHash() => r'25f5695b4a5154163d77f1769876d826bf736609'; | ||||||
|  |  | ||||||
|  | /// See also [accountPublishers]. | ||||||
|  | @ProviderFor(accountPublishers) | ||||||
|  | const accountPublishersProvider = AccountPublishersFamily(); | ||||||
|  |  | ||||||
|  | /// See also [accountPublishers]. | ||||||
|  | class AccountPublishersFamily extends Family<AsyncValue<List<SnPublisher>>> { | ||||||
|  |   /// See also [accountPublishers]. | ||||||
|  |   const AccountPublishersFamily(); | ||||||
|  |  | ||||||
|  |   /// See also [accountPublishers]. | ||||||
|  |   AccountPublishersProvider call(String id) { | ||||||
|  |     return AccountPublishersProvider(id); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   AccountPublishersProvider getProviderOverride( | ||||||
|  |     covariant AccountPublishersProvider provider, | ||||||
|  |   ) { | ||||||
|  |     return call(provider.id); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   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'accountPublishersProvider'; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /// See also [accountPublishers]. | ||||||
|  | class AccountPublishersProvider | ||||||
|  |     extends AutoDisposeFutureProvider<List<SnPublisher>> { | ||||||
|  |   /// See also [accountPublishers]. | ||||||
|  |   AccountPublishersProvider(String id) | ||||||
|  |     : this._internal( | ||||||
|  |         (ref) => accountPublishers(ref as AccountPublishersRef, id), | ||||||
|  |         from: accountPublishersProvider, | ||||||
|  |         name: r'accountPublishersProvider', | ||||||
|  |         debugGetCreateSourceHash: | ||||||
|  |             const bool.fromEnvironment('dart.vm.product') | ||||||
|  |                 ? null | ||||||
|  |                 : _$accountPublishersHash, | ||||||
|  |         dependencies: AccountPublishersFamily._dependencies, | ||||||
|  |         allTransitiveDependencies: | ||||||
|  |             AccountPublishersFamily._allTransitiveDependencies, | ||||||
|  |         id: id, | ||||||
|  |       ); | ||||||
|  |  | ||||||
|  |   AccountPublishersProvider._internal( | ||||||
|  |     super._createNotifier, { | ||||||
|  |     required super.name, | ||||||
|  |     required super.dependencies, | ||||||
|  |     required super.allTransitiveDependencies, | ||||||
|  |     required super.debugGetCreateSourceHash, | ||||||
|  |     required super.from, | ||||||
|  |     required this.id, | ||||||
|  |   }) : super.internal(); | ||||||
|  |  | ||||||
|  |   final String id; | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Override overrideWith( | ||||||
|  |     FutureOr<List<SnPublisher>> Function(AccountPublishersRef provider) create, | ||||||
|  |   ) { | ||||||
|  |     return ProviderOverride( | ||||||
|  |       origin: this, | ||||||
|  |       override: AccountPublishersProvider._internal( | ||||||
|  |         (ref) => create(ref as AccountPublishersRef), | ||||||
|  |         from: from, | ||||||
|  |         name: null, | ||||||
|  |         dependencies: null, | ||||||
|  |         allTransitiveDependencies: null, | ||||||
|  |         debugGetCreateSourceHash: null, | ||||||
|  |         id: id, | ||||||
|  |       ), | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   AutoDisposeFutureProviderElement<List<SnPublisher>> createElement() { | ||||||
|  |     return _AccountPublishersProviderElement(this); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   bool operator ==(Object other) { | ||||||
|  |     return other is AccountPublishersProvider && other.id == id; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   int get hashCode { | ||||||
|  |     var hash = _SystemHash.combine(0, runtimeType.hashCode); | ||||||
|  |     hash = _SystemHash.combine(hash, id.hashCode); | ||||||
|  |  | ||||||
|  |     return _SystemHash.finish(hash); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @Deprecated('Will be removed in 3.0. Use Ref instead') | ||||||
|  | // ignore: unused_element | ||||||
|  | mixin AccountPublishersRef on AutoDisposeFutureProviderRef<List<SnPublisher>> { | ||||||
|  |   /// The parameter `id` of this provider. | ||||||
|  |   String get id; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class _AccountPublishersProviderElement | ||||||
|  |     extends AutoDisposeFutureProviderElement<List<SnPublisher>> | ||||||
|  |     with AccountPublishersRef { | ||||||
|  |   _AccountPublishersProviderElement(super.provider); | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   String get id => (origin as AccountPublishersProvider).id; | ||||||
|  | } | ||||||
|  |  | ||||||
| // ignore_for_file: type=lint | // 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 | // 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 | ||||||
|   | |||||||
| @@ -4,6 +4,7 @@ import 'package:flutter/services.dart'; | |||||||
| import 'package:flutter_hooks/flutter_hooks.dart'; | import 'package:flutter_hooks/flutter_hooks.dart'; | ||||||
| import 'package:hooks_riverpod/hooks_riverpod.dart'; | import 'package:hooks_riverpod/hooks_riverpod.dart'; | ||||||
| import 'package:island/pods/userinfo.dart'; | import 'package:island/pods/userinfo.dart'; | ||||||
|  | import 'package:island/widgets/account/account_pfc.dart'; | ||||||
| import 'package:island/widgets/account/account_picker.dart'; | import 'package:island/widgets/account/account_picker.dart'; | ||||||
| import 'package:island/widgets/alert.dart'; | import 'package:island/widgets/alert.dart'; | ||||||
| import 'package:island/widgets/app_scaffold.dart'; | import 'package:island/widgets/app_scaffold.dart'; | ||||||
| @@ -99,7 +100,10 @@ class RelationshipListTile extends StatelessWidget { | |||||||
|  |  | ||||||
|     return ListTile( |     return ListTile( | ||||||
|       contentPadding: const EdgeInsets.only(left: 16, right: 12), |       contentPadding: const EdgeInsets.only(left: 16, right: 12), | ||||||
|       leading: ProfilePictureWidget(fileId: account.profile.picture?.id), |       leading: AccountPfcGestureDetector( | ||||||
|  |         uname: account.name, | ||||||
|  |         child: ProfilePictureWidget(fileId: account.profile.picture?.id), | ||||||
|  |       ), | ||||||
|       title: Row( |       title: Row( | ||||||
|         spacing: 6, |         spacing: 6, | ||||||
|         children: [ |         children: [ | ||||||
|   | |||||||
| @@ -700,45 +700,48 @@ class _LoginLookupScreen extends HookConsumerWidget { | |||||||
|           onTapOutside: (_) => FocusManager.instance.primaryFocus?.unfocus(), |           onTapOutside: (_) => FocusManager.instance.primaryFocus?.unfocus(), | ||||||
|           onSubmitted: isBusy.value ? null : (_) => performNewTicket(), |           onSubmitted: isBusy.value ? null : (_) => performNewTicket(), | ||||||
|         ).padding(horizontal: 7), |         ).padding(horizontal: 7), | ||||||
|         Row( |         if (!kIsWeb) | ||||||
|           spacing: 6, |           Row( | ||||||
|           crossAxisAlignment: CrossAxisAlignment.center, |             spacing: 6, | ||||||
|           children: <Widget>[ |             crossAxisAlignment: CrossAxisAlignment.center, | ||||||
|             Text("loginOr").tr().fontSize(11).opacity(0.85), |             children: <Widget>[ | ||||||
|             const Gap(8), |               Text("loginOr").tr().fontSize(11).opacity(0.85), | ||||||
|             Spacer(), |               const Gap(8), | ||||||
|             IconButton.filledTonal( |               Spacer(), | ||||||
|               onPressed: () => withOidc('github'), |               IconButton.filledTonal( | ||||||
|               padding: EdgeInsets.zero, |                 onPressed: () => withOidc('github'), | ||||||
|               icon: getProviderIcon( |                 padding: EdgeInsets.zero, | ||||||
|                 "github", |                 icon: getProviderIcon( | ||||||
|                 size: 16, |                   "github", | ||||||
|                 color: Theme.of(context).colorScheme.onPrimaryContainer, |                   size: 16, | ||||||
|  |                   color: Theme.of(context).colorScheme.onPrimaryContainer, | ||||||
|  |                 ), | ||||||
|  |                 tooltip: 'GitHub', | ||||||
|               ), |               ), | ||||||
|               tooltip: 'GitHub', |               IconButton.filledTonal( | ||||||
|             ), |                 onPressed: () => withOidc('google'), | ||||||
|             IconButton.filledTonal( |                 padding: EdgeInsets.zero, | ||||||
|               onPressed: () => withOidc('google'), |                 icon: getProviderIcon( | ||||||
|               padding: EdgeInsets.zero, |                   "google", | ||||||
|               icon: getProviderIcon( |                   size: 16, | ||||||
|                 "google", |                   color: Theme.of(context).colorScheme.onPrimaryContainer, | ||||||
|                 size: 16, |                 ), | ||||||
|                 color: Theme.of(context).colorScheme.onPrimaryContainer, |                 tooltip: 'Google', | ||||||
|               ), |               ), | ||||||
|               tooltip: 'Google', |               IconButton.filledTonal( | ||||||
|             ), |                 onPressed: withApple, | ||||||
|             IconButton.filledTonal( |                 padding: EdgeInsets.zero, | ||||||
|               onPressed: withApple, |                 icon: getProviderIcon( | ||||||
|               padding: EdgeInsets.zero, |                   "apple", | ||||||
|               icon: getProviderIcon( |                   size: 16, | ||||||
|                 "apple", |                   color: Theme.of(context).colorScheme.onPrimaryContainer, | ||||||
|                 size: 16, |                 ), | ||||||
|                 color: Theme.of(context).colorScheme.onPrimaryContainer, |                 tooltip: 'Apple Account', | ||||||
|               ), |               ), | ||||||
|               tooltip: 'Apple Account', |             ], | ||||||
|             ), |           ).padding(horizontal: 8, vertical: 8) | ||||||
|           ], |         else | ||||||
|         ).padding(horizontal: 8, vertical: 8), |           const Gap(12), | ||||||
|         Row( |         Row( | ||||||
|           mainAxisAlignment: MainAxisAlignment.spaceBetween, |           mainAxisAlignment: MainAxisAlignment.spaceBetween, | ||||||
|           children: [ |           children: [ | ||||||
|   | |||||||
| @@ -4,6 +4,7 @@ import "dart:developer" as developer; | |||||||
| import "dart:io"; | import "dart:io"; | ||||||
| import "package:dio/dio.dart"; | import "package:dio/dio.dart"; | ||||||
| import "package:easy_localization/easy_localization.dart"; | import "package:easy_localization/easy_localization.dart"; | ||||||
|  | import "package:file_picker/file_picker.dart"; | ||||||
| import "package:flutter/foundation.dart"; | import "package:flutter/foundation.dart"; | ||||||
| import "package:flutter/material.dart"; | import "package:flutter/material.dart"; | ||||||
| import "package:go_router/go_router.dart"; | import "package:go_router/go_router.dart"; | ||||||
| @@ -72,6 +73,207 @@ class _AppLifecycleObserver extends WidgetsBindingObserver { | |||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | class _PublicRoomPreview extends HookConsumerWidget { | ||||||
|  |   final String id; | ||||||
|  |   final SnChatRoom room; | ||||||
|  |  | ||||||
|  |   const _PublicRoomPreview({required this.id, required this.room}); | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Widget build(BuildContext context, WidgetRef ref) { | ||||||
|  |     final messages = ref.watch(messagesNotifierProvider(id)); | ||||||
|  |     final messagesNotifier = ref.read(messagesNotifierProvider(id).notifier); | ||||||
|  |     final scrollController = useScrollController(); | ||||||
|  |  | ||||||
|  |     final listController = useMemoized(() => ListController(), []); | ||||||
|  |  | ||||||
|  |     var isLoading = false; | ||||||
|  |  | ||||||
|  |     // Add scroll listener for pagination | ||||||
|  |     useEffect(() { | ||||||
|  |       void onScroll() { | ||||||
|  |         if (scrollController.position.pixels >= | ||||||
|  |             scrollController.position.maxScrollExtent - 200) { | ||||||
|  |           if (isLoading) return; | ||||||
|  |           isLoading = true; | ||||||
|  |           messagesNotifier.loadMore().then((_) => isLoading = false); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       scrollController.addListener(onScroll); | ||||||
|  |       return () => scrollController.removeListener(onScroll); | ||||||
|  |     }, [scrollController]); | ||||||
|  |  | ||||||
|  |     Widget chatMessageListWidget(List<LocalChatMessage> messageList) => | ||||||
|  |         SuperListView.builder( | ||||||
|  |           listController: listController, | ||||||
|  |           padding: EdgeInsets.symmetric(vertical: 16), | ||||||
|  |           controller: scrollController, | ||||||
|  |           reverse: true, // Show newest messages at the bottom | ||||||
|  |           itemCount: messageList.length, | ||||||
|  |           findChildIndexCallback: (key) { | ||||||
|  |             final valueKey = key as ValueKey; | ||||||
|  |             final messageId = valueKey.value as String; | ||||||
|  |             return messageList.indexWhere((m) => m.id == messageId); | ||||||
|  |           }, | ||||||
|  |           extentEstimation: (_, _) => 40, | ||||||
|  |           itemBuilder: (context, index) { | ||||||
|  |             final message = messageList[index]; | ||||||
|  |             final nextMessage = | ||||||
|  |                 index < messageList.length - 1 ? messageList[index + 1] : null; | ||||||
|  |             final isLastInGroup = | ||||||
|  |                 nextMessage == null || | ||||||
|  |                 nextMessage.senderId != message.senderId || | ||||||
|  |                 nextMessage.createdAt | ||||||
|  |                         .difference(message.createdAt) | ||||||
|  |                         .inMinutes | ||||||
|  |                         .abs() > | ||||||
|  |                     3; | ||||||
|  |  | ||||||
|  |             return MessageItem( | ||||||
|  |               message: message, | ||||||
|  |               isCurrentUser: false, // User is not a member, so not current user | ||||||
|  |               onAction: null, // No actions allowed in preview mode | ||||||
|  |               onJump: (_) {}, // No jump functionality in preview | ||||||
|  |               progress: null, | ||||||
|  |               showAvatar: isLastInGroup, | ||||||
|  |             ); | ||||||
|  |           }, | ||||||
|  |         ); | ||||||
|  |  | ||||||
|  |     final compactHeader = isWideScreen(context); | ||||||
|  |  | ||||||
|  |     Widget comfortHeaderWidget() => Column( | ||||||
|  |       spacing: 4, | ||||||
|  |       mainAxisAlignment: MainAxisAlignment.center, | ||||||
|  |       crossAxisAlignment: CrossAxisAlignment.center, | ||||||
|  |       children: [ | ||||||
|  |         SizedBox( | ||||||
|  |           height: 26, | ||||||
|  |           width: 26, | ||||||
|  |           child: | ||||||
|  |               (room.type == 1 && room.picture?.id == null) | ||||||
|  |                   ? SplitAvatarWidget( | ||||||
|  |                     filesId: | ||||||
|  |                         room.members! | ||||||
|  |                             .map((e) => e.account.profile.picture?.id) | ||||||
|  |                             .toList(), | ||||||
|  |                   ) | ||||||
|  |                   : room.picture?.id != null | ||||||
|  |                   ? ProfilePictureWidget( | ||||||
|  |                     fileId: room.picture?.id, | ||||||
|  |                     fallbackIcon: Symbols.chat, | ||||||
|  |                   ) | ||||||
|  |                   : CircleAvatar( | ||||||
|  |                     child: Text( | ||||||
|  |                       room.name![0].toUpperCase(), | ||||||
|  |                       style: const TextStyle(fontSize: 12), | ||||||
|  |                     ), | ||||||
|  |                   ), | ||||||
|  |         ), | ||||||
|  |         Text( | ||||||
|  |           (room.type == 1 && room.name == null) | ||||||
|  |               ? room.members!.map((e) => e.account.nick).join(', ') | ||||||
|  |               : room.name!, | ||||||
|  |         ).fontSize(15), | ||||||
|  |       ], | ||||||
|  |     ); | ||||||
|  |  | ||||||
|  |     Widget compactHeaderWidget() => Row( | ||||||
|  |       spacing: 8, | ||||||
|  |       crossAxisAlignment: CrossAxisAlignment.center, | ||||||
|  |       children: [ | ||||||
|  |         SizedBox( | ||||||
|  |           height: 26, | ||||||
|  |           width: 26, | ||||||
|  |           child: | ||||||
|  |               (room.type == 1 && room.picture?.id == null) | ||||||
|  |                   ? SplitAvatarWidget( | ||||||
|  |                     filesId: | ||||||
|  |                         room.members! | ||||||
|  |                             .map((e) => e.account.profile.picture?.id) | ||||||
|  |                             .toList(), | ||||||
|  |                   ) | ||||||
|  |                   : room.picture?.id != null | ||||||
|  |                   ? ProfilePictureWidget( | ||||||
|  |                     fileId: room.picture?.id, | ||||||
|  |                     fallbackIcon: Symbols.chat, | ||||||
|  |                   ) | ||||||
|  |                   : CircleAvatar( | ||||||
|  |                     child: Text( | ||||||
|  |                       room.name![0].toUpperCase(), | ||||||
|  |                       style: const TextStyle(fontSize: 12), | ||||||
|  |                     ), | ||||||
|  |                   ), | ||||||
|  |         ), | ||||||
|  |         Text( | ||||||
|  |           (room.type == 1 && room.name == null) | ||||||
|  |               ? room.members!.map((e) => e.account.nick).join(', ') | ||||||
|  |               : room.name!, | ||||||
|  |         ).fontSize(19), | ||||||
|  |       ], | ||||||
|  |     ); | ||||||
|  |  | ||||||
|  |     return AppScaffold( | ||||||
|  |       appBar: AppBar( | ||||||
|  |         leading: !compactHeader ? const Center(child: PageBackButton()) : null, | ||||||
|  |         automaticallyImplyLeading: false, | ||||||
|  |         toolbarHeight: compactHeader ? null : 64, | ||||||
|  |         title: compactHeader ? compactHeaderWidget() : comfortHeaderWidget(), | ||||||
|  |         actions: [ | ||||||
|  |           IconButton( | ||||||
|  |             icon: const Icon(Icons.more_vert), | ||||||
|  |             onPressed: () { | ||||||
|  |               context.pushNamed('chatDetail', pathParameters: {'id': id}); | ||||||
|  |             }, | ||||||
|  |           ), | ||||||
|  |           const Gap(8), | ||||||
|  |         ], | ||||||
|  |       ), | ||||||
|  |       body: Column( | ||||||
|  |         crossAxisAlignment: CrossAxisAlignment.stretch, | ||||||
|  |         children: [ | ||||||
|  |           Expanded( | ||||||
|  |             child: messages.when( | ||||||
|  |               data: | ||||||
|  |                   (messageList) => | ||||||
|  |                       messageList.isEmpty | ||||||
|  |                           ? Center(child: Text('No messages yet'.tr())) | ||||||
|  |                           : chatMessageListWidget(messageList), | ||||||
|  |               loading: () => const Center(child: CircularProgressIndicator()), | ||||||
|  |               error: | ||||||
|  |                   (error, _) => ResponseErrorWidget( | ||||||
|  |                     error: error, | ||||||
|  |                     onRetry: () => messagesNotifier.loadInitial(), | ||||||
|  |                   ), | ||||||
|  |             ), | ||||||
|  |           ), | ||||||
|  |           // Join button at the bottom for public rooms | ||||||
|  |           Container( | ||||||
|  |             padding: const EdgeInsets.all(16), | ||||||
|  |             child: FilledButton.tonalIcon( | ||||||
|  |               onPressed: () async { | ||||||
|  |                 try { | ||||||
|  |                   showLoadingModal(context); | ||||||
|  |                   final apiClient = ref.read(apiClientProvider); | ||||||
|  |                   await apiClient.post('/sphere/chat/${room.id}/members/me'); | ||||||
|  |                   ref.invalidate(chatroomIdentityProvider(id)); | ||||||
|  |                 } catch (err) { | ||||||
|  |                   showErrorAlert(err); | ||||||
|  |                 } finally { | ||||||
|  |                   if (context.mounted) hideLoadingModal(context); | ||||||
|  |                 } | ||||||
|  |               }, | ||||||
|  |               label: Text('chatJoin').tr(), | ||||||
|  |               icon: const Icon(Icons.add), | ||||||
|  |             ), | ||||||
|  |           ), | ||||||
|  |         ], | ||||||
|  |       ), | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
| @riverpod | @riverpod | ||||||
| class MessagesNotifier extends _$MessagesNotifier { | class MessagesNotifier extends _$MessagesNotifier { | ||||||
|   late final Dio _apiClient; |   late final Dio _apiClient; | ||||||
| @@ -82,6 +284,9 @@ class MessagesNotifier extends _$MessagesNotifier { | |||||||
|   final Map<String, LocalChatMessage> _pendingMessages = {}; |   final Map<String, LocalChatMessage> _pendingMessages = {}; | ||||||
|   final Map<String, Map<int, double>> _fileUploadProgress = {}; |   final Map<String, Map<int, double>> _fileUploadProgress = {}; | ||||||
|   int? _totalCount; |   int? _totalCount; | ||||||
|  |   String? _searchQuery; | ||||||
|  |   bool? _withLinks; | ||||||
|  |   bool? _withAttachments; | ||||||
|  |  | ||||||
|   late final String _roomId; |   late final String _roomId; | ||||||
|   int _currentPage = 0; |   int _currentPage = 0; | ||||||
| @@ -96,28 +301,42 @@ class MessagesNotifier extends _$MessagesNotifier { | |||||||
|     _database = ref.watch(databaseProvider); |     _database = ref.watch(databaseProvider); | ||||||
|     final room = await ref.watch(chatroomProvider(roomId).future); |     final room = await ref.watch(chatroomProvider(roomId).future); | ||||||
|     final identity = await ref.watch(chatroomIdentityProvider(roomId).future); |     final identity = await ref.watch(chatroomIdentityProvider(roomId).future); | ||||||
|     if (room == null || identity == null) { |  | ||||||
|       throw Exception('Room or identity not found'); |     if (room == null) { | ||||||
|  |       throw Exception('Room not found'); | ||||||
|     } |     } | ||||||
|     _room = room; |     _room = room; | ||||||
|     _identity = identity; |  | ||||||
|  |     // Allow building even if identity is null for public rooms | ||||||
|  |     if (identity != null) { | ||||||
|  |       _identity = identity; | ||||||
|  |     } | ||||||
|  |  | ||||||
|     developer.log( |     developer.log( | ||||||
|       'MessagesNotifier built for room $roomId', |       'MessagesNotifier built for room $roomId', | ||||||
|       name: 'MessagesNotifier', |       name: 'MessagesNotifier', | ||||||
|     ); |     ); | ||||||
|  |  | ||||||
|     ref.listen(appLifecycleStateProvider, (_, next) { |     // Only setup sync and lifecycle listeners if user is a member | ||||||
|       if (next.hasValue && next.value == AppLifecycleState.resumed) { |     if (identity != null) { | ||||||
|         developer.log( |       ref.listen(appLifecycleStateProvider, (_, next) { | ||||||
|           'App resumed, syncing messages', |         if (next.hasValue && next.value == AppLifecycleState.resumed) { | ||||||
|           name: 'MessagesNotifier', |           developer.log( | ||||||
|         ); |             'App resumed, syncing messages', | ||||||
|         syncMessages(); |             name: 'MessagesNotifier', | ||||||
|       } |           ); | ||||||
|     }); |           syncMessages(); | ||||||
|  |         } | ||||||
|  |       }); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     return await loadInitial(); |     loadInitial(); | ||||||
|  |     return []; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   List<LocalChatMessage> _sortMessages(List<LocalChatMessage> messages) { | ||||||
|  |     messages.sort((a, b) => b.createdAt.compareTo(a.createdAt)); | ||||||
|  |     return messages; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   Future<List<LocalChatMessage>> _getCachedMessages({ |   Future<List<LocalChatMessage>> _getCachedMessages({ | ||||||
| @@ -128,13 +347,32 @@ class MessagesNotifier extends _$MessagesNotifier { | |||||||
|       'Getting cached messages from offset $offset, take $take', |       'Getting cached messages from offset $offset, take $take', | ||||||
|       name: 'MessagesNotifier', |       name: 'MessagesNotifier', | ||||||
|     ); |     ); | ||||||
|     final dbMessages = await _database.getMessagesForRoom( |     final List<LocalChatMessage> dbMessages; | ||||||
|       _roomId, |     if (_searchQuery != null && _searchQuery!.isNotEmpty) { | ||||||
|       offset: offset, |       dbMessages = await _database.searchMessages(_roomId, _searchQuery ?? ''); | ||||||
|       limit: take, |     } else { | ||||||
|     ); |       final chatMessagesFromDb = await _database.getMessagesForRoom( | ||||||
|     final dbLocalMessages = |         _roomId, | ||||||
|         dbMessages.map(_database.companionToMessage).toList(); |         offset: offset, | ||||||
|  |         limit: take, | ||||||
|  |       ); | ||||||
|  |       dbMessages = | ||||||
|  |           chatMessagesFromDb.map(_database.companionToMessage).toList(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     List<LocalChatMessage> filteredMessages = dbMessages; | ||||||
|  |  | ||||||
|  |     if (_withLinks == true) { | ||||||
|  |       filteredMessages = | ||||||
|  |           filteredMessages.where((msg) => _hasLink(msg)).toList(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     if (_withAttachments == true) { | ||||||
|  |       filteredMessages = | ||||||
|  |           filteredMessages.where((msg) => _hasAttachment(msg)).toList(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     final dbLocalMessages = filteredMessages; | ||||||
|  |  | ||||||
|     if (offset == 0) { |     if (offset == 0) { | ||||||
|       final pendingForRoom = |       final pendingForRoom = | ||||||
| @@ -143,7 +381,7 @@ class MessagesNotifier extends _$MessagesNotifier { | |||||||
|               .toList(); |               .toList(); | ||||||
|  |  | ||||||
|       final allMessages = [...pendingForRoom, ...dbLocalMessages]; |       final allMessages = [...pendingForRoom, ...dbLocalMessages]; | ||||||
|       allMessages.sort((a, b) => b.createdAt.compareTo(a.createdAt)); |       _sortMessages(allMessages); // Use the helper function | ||||||
|  |  | ||||||
|       final uniqueMessages = <LocalChatMessage>[]; |       final uniqueMessages = <LocalChatMessage>[]; | ||||||
|       final seenIds = <String>{}; |       final seenIds = <String>{}; | ||||||
| @@ -218,7 +456,7 @@ class MessagesNotifier extends _$MessagesNotifier { | |||||||
|     _isSyncing = true; |     _isSyncing = true; | ||||||
|  |  | ||||||
|     developer.log('Starting message sync', name: 'MessagesNotifier'); |     developer.log('Starting message sync', name: 'MessagesNotifier'); | ||||||
|     ref.read(isSyncingProvider.notifier).state = true; |     Future.microtask(() => ref.read(isSyncingProvider.notifier).state = true); | ||||||
|     try { |     try { | ||||||
|       final dbMessages = await _database.getMessagesForRoom( |       final dbMessages = await _database.getMessagesForRoom( | ||||||
|         _room.id, |         _room.id, | ||||||
| @@ -279,7 +517,9 @@ class MessagesNotifier extends _$MessagesNotifier { | |||||||
|       showErrorAlert(err); |       showErrorAlert(err); | ||||||
|     } finally { |     } finally { | ||||||
|       developer.log('Finished message sync', name: 'MessagesNotifier'); |       developer.log('Finished message sync', name: 'MessagesNotifier'); | ||||||
|       ref.read(isSyncingProvider.notifier).state = false; |       Future.microtask( | ||||||
|  |         () => ref.read(isSyncingProvider.notifier).state = false, | ||||||
|  |       ); | ||||||
|       _isSyncing = false; |       _isSyncing = false; | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| @@ -290,7 +530,9 @@ class MessagesNotifier extends _$MessagesNotifier { | |||||||
|     bool synced = false, |     bool synced = false, | ||||||
|   }) async { |   }) async { | ||||||
|     try { |     try { | ||||||
|       if (offset == 0 && !synced) { |       if (offset == 0 && | ||||||
|  |           !synced && | ||||||
|  |           (_searchQuery == null || _searchQuery!.isEmpty)) { | ||||||
|         _fetchAndCacheMessages(offset: 0, take: take).catchError((_) { |         _fetchAndCacheMessages(offset: 0, take: take).catchError((_) { | ||||||
|           return <LocalChatMessage>[]; |           return <LocalChatMessage>[]; | ||||||
|         }); |         }); | ||||||
| @@ -305,7 +547,11 @@ class MessagesNotifier extends _$MessagesNotifier { | |||||||
|         return localMessages; |         return localMessages; | ||||||
|       } |       } | ||||||
|  |  | ||||||
|       return await _fetchAndCacheMessages(offset: offset, take: take); |       if (_searchQuery == null || _searchQuery!.isEmpty) { | ||||||
|  |         return await _fetchAndCacheMessages(offset: offset, take: take); | ||||||
|  |       } else { | ||||||
|  |         return []; // If searching, and no local messages, don't fetch from network | ||||||
|  |       } | ||||||
|     } catch (e) { |     } catch (e) { | ||||||
|       final localMessages = await _getCachedMessages( |       final localMessages = await _getCachedMessages( | ||||||
|         offset: offset, |         offset: offset, | ||||||
| @@ -319,13 +565,15 @@ class MessagesNotifier extends _$MessagesNotifier { | |||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   Future<List<LocalChatMessage>> loadInitial() async { |   Future<void> loadInitial() async { | ||||||
|     developer.log('Loading initial messages', name: 'MessagesNotifier'); |     developer.log('Loading initial messages', name: 'MessagesNotifier'); | ||||||
|     syncMessages(); |     if (_searchQuery == null || _searchQuery!.isEmpty) { | ||||||
|  |       syncMessages(); | ||||||
|  |     } | ||||||
|     final messages = await _getCachedMessages(offset: 0, take: 100); |     final messages = await _getCachedMessages(offset: 0, take: 100); | ||||||
|     _currentPage = 0; |     _currentPage = 0; | ||||||
|     _hasMore = messages.length == _pageSize; |     _hasMore = messages.length == _pageSize; | ||||||
|     return messages; |     state = AsyncValue.data(messages); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   Future<void> loadMore() async { |   Future<void> loadMore() async { | ||||||
| @@ -344,7 +592,9 @@ class MessagesNotifier extends _$MessagesNotifier { | |||||||
|         _hasMore = false; |         _hasMore = false; | ||||||
|       } |       } | ||||||
|  |  | ||||||
|       state = AsyncValue.data([...currentMessages, ...newMessages]); |       state = AsyncValue.data( | ||||||
|  |         _sortMessages([...currentMessages, ...newMessages]), | ||||||
|  |       ); | ||||||
|     } catch (err, stackTrace) { |     } catch (err, stackTrace) { | ||||||
|       developer.log( |       developer.log( | ||||||
|         'Error loading more messages', |         'Error loading more messages', | ||||||
| @@ -455,10 +705,13 @@ class MessagesNotifier extends _$MessagesNotifier { | |||||||
|  |  | ||||||
|       final currentMessages = state.value ?? []; |       final currentMessages = state.value ?? []; | ||||||
|       if (editingTo != null) { |       if (editingTo != null) { | ||||||
|         final newMessages = currentMessages |         final newMessages = | ||||||
|             .where((m) => m.id != localMessage.id) // remove pending message |             currentMessages | ||||||
|             .map((m) => m.id == editingTo.id ? updatedMessage : m) // update original message |                 .where((m) => m.id != localMessage.id) // remove pending message | ||||||
|             .toList(); |                 .map( | ||||||
|  |                   (m) => m.id == editingTo.id ? updatedMessage : m, | ||||||
|  |                 ) // update original message | ||||||
|  |                 .toList(); | ||||||
|         state = AsyncValue.data(newMessages); |         state = AsyncValue.data(newMessages); | ||||||
|       } else { |       } else { | ||||||
|         final newMessages = |         final newMessages = | ||||||
| @@ -566,7 +819,7 @@ class MessagesNotifier extends _$MessagesNotifier { | |||||||
|             } |             } | ||||||
|             return m; |             return m; | ||||||
|           }).toList(); |           }).toList(); | ||||||
|       state = AsyncValue.data(newMessages); |       state = AsyncValue.data(_sortMessages(newMessages)); | ||||||
|       showErrorAlert(e); |       showErrorAlert(e); | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| @@ -626,7 +879,7 @@ class MessagesNotifier extends _$MessagesNotifier { | |||||||
|     if (index >= 0) { |     if (index >= 0) { | ||||||
|       final newList = [...currentMessages]; |       final newList = [...currentMessages]; | ||||||
|       newList[index] = updatedMessage; |       newList[index] = updatedMessage; | ||||||
|       state = AsyncValue.data(newList); |       state = AsyncValue.data(_sortMessages(newList)); | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
| @@ -686,6 +939,20 @@ class MessagesNotifier extends _$MessagesNotifier { | |||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   void searchMessages(String query, {bool? withLinks, bool? withAttachments}) { | ||||||
|  |     _searchQuery = query.trim(); | ||||||
|  |     _withLinks = withLinks; | ||||||
|  |     _withAttachments = withAttachments; | ||||||
|  |     loadInitial(); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   void clearSearch() { | ||||||
|  |     _searchQuery = null; | ||||||
|  |     _withLinks = null; | ||||||
|  |     _withAttachments = null; | ||||||
|  |     loadInitial(); | ||||||
|  |   } | ||||||
|  |  | ||||||
|   Future<LocalChatMessage?> fetchMessageById(String messageId) async { |   Future<LocalChatMessage?> fetchMessageById(String messageId) async { | ||||||
|     developer.log( |     developer.log( | ||||||
|       'Fetching message by id $messageId', |       'Fetching message by id $messageId', | ||||||
| @@ -715,6 +982,18 @@ class MessagesNotifier extends _$MessagesNotifier { | |||||||
|       rethrow; |       rethrow; | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   bool _hasLink(LocalChatMessage message) { | ||||||
|  |     final content = message.toRemoteMessage().content; | ||||||
|  |     if (content == null) return false; | ||||||
|  |     final urlRegex = RegExp(r'https?://[^\s/$.?#].[^\s]*'); | ||||||
|  |     return urlRegex.hasMatch(content); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   bool _hasAttachment(LocalChatMessage message) { | ||||||
|  |     final remoteMessage = message.toRemoteMessage(); | ||||||
|  |     return remoteMessage.attachments.isNotEmpty; | ||||||
|  |   } | ||||||
| } | } | ||||||
|  |  | ||||||
| class ChatRoomScreen extends HookConsumerWidget { | class ChatRoomScreen extends HookConsumerWidget { | ||||||
| @@ -734,57 +1013,77 @@ class ChatRoomScreen extends HookConsumerWidget { | |||||||
|       ); |       ); | ||||||
|     } else if (chatIdentity.value == null) { |     } else if (chatIdentity.value == null) { | ||||||
|       // Identity was not found, user was not joined |       // Identity was not found, user was not joined | ||||||
|       return AppScaffold( |       return chatRoom.when( | ||||||
|         appBar: AppBar(leading: const PageBackButton()), |         data: (room) { | ||||||
|         body: Center( |           if (room!.isPublic) { | ||||||
|           child: |             // Show public room preview with messages but no input | ||||||
|               ConstrainedBox( |             return _PublicRoomPreview(id: id, room: room); | ||||||
|                 constraints: const BoxConstraints(maxWidth: 280), |           } else { | ||||||
|                 child: Column( |             // Show regular "not joined" screen for private rooms | ||||||
|                   crossAxisAlignment: CrossAxisAlignment.center, |             return AppScaffold( | ||||||
|                   mainAxisAlignment: MainAxisAlignment.center, |               appBar: AppBar(leading: const PageBackButton()), | ||||||
|                   children: [ |               body: Center( | ||||||
|                     Icon( |                 child: | ||||||
|                       chatRoom.value?.isCommunity == true |                     ConstrainedBox( | ||||||
|                           ? Symbols.person_add |                       constraints: const BoxConstraints(maxWidth: 280), | ||||||
|                           : Symbols.person_remove, |                       child: Column( | ||||||
|                       size: 36, |                         crossAxisAlignment: CrossAxisAlignment.center, | ||||||
|                       fill: 1, |                         mainAxisAlignment: MainAxisAlignment.center, | ||||||
|                     ).padding(bottom: 4), |                         children: [ | ||||||
|                     Text('chatNotJoined').tr(), |                           Icon( | ||||||
|                     if (chatRoom.value?.isCommunity != true) |                             room.isCommunity == true | ||||||
|                       Text( |                                 ? Symbols.person_add | ||||||
|                         'chatUnableJoin', |                                 : Symbols.person_remove, | ||||||
|                         textAlign: TextAlign.center, |                             size: 36, | ||||||
|                       ).tr().bold() |                             fill: 1, | ||||||
|                     else |                           ).padding(bottom: 4), | ||||||
|                       FilledButton.tonalIcon( |                           Text('chatNotJoined').tr(), | ||||||
|                         onPressed: () async { |                           if (room.isCommunity != true) | ||||||
|                           try { |                             Text( | ||||||
|                             showLoadingModal(context); |                               'chatUnableJoin', | ||||||
|                             final apiClient = ref.read(apiClientProvider); |                               textAlign: TextAlign.center, | ||||||
|                             if (chatRoom.value == null) { |                             ).tr().bold() | ||||||
|                               hideLoadingModal(context); |                           else | ||||||
|                               return; |                             FilledButton.tonalIcon( | ||||||
|                             } |                               onPressed: () async { | ||||||
|  |                                 try { | ||||||
|                             await apiClient.post( |                                   showLoadingModal(context); | ||||||
|                               '/sphere/chat/${chatRoom.value!.id}/members/me', |                                   final apiClient = ref.read(apiClientProvider); | ||||||
|                             ); |                                   await apiClient.post( | ||||||
|                             ref.invalidate(chatroomIdentityProvider(id)); |                                     '/sphere/chat/${room.id}/members/me', | ||||||
|                           } catch (err) { |                                   ); | ||||||
|                             showErrorAlert(err); |                                   ref.invalidate(chatroomIdentityProvider(id)); | ||||||
|                           } finally { |                                 } catch (err) { | ||||||
|                             if (context.mounted) hideLoadingModal(context); |                                   showErrorAlert(err); | ||||||
|                           } |                                 } finally { | ||||||
|                         }, |                                   if (context.mounted) { | ||||||
|                         label: Text('chatJoin').tr(), |                                     hideLoadingModal(context); | ||||||
|                         icon: const Icon(Icons.add), |                                   } | ||||||
|                       ).padding(top: 8), |                                 } | ||||||
|                   ], |                               }, | ||||||
|                 ), |                               label: Text('chatJoin').tr(), | ||||||
|               ).center(), |                               icon: const Icon(Icons.add), | ||||||
|         ), |                             ).padding(top: 8), | ||||||
|  |                         ], | ||||||
|  |                       ), | ||||||
|  |                     ).center(), | ||||||
|  |               ), | ||||||
|  |             ); | ||||||
|  |           } | ||||||
|  |         }, | ||||||
|  |         loading: | ||||||
|  |             () => AppScaffold( | ||||||
|  |               appBar: AppBar(leading: const PageBackButton()), | ||||||
|  |               body: CircularProgressIndicator().center(), | ||||||
|  |             ), | ||||||
|  |         error: | ||||||
|  |             (error, _) => AppScaffold( | ||||||
|  |               appBar: AppBar(leading: const PageBackButton()), | ||||||
|  |               body: ResponseErrorWidget( | ||||||
|  |                 error: error, | ||||||
|  |                 onRetry: () => ref.refresh(chatroomProvider(id)), | ||||||
|  |               ), | ||||||
|  |             ), | ||||||
|       ); |       ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -953,26 +1252,32 @@ class ChatRoomScreen extends HookConsumerWidget { | |||||||
|     }, [id]); |     }, [id]); | ||||||
|  |  | ||||||
|     Future<void> pickPhotoMedia() async { |     Future<void> pickPhotoMedia() async { | ||||||
|       final result = await ref |       final result = await FilePicker.platform.pickFiles( | ||||||
|           .watch(imagePickerProvider) |         type: FileType.image, | ||||||
|           .pickMultiImage(requestFullMetadata: true); |         allowMultiple: true, | ||||||
|       if (result.isEmpty) return; |         allowCompression: false, | ||||||
|  |       ); | ||||||
|  |       if (result == null || result.count == 0) return; | ||||||
|       attachments.value = [ |       attachments.value = [ | ||||||
|         ...attachments.value, |         ...attachments.value, | ||||||
|         ...result.map( |         ...result.files.map( | ||||||
|           (e) => UniversalFile(data: e, type: UniversalFileType.image), |           (e) => UniversalFile(data: e.xFile, type: UniversalFileType.image), | ||||||
|         ), |         ), | ||||||
|       ]; |       ]; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     Future<void> pickVideoMedia() async { |     Future<void> pickVideoMedia() async { | ||||||
|       final result = await ref |       final result = await FilePicker.platform.pickFiles( | ||||||
|           .watch(imagePickerProvider) |         type: FileType.video, | ||||||
|           .pickVideo(source: ImageSource.gallery); |         allowMultiple: true, | ||||||
|       if (result == null) return; |         allowCompression: false, | ||||||
|  |       ); | ||||||
|  |       if (result == null || result.count == 0) return; | ||||||
|       attachments.value = [ |       attachments.value = [ | ||||||
|         ...attachments.value, |         ...attachments.value, | ||||||
|         UniversalFile(data: result, type: UniversalFileType.video), |         ...result.files.map( | ||||||
|  |           (e) => UniversalFile(data: e.xFile, type: UniversalFileType.video), | ||||||
|  |         ), | ||||||
|       ]; |       ]; | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -1089,6 +1394,8 @@ class ChatRoomScreen extends HookConsumerWidget { | |||||||
|       ], |       ], | ||||||
|     ); |     ); | ||||||
|  |  | ||||||
|  |     const messageKeyPrefix = 'message-'; | ||||||
|  |  | ||||||
|     Widget chatMessageListWidget(List<LocalChatMessage> messageList) => |     Widget chatMessageListWidget(List<LocalChatMessage> messageList) => | ||||||
|         SuperListView.builder( |         SuperListView.builder( | ||||||
|           listController: listController, |           listController: listController, | ||||||
| @@ -1098,7 +1405,9 @@ class ChatRoomScreen extends HookConsumerWidget { | |||||||
|           itemCount: messageList.length, |           itemCount: messageList.length, | ||||||
|           findChildIndexCallback: (key) { |           findChildIndexCallback: (key) { | ||||||
|             final valueKey = key as ValueKey; |             final valueKey = key as ValueKey; | ||||||
|             final messageId = valueKey.value as String; |             final messageId = (valueKey.value as String).substring( | ||||||
|  |               messageKeyPrefix.length, | ||||||
|  |             ); | ||||||
|             return messageList.indexWhere((m) => m.id == messageId); |             return messageList.indexWhere((m) => m.id == messageId); | ||||||
|           }, |           }, | ||||||
|           extentEstimation: (_, _) => 40, |           extentEstimation: (_, _) => 40, | ||||||
| @@ -1115,10 +1424,13 @@ class ChatRoomScreen extends HookConsumerWidget { | |||||||
|                         .abs() > |                         .abs() > | ||||||
|                     3; |                     3; | ||||||
|  |  | ||||||
|  |             final key = ValueKey('$messageKeyPrefix${message.id}'); | ||||||
|  |  | ||||||
|             return chatIdentity.when( |             return chatIdentity.when( | ||||||
|               skipError: true, |               skipError: true, | ||||||
|               data: |               data: | ||||||
|                   (identity) => MessageItem( |                   (identity) => MessageItem( | ||||||
|  |                     key: key, | ||||||
|                     message: message, |                     message: message, | ||||||
|                     isCurrentUser: identity?.id == message.senderId, |                     isCurrentUser: identity?.id == message.senderId, | ||||||
|                     onAction: (action) { |                     onAction: (action) { | ||||||
| @@ -1161,6 +1473,7 @@ class ChatRoomScreen extends HookConsumerWidget { | |||||||
|                   ), |                   ), | ||||||
|               loading: |               loading: | ||||||
|                   () => MessageItem( |                   () => MessageItem( | ||||||
|  |                     key: key, | ||||||
|                     message: message, |                     message: message, | ||||||
|                     isCurrentUser: false, |                     isCurrentUser: false, | ||||||
|                     onAction: null, |                     onAction: null, | ||||||
| @@ -1168,7 +1481,7 @@ class ChatRoomScreen extends HookConsumerWidget { | |||||||
|                     showAvatar: false, |                     showAvatar: false, | ||||||
|                     onJump: (_) {}, |                     onJump: (_) {}, | ||||||
|                   ), |                   ), | ||||||
|               error: (_, _) => const SizedBox.shrink(), |               error: (_, _) => SizedBox.shrink(key: key), | ||||||
|             ); |             ); | ||||||
|           }, |           }, | ||||||
|         ); |         ); | ||||||
| @@ -1549,7 +1862,7 @@ class _ChatInput extends HookConsumerWidget { | |||||||
|                   children: [ |                   children: [ | ||||||
|                     IconButton( |                     IconButton( | ||||||
|                       tooltip: 'stickers'.tr(), |                       tooltip: 'stickers'.tr(), | ||||||
|                       icon: const Icon(Symbols.emoji_symbols), |                       icon: const Icon(Symbols.add_reaction), | ||||||
|                       onPressed: () { |                       onPressed: () { | ||||||
|                         final size = MediaQuery.of(context).size; |                         final size = MediaQuery.of(context).size; | ||||||
|                         showStickerPickerPopover( |                         showStickerPickerPopover( | ||||||
| @@ -1659,8 +1972,13 @@ class _ChatInput extends HookConsumerWidget { | |||||||
|                           horizontal: 12, |                           horizontal: 12, | ||||||
|                           vertical: 4, |                           vertical: 4, | ||||||
|                         ), |                         ), | ||||||
|  |                         counterText: | ||||||
|  |                             messageController.text.length > 1024 | ||||||
|  |                                 ? '${messageController.text.length}/4096' | ||||||
|  |                                 : null, | ||||||
|                       ), |                       ), | ||||||
|                       maxLines: null, |                       maxLines: 3, | ||||||
|  |                       minLines: 1, | ||||||
|                       onTapOutside: |                       onTapOutside: | ||||||
|                           (_) => FocusManager.instance.primaryFocus?.unfocus(), |                           (_) => FocusManager.instance.primaryFocus?.unfocus(), | ||||||
|                     ), |                     ), | ||||||
|   | |||||||
| @@ -6,7 +6,7 @@ part of 'room.dart'; | |||||||
| // RiverpodGenerator | // RiverpodGenerator | ||||||
| // ************************************************************************** | // ************************************************************************** | ||||||
|  |  | ||||||
| String _$messagesNotifierHash() => r'32afe6ea24086d869cc47bd3389c8fd734409ca0'; | String _$messagesNotifierHash() => r'fc3b66dfb8dd3fc55d142dae5c5e7bdc67eca5d4'; | ||||||
|  |  | ||||||
| /// Copied from Dart SDK | /// Copied from Dart SDK | ||||||
| class _SystemHash { | class _SystemHash { | ||||||
|   | |||||||
| @@ -9,6 +9,7 @@ import 'package:hooks_riverpod/hooks_riverpod.dart'; | |||||||
| import 'package:island/models/chat.dart'; | import 'package:island/models/chat.dart'; | ||||||
| import 'package:island/pods/network.dart'; | import 'package:island/pods/network.dart'; | ||||||
| import 'package:island/screens/chat/chat.dart'; | import 'package:island/screens/chat/chat.dart'; | ||||||
|  | import 'package:island/widgets/account/account_pfc.dart'; | ||||||
| import 'package:island/widgets/account/account_picker.dart'; | import 'package:island/widgets/account/account_picker.dart'; | ||||||
| import 'package:island/widgets/account/status.dart'; | import 'package:island/widgets/account/status.dart'; | ||||||
| import 'package:island/widgets/alert.dart'; | import 'package:island/widgets/alert.dart'; | ||||||
| @@ -19,10 +20,17 @@ import 'package:material_symbols_icons/symbols.dart'; | |||||||
| import 'package:riverpod_annotation/riverpod_annotation.dart'; | import 'package:riverpod_annotation/riverpod_annotation.dart'; | ||||||
| import 'package:riverpod_paging_utils/riverpod_paging_utils.dart'; | import 'package:riverpod_paging_utils/riverpod_paging_utils.dart'; | ||||||
| import 'package:styled_widget/styled_widget.dart'; | import 'package:styled_widget/styled_widget.dart'; | ||||||
|  | import 'package:island/pods/database.dart'; | ||||||
|  |  | ||||||
| part 'room_detail.freezed.dart'; | part 'room_detail.freezed.dart'; | ||||||
| part 'room_detail.g.dart'; | part 'room_detail.g.dart'; | ||||||
|  |  | ||||||
|  | @riverpod | ||||||
|  | Future<int> totalMessagesCount(Ref ref, String roomId) async { | ||||||
|  |   final database = ref.watch(databaseProvider); | ||||||
|  |   return database.getTotalMessagesForRoom(roomId); | ||||||
|  | } | ||||||
|  |  | ||||||
| class ChatDetailScreen extends HookConsumerWidget { | class ChatDetailScreen extends HookConsumerWidget { | ||||||
|   final String id; |   final String id; | ||||||
|   const ChatDetailScreen({super.key, required this.id}); |   const ChatDetailScreen({super.key, required this.id}); | ||||||
| @@ -31,6 +39,7 @@ class ChatDetailScreen extends HookConsumerWidget { | |||||||
|   Widget build(BuildContext context, WidgetRef ref) { |   Widget build(BuildContext context, WidgetRef ref) { | ||||||
|     final roomState = ref.watch(chatroomProvider(id)); |     final roomState = ref.watch(chatroomProvider(id)); | ||||||
|     final roomIdentity = ref.watch(chatroomIdentityProvider(id)); |     final roomIdentity = ref.watch(chatroomIdentityProvider(id)); | ||||||
|  |     final totalMessages = ref.watch(totalMessagesCountProvider(id)); | ||||||
|  |  | ||||||
|     const kNotifyLevelText = [ |     const kNotifyLevelText = [ | ||||||
|       'chatNotifyLevelAll', |       'chatNotifyLevelAll', | ||||||
| @@ -131,7 +140,7 @@ class ChatDetailScreen extends HookConsumerWidget { | |||||||
|                   const Text('chatBreakDescription').tr(), |                   const Text('chatBreakDescription').tr(), | ||||||
|                   const Gap(16), |                   const Gap(16), | ||||||
|                   ListTile( |                   ListTile( | ||||||
|                     title: const Text('Clear').tr(), |                     title: const Text('chatBreakClearButton').tr(), | ||||||
|                     subtitle: const Text('chatBreakClear').tr(), |                     subtitle: const Text('chatBreakClear').tr(), | ||||||
|                     leading: const Icon(Icons.notifications_active), |                     leading: const Icon(Icons.notifications_active), | ||||||
|                     onTap: () { |                     onTap: () { | ||||||
| @@ -143,8 +152,10 @@ class ChatDetailScreen extends HookConsumerWidget { | |||||||
|                     }, |                     }, | ||||||
|                   ), |                   ), | ||||||
|                   ListTile( |                   ListTile( | ||||||
|                     title: const Text('5m'), |                     title: const Text('chatBreak5m').tr(), | ||||||
|                     subtitle: const Text('chatBreakHour').tr(args: ['5m']), |                     subtitle: const Text( | ||||||
|  |                       'chatBreakHour', | ||||||
|  |                     ).tr(args: ['chatBreak5m'.tr()]), | ||||||
|                     leading: const Icon(Symbols.circle), |                     leading: const Icon(Symbols.circle), | ||||||
|                     onTap: () { |                     onTap: () { | ||||||
|                       setChatBreak(now.add(const Duration(minutes: 5))); |                       setChatBreak(now.add(const Duration(minutes: 5))); | ||||||
| @@ -155,8 +166,10 @@ class ChatDetailScreen extends HookConsumerWidget { | |||||||
|                     }, |                     }, | ||||||
|                   ), |                   ), | ||||||
|                   ListTile( |                   ListTile( | ||||||
|                     title: const Text('10m'), |                     title: const Text('chatBreak10m').tr(), | ||||||
|                     subtitle: const Text('chatBreakHour').tr(args: ['10m']), |                     subtitle: const Text( | ||||||
|  |                       'chatBreakHour', | ||||||
|  |                     ).tr(args: ['chatBreak10m'.tr()]), | ||||||
|                     leading: const Icon(Symbols.circle), |                     leading: const Icon(Symbols.circle), | ||||||
|                     onTap: () { |                     onTap: () { | ||||||
|                       setChatBreak(now.add(const Duration(minutes: 10))); |                       setChatBreak(now.add(const Duration(minutes: 10))); | ||||||
| @@ -167,8 +180,10 @@ class ChatDetailScreen extends HookConsumerWidget { | |||||||
|                     }, |                     }, | ||||||
|                   ), |                   ), | ||||||
|                   ListTile( |                   ListTile( | ||||||
|                     title: const Text('15m'), |                     title: const Text('chatBreak15m').tr(), | ||||||
|                     subtitle: const Text('chatBreakHour').tr(args: ['15m']), |                     subtitle: const Text( | ||||||
|  |                       'chatBreakHour', | ||||||
|  |                     ).tr(args: ['chatBreak15m'.tr()]), | ||||||
|                     leading: const Icon(Symbols.timer_3), |                     leading: const Icon(Symbols.timer_3), | ||||||
|                     onTap: () { |                     onTap: () { | ||||||
|                       setChatBreak(now.add(const Duration(minutes: 15))); |                       setChatBreak(now.add(const Duration(minutes: 15))); | ||||||
| @@ -179,8 +194,10 @@ class ChatDetailScreen extends HookConsumerWidget { | |||||||
|                     }, |                     }, | ||||||
|                   ), |                   ), | ||||||
|                   ListTile( |                   ListTile( | ||||||
|                     title: const Text('30m'), |                     title: const Text('chatBreak30m').tr(), | ||||||
|                     subtitle: const Text('chatBreakHour').tr(args: ['30m']), |                     subtitle: const Text( | ||||||
|  |                       'chatBreakHour', | ||||||
|  |                     ).tr(args: ['chatBreak30m'.tr()]), | ||||||
|                     leading: const Icon(Symbols.timer), |                     leading: const Icon(Symbols.timer), | ||||||
|                     onTap: () { |                     onTap: () { | ||||||
|                       setChatBreak(now.add(const Duration(minutes: 30))); |                       setChatBreak(now.add(const Duration(minutes: 30))); | ||||||
| @@ -194,8 +211,8 @@ class ChatDetailScreen extends HookConsumerWidget { | |||||||
|                   TextField( |                   TextField( | ||||||
|                     controller: durationController, |                     controller: durationController, | ||||||
|                     decoration: InputDecoration( |                     decoration: InputDecoration( | ||||||
|                       labelText: 'Custom (minutes)'.tr(), |                       labelText: 'chatBreakCustomMinutes'.tr(), | ||||||
|                       hintText: 'Enter minutes'.tr(), |                       hintText: 'chatBreakEnterMinutes'.tr(), | ||||||
|                       border: const OutlineInputBorder(), |                       border: const OutlineInputBorder(), | ||||||
|                       suffixIcon: IconButton( |                       suffixIcon: IconButton( | ||||||
|                         icon: const Icon(Icons.check), |                         icon: const Icon(Icons.check), | ||||||
| @@ -238,7 +255,10 @@ class ChatDetailScreen extends HookConsumerWidget { | |||||||
|     return AppScaffold( |     return AppScaffold( | ||||||
|       body: roomState.when( |       body: roomState.when( | ||||||
|         loading: () => const Center(child: CircularProgressIndicator()), |         loading: () => const Center(child: CircularProgressIndicator()), | ||||||
|         error: (error, _) => Center(child: Text('Error: $error')), |         error: | ||||||
|  |             (error, _) => Center( | ||||||
|  |               child: Text('errorGeneric'.tr(args: [error.toString()])), | ||||||
|  |             ), | ||||||
|         data: |         data: | ||||||
|             (currentRoom) => CustomScrollView( |             (currentRoom) => CustomScrollView( | ||||||
|               slivers: [ |               slivers: [ | ||||||
| @@ -358,6 +378,36 @@ class ChatDetailScreen extends HookConsumerWidget { | |||||||
|                                           : const Text('chatBreakNone').tr(), |                                           : const Text('chatBreakNone').tr(), | ||||||
|                                   onTap: () => showChatBreakDialog(), |                                   onTap: () => showChatBreakDialog(), | ||||||
|                                 ), |                                 ), | ||||||
|  |                                 ListTile( | ||||||
|  |                                   contentPadding: EdgeInsets.symmetric( | ||||||
|  |                                     horizontal: 24, | ||||||
|  |                                   ), | ||||||
|  |                                   leading: const Icon(Icons.search), | ||||||
|  |                                   trailing: const Icon(Symbols.chevron_right), | ||||||
|  |                                   title: const Text('searchMessages').tr(), | ||||||
|  |                                   subtitle: totalMessages.when( | ||||||
|  |                                     data: | ||||||
|  |                                         (count) => Text( | ||||||
|  |                                           'messagesCount'.tr( | ||||||
|  |                                             args: [count.toString()], | ||||||
|  |                                           ), | ||||||
|  |                                         ), | ||||||
|  |                                     loading: | ||||||
|  |                                         () => const CircularProgressIndicator(), | ||||||
|  |                                     error: | ||||||
|  |                                         (err, stack) => Text( | ||||||
|  |                                           'errorGeneric'.tr( | ||||||
|  |                                             args: [err.toString()], | ||||||
|  |                                           ), | ||||||
|  |                                         ), | ||||||
|  |                                   ), | ||||||
|  |                                   onTap: () { | ||||||
|  |                                     context.pushNamed( | ||||||
|  |                                       'searchMessages', | ||||||
|  |                                       pathParameters: {'id': id}, | ||||||
|  |                                     ); | ||||||
|  |                                   }, | ||||||
|  |                                 ), | ||||||
|                               ], |                               ], | ||||||
|                             ), |                             ), | ||||||
|                         error: (_, _) => const SizedBox.shrink(), |                         error: (_, _) => const SizedBox.shrink(), | ||||||
| @@ -666,8 +716,11 @@ class _ChatMemberListSheet extends HookConsumerWidget { | |||||||
|                     final member = data.items[index]; |                     final member = data.items[index]; | ||||||
|                     return ListTile( |                     return ListTile( | ||||||
|                       contentPadding: EdgeInsets.only(left: 16, right: 12), |                       contentPadding: EdgeInsets.only(left: 16, right: 12), | ||||||
|                       leading: ProfilePictureWidget( |                       leading: AccountPfcGestureDetector( | ||||||
|                         fileId: member.account.profile.picture?.id, |                         uname: member.account.name, | ||||||
|  |                         child: ProfilePictureWidget( | ||||||
|  |                           fileId: member.account.profile.picture?.id, | ||||||
|  |                         ), | ||||||
|                       ), |                       ), | ||||||
|                       title: Row( |                       title: Row( | ||||||
|                         spacing: 6, |                         spacing: 6, | ||||||
| @@ -848,7 +901,7 @@ class _ChatMemberRoleSheet extends HookConsumerWidget { | |||||||
|                     try { |                     try { | ||||||
|                       final newRole = int.parse(roleController.text); |                       final newRole = int.parse(roleController.text); | ||||||
|                       if (newRole < 0 || newRole > 100) { |                       if (newRole < 0 || newRole > 100) { | ||||||
|                         throw 'Role must be between 0 and 100'; |                         throw 'roleValidationHint'.tr(); | ||||||
|                       } |                       } | ||||||
|  |  | ||||||
|                       final apiClient = ref.read(apiClientProvider); |                       final apiClient = ref.read(apiClientProvider); | ||||||
|   | |||||||
| @@ -6,8 +6,8 @@ part of 'room_detail.dart'; | |||||||
| // RiverpodGenerator | // RiverpodGenerator | ||||||
| // ************************************************************************** | // ************************************************************************** | ||||||
|  |  | ||||||
| String _$chatMemberListNotifierHash() => | String _$totalMessagesCountHash() => | ||||||
|     r'3ea30150278523e9f6b23f9200ea9a9fbae9c973'; |     r'd55f1507aba2acdce5e468c1c2e15dba7640c571'; | ||||||
|  |  | ||||||
| /// Copied from Dart SDK | /// Copied from Dart SDK | ||||||
| class _SystemHash { | class _SystemHash { | ||||||
| @@ -30,6 +30,128 @@ class _SystemHash { | |||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | /// See also [totalMessagesCount]. | ||||||
|  | @ProviderFor(totalMessagesCount) | ||||||
|  | const totalMessagesCountProvider = TotalMessagesCountFamily(); | ||||||
|  |  | ||||||
|  | /// See also [totalMessagesCount]. | ||||||
|  | class TotalMessagesCountFamily extends Family<AsyncValue<int>> { | ||||||
|  |   /// See also [totalMessagesCount]. | ||||||
|  |   const TotalMessagesCountFamily(); | ||||||
|  |  | ||||||
|  |   /// See also [totalMessagesCount]. | ||||||
|  |   TotalMessagesCountProvider call(String roomId) { | ||||||
|  |     return TotalMessagesCountProvider(roomId); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   TotalMessagesCountProvider getProviderOverride( | ||||||
|  |     covariant TotalMessagesCountProvider provider, | ||||||
|  |   ) { | ||||||
|  |     return call(provider.roomId); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   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'totalMessagesCountProvider'; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /// See also [totalMessagesCount]. | ||||||
|  | class TotalMessagesCountProvider extends AutoDisposeFutureProvider<int> { | ||||||
|  |   /// See also [totalMessagesCount]. | ||||||
|  |   TotalMessagesCountProvider(String roomId) | ||||||
|  |     : this._internal( | ||||||
|  |         (ref) => totalMessagesCount(ref as TotalMessagesCountRef, roomId), | ||||||
|  |         from: totalMessagesCountProvider, | ||||||
|  |         name: r'totalMessagesCountProvider', | ||||||
|  |         debugGetCreateSourceHash: | ||||||
|  |             const bool.fromEnvironment('dart.vm.product') | ||||||
|  |                 ? null | ||||||
|  |                 : _$totalMessagesCountHash, | ||||||
|  |         dependencies: TotalMessagesCountFamily._dependencies, | ||||||
|  |         allTransitiveDependencies: | ||||||
|  |             TotalMessagesCountFamily._allTransitiveDependencies, | ||||||
|  |         roomId: roomId, | ||||||
|  |       ); | ||||||
|  |  | ||||||
|  |   TotalMessagesCountProvider._internal( | ||||||
|  |     super._createNotifier, { | ||||||
|  |     required super.name, | ||||||
|  |     required super.dependencies, | ||||||
|  |     required super.allTransitiveDependencies, | ||||||
|  |     required super.debugGetCreateSourceHash, | ||||||
|  |     required super.from, | ||||||
|  |     required this.roomId, | ||||||
|  |   }) : super.internal(); | ||||||
|  |  | ||||||
|  |   final String roomId; | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Override overrideWith( | ||||||
|  |     FutureOr<int> Function(TotalMessagesCountRef provider) create, | ||||||
|  |   ) { | ||||||
|  |     return ProviderOverride( | ||||||
|  |       origin: this, | ||||||
|  |       override: TotalMessagesCountProvider._internal( | ||||||
|  |         (ref) => create(ref as TotalMessagesCountRef), | ||||||
|  |         from: from, | ||||||
|  |         name: null, | ||||||
|  |         dependencies: null, | ||||||
|  |         allTransitiveDependencies: null, | ||||||
|  |         debugGetCreateSourceHash: null, | ||||||
|  |         roomId: roomId, | ||||||
|  |       ), | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   AutoDisposeFutureProviderElement<int> createElement() { | ||||||
|  |     return _TotalMessagesCountProviderElement(this); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   bool operator ==(Object other) { | ||||||
|  |     return other is TotalMessagesCountProvider && other.roomId == roomId; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   int get hashCode { | ||||||
|  |     var hash = _SystemHash.combine(0, runtimeType.hashCode); | ||||||
|  |     hash = _SystemHash.combine(hash, roomId.hashCode); | ||||||
|  |  | ||||||
|  |     return _SystemHash.finish(hash); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @Deprecated('Will be removed in 3.0. Use Ref instead') | ||||||
|  | // ignore: unused_element | ||||||
|  | mixin TotalMessagesCountRef on AutoDisposeFutureProviderRef<int> { | ||||||
|  |   /// The parameter `roomId` of this provider. | ||||||
|  |   String get roomId; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class _TotalMessagesCountProviderElement | ||||||
|  |     extends AutoDisposeFutureProviderElement<int> | ||||||
|  |     with TotalMessagesCountRef { | ||||||
|  |   _TotalMessagesCountProviderElement(super.provider); | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   String get roomId => (origin as TotalMessagesCountProvider).roomId; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | String _$chatMemberListNotifierHash() => | ||||||
|  |     r'3ea30150278523e9f6b23f9200ea9a9fbae9c973'; | ||||||
|  |  | ||||||
| abstract class _$ChatMemberListNotifier | abstract class _$ChatMemberListNotifier | ||||||
|     extends BuildlessAutoDisposeAsyncNotifier<CursorPagingData<SnChatMember>> { |     extends BuildlessAutoDisposeAsyncNotifier<CursorPagingData<SnChatMember>> { | ||||||
|   late final String roomId; |   late final String roomId; | ||||||
|   | |||||||
							
								
								
									
										139
									
								
								lib/screens/chat/search_messages_screen.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										139
									
								
								lib/screens/chat/search_messages_screen.dart
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,139 @@ | |||||||
|  | import 'package:easy_localization/easy_localization.dart'; | ||||||
|  | import 'package:flutter/material.dart'; | ||||||
|  | import 'package:flutter_hooks/flutter_hooks.dart'; | ||||||
|  | import 'package:hooks_riverpod/hooks_riverpod.dart'; | ||||||
|  | import 'package:island/screens/chat/room.dart'; | ||||||
|  | import 'package:island/widgets/app_scaffold.dart'; | ||||||
|  | import 'package:island/widgets/chat/message_item.dart'; | ||||||
|  | import 'package:material_symbols_icons/material_symbols_icons.dart'; | ||||||
|  | import 'package:super_sliver_list/super_sliver_list.dart'; | ||||||
|  |  | ||||||
|  | class SearchMessagesScreen extends HookConsumerWidget { | ||||||
|  |   final String roomId; | ||||||
|  |  | ||||||
|  |   const SearchMessagesScreen({super.key, required this.roomId}); | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Widget build(BuildContext context, WidgetRef ref) { | ||||||
|  |     final searchController = useTextEditingController(); | ||||||
|  |     final withLinks = useState(false); | ||||||
|  |     final withAttachments = useState(false); | ||||||
|  |  | ||||||
|  |     final messagesNotifier = ref.read( | ||||||
|  |       messagesNotifierProvider(roomId).notifier, | ||||||
|  |     ); | ||||||
|  |     final messages = ref.watch(messagesNotifierProvider(roomId)); | ||||||
|  |  | ||||||
|  |     useEffect(() { | ||||||
|  |       // Clear search when screen is disposed | ||||||
|  |       return () { | ||||||
|  |         messagesNotifier.clearSearch(); | ||||||
|  |       }; | ||||||
|  |     }, []); | ||||||
|  |  | ||||||
|  |     return AppScaffold( | ||||||
|  |       appBar: AppBar(title: const Text('searchMessages').tr()), | ||||||
|  |       body: Column( | ||||||
|  |         children: [ | ||||||
|  |           Column( | ||||||
|  |             children: [ | ||||||
|  |               TextField( | ||||||
|  |                 controller: searchController, | ||||||
|  |                 decoration: InputDecoration( | ||||||
|  |                   hintText: 'searchMessagesHint'.tr(), | ||||||
|  |                   border: InputBorder.none, | ||||||
|  |                   isDense: true, | ||||||
|  |                   contentPadding: EdgeInsets.only( | ||||||
|  |                     left: 16, | ||||||
|  |                     right: 16, | ||||||
|  |                     top: 12, | ||||||
|  |                     bottom: 16, | ||||||
|  |                   ), | ||||||
|  |                   suffix: IconButton( | ||||||
|  |                     iconSize: 18, | ||||||
|  |                     visualDensity: VisualDensity.compact, | ||||||
|  |                     icon: const Icon(Icons.clear), | ||||||
|  |                     onPressed: () { | ||||||
|  |                       searchController.clear(); | ||||||
|  |                       messagesNotifier.clearSearch(); | ||||||
|  |                     }, | ||||||
|  |                   ), | ||||||
|  |                 ), | ||||||
|  |                 onChanged: (query) { | ||||||
|  |                   messagesNotifier.searchMessages( | ||||||
|  |                     query, | ||||||
|  |                     withLinks: withLinks.value, | ||||||
|  |                     withAttachments: withAttachments.value, | ||||||
|  |                   ); | ||||||
|  |                 }, | ||||||
|  |               ), | ||||||
|  |               Row( | ||||||
|  |                 children: [ | ||||||
|  |                   Expanded( | ||||||
|  |                     child: CheckboxListTile( | ||||||
|  |                       secondary: const Icon(Symbols.link), | ||||||
|  |                       title: const Text('searchLinks').tr(), | ||||||
|  |                       value: withLinks.value, | ||||||
|  |                       onChanged: (bool? value) { | ||||||
|  |                         withLinks.value = value!; | ||||||
|  |                         messagesNotifier.searchMessages( | ||||||
|  |                           searchController.text, | ||||||
|  |                           withLinks: withLinks.value, | ||||||
|  |                           withAttachments: withAttachments.value, | ||||||
|  |                         ); | ||||||
|  |                       }, | ||||||
|  |                     ), | ||||||
|  |                   ), | ||||||
|  |                   Expanded( | ||||||
|  |                     child: CheckboxListTile( | ||||||
|  |                       secondary: const Icon(Symbols.file_copy), | ||||||
|  |                       title: const Text('searchAttachments').tr(), | ||||||
|  |                       value: withAttachments.value, | ||||||
|  |                       onChanged: (bool? value) { | ||||||
|  |                         withAttachments.value = value!; | ||||||
|  |                         messagesNotifier.searchMessages( | ||||||
|  |                           searchController.text, | ||||||
|  |                           withLinks: withLinks.value, | ||||||
|  |                           withAttachments: withAttachments.value, | ||||||
|  |                         ); | ||||||
|  |                       }, | ||||||
|  |                     ), | ||||||
|  |                   ), | ||||||
|  |                 ], | ||||||
|  |               ), | ||||||
|  |             ], | ||||||
|  |           ), | ||||||
|  |           const Divider(height: 1), | ||||||
|  |           Expanded( | ||||||
|  |             child: messages.when( | ||||||
|  |               data: | ||||||
|  |                   (messageList) => | ||||||
|  |                       messageList.isEmpty | ||||||
|  |                           ? Center(child: Text('noMessagesFound'.tr())) | ||||||
|  |                           : SuperListView.builder( | ||||||
|  |                             padding: const EdgeInsets.symmetric(vertical: 16), | ||||||
|  |                             reverse: true, // Show newest messages at the bottom | ||||||
|  |                             itemCount: messageList.length, | ||||||
|  |                             itemBuilder: (context, index) { | ||||||
|  |                               final message = messageList[index]; | ||||||
|  |                               // Simplified MessageItem for search results, no grouping logic | ||||||
|  |                               return MessageItem( | ||||||
|  |                                 message: message, | ||||||
|  |                                 isCurrentUser: | ||||||
|  |                                     false, // Or determine based on actual user | ||||||
|  |                                 onAction: null, | ||||||
|  |                                 onJump: (_) {}, | ||||||
|  |                                 progress: null, | ||||||
|  |                                 showAvatar: true, | ||||||
|  |                               ); | ||||||
|  |                             }, | ||||||
|  |                           ), | ||||||
|  |               loading: () => const Center(child: CircularProgressIndicator()), | ||||||
|  |               error: (error, _) => Center(child: Text('errorGeneric'.tr(args: [error.toString()]))), | ||||||
|  |             ), | ||||||
|  |           ), | ||||||
|  |         ], | ||||||
|  |       ), | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  | } | ||||||
| @@ -11,7 +11,7 @@ import 'package:island/models/publisher.dart'; | |||||||
| import 'package:island/pods/network.dart'; | import 'package:island/pods/network.dart'; | ||||||
| import 'package:island/screens/creators/publishers.dart'; | import 'package:island/screens/creators/publishers.dart'; | ||||||
| import 'package:island/services/responsive.dart'; | import 'package:island/services/responsive.dart'; | ||||||
| import 'package:island/services/text.dart'; | import 'package:island/utils/text.dart'; | ||||||
| import 'package:island/widgets/account/account_picker.dart'; | import 'package:island/widgets/account/account_picker.dart'; | ||||||
| import 'package:island/widgets/alert.dart'; | import 'package:island/widgets/alert.dart'; | ||||||
| import 'package:island/widgets/app_scaffold.dart'; | import 'package:island/widgets/app_scaffold.dart'; | ||||||
|   | |||||||
| @@ -9,6 +9,7 @@ import 'package:island/widgets/poll/poll_feedback.dart'; | |||||||
| import 'package:material_symbols_icons/symbols.dart'; | import 'package:material_symbols_icons/symbols.dart'; | ||||||
| import 'package:riverpod_annotation/riverpod_annotation.dart'; | import 'package:riverpod_annotation/riverpod_annotation.dart'; | ||||||
| import 'package:riverpod_paging_utils/riverpod_paging_utils.dart'; | import 'package:riverpod_paging_utils/riverpod_paging_utils.dart'; | ||||||
|  | import 'package:island/widgets/extended_refresh_indicator.dart'; | ||||||
|  |  | ||||||
| part 'poll_list.g.dart'; | part 'poll_list.g.dart'; | ||||||
|  |  | ||||||
| @@ -86,7 +87,7 @@ class CreatorPollListScreen extends HookConsumerWidget { | |||||||
|         onPressed: () => _createPoll(context), |         onPressed: () => _createPoll(context), | ||||||
|         child: const Icon(Icons.add), |         child: const Icon(Icons.add), | ||||||
|       ), |       ), | ||||||
|       body: RefreshIndicator( |       body: ExtendedRefreshIndicator( | ||||||
|         onRefresh: () => ref.refresh(pollListNotifierProvider(pubName).future), |         onRefresh: () => ref.refresh(pollListNotifierProvider(pubName).future), | ||||||
|         child: CustomScrollView( |         child: CustomScrollView( | ||||||
|           slivers: [ |           slivers: [ | ||||||
|   | |||||||
| @@ -4,6 +4,7 @@ import 'package:go_router/go_router.dart'; | |||||||
| import 'package:island/pods/webfeed.dart'; | import 'package:island/pods/webfeed.dart'; | ||||||
| import 'package:island/widgets/app_scaffold.dart'; | import 'package:island/widgets/app_scaffold.dart'; | ||||||
| import 'package:island/widgets/empty_state.dart'; | import 'package:island/widgets/empty_state.dart'; | ||||||
|  | import 'package:island/widgets/extended_refresh_indicator.dart'; | ||||||
| import 'package:material_symbols_icons/symbols.dart'; | import 'package:material_symbols_icons/symbols.dart'; | ||||||
|  |  | ||||||
| class WebFeedListScreen extends ConsumerWidget { | class WebFeedListScreen extends ConsumerWidget { | ||||||
| @@ -20,7 +21,10 @@ class WebFeedListScreen extends ConsumerWidget { | |||||||
|       floatingActionButton: FloatingActionButton( |       floatingActionButton: FloatingActionButton( | ||||||
|         child: const Icon(Symbols.add), |         child: const Icon(Symbols.add), | ||||||
|         onPressed: () { |         onPressed: () { | ||||||
|           context.pushNamed('creatorFeedNew', pathParameters: {'name': pubName}); |           context.pushNamed( | ||||||
|  |             'creatorFeedNew', | ||||||
|  |             pathParameters: {'name': pubName}, | ||||||
|  |           ); | ||||||
|         }, |         }, | ||||||
|       ), |       ), | ||||||
|       body: feedsAsync.when( |       body: feedsAsync.when( | ||||||
| @@ -32,7 +36,7 @@ class WebFeedListScreen extends ConsumerWidget { | |||||||
|               description: 'Add a new web feed to get started', |               description: 'Add a new web feed to get started', | ||||||
|             ); |             ); | ||||||
|           } |           } | ||||||
|           return RefreshIndicator( |           return ExtendedRefreshIndicator( | ||||||
|             onRefresh: () => ref.refresh(webFeedListProvider(pubName).future), |             onRefresh: () => ref.refresh(webFeedListProvider(pubName).future), | ||||||
|             child: ListView.builder( |             child: ListView.builder( | ||||||
|               padding: EdgeInsets.only(top: 8), |               padding: EdgeInsets.only(top: 8), | ||||||
| @@ -62,7 +66,10 @@ class WebFeedListScreen extends ConsumerWidget { | |||||||
|                     ), |                     ), | ||||||
|                     trailing: const Icon(Symbols.chevron_right), |                     trailing: const Icon(Symbols.chevron_right), | ||||||
|                     onTap: () { |                     onTap: () { | ||||||
|                       context.pushNamed('creatorFeedEdit', pathParameters: {'name': pubName, 'feedId': feed.id}); |                       context.pushNamed( | ||||||
|  |                         'creatorFeedEdit', | ||||||
|  |                         pathParameters: {'name': pubName, 'feedId': feed.id}, | ||||||
|  |                       ); | ||||||
|                     }, |                     }, | ||||||
|                   ), |                   ), | ||||||
|                 ); |                 ); | ||||||
|   | |||||||
| @@ -27,7 +27,9 @@ class AppDetailScreen extends HookConsumerWidget { | |||||||
|   @override |   @override | ||||||
|   Widget build(BuildContext context, WidgetRef ref) { |   Widget build(BuildContext context, WidgetRef ref) { | ||||||
|     final tabController = useTabController(initialLength: 2); |     final tabController = useTabController(initialLength: 2); | ||||||
|     final appData = ref.watch(customAppProvider(publisherName, projectId, appId)); |     final appData = ref.watch( | ||||||
|  |       customAppProvider(publisherName, projectId, appId), | ||||||
|  |     ); | ||||||
|  |  | ||||||
|     return AppScaffold( |     return AppScaffold( | ||||||
|       appBar: AppBar( |       appBar: AppBar( | ||||||
| @@ -35,23 +37,43 @@ class AppDetailScreen extends HookConsumerWidget { | |||||||
|         actions: [ |         actions: [ | ||||||
|           IconButton( |           IconButton( | ||||||
|             icon: const Icon(Symbols.edit), |             icon: const Icon(Symbols.edit), | ||||||
|             onPressed: appData.value == null |             onPressed: | ||||||
|                 ? null |                 appData.value == null | ||||||
|                 : () { |                     ? null | ||||||
|                     context.pushNamed( |                     : () { | ||||||
|                       'developerAppEdit', |                       context.pushNamed( | ||||||
|                       pathParameters: { |                         'developerAppEdit', | ||||||
|                         'name': publisherName, |                         pathParameters: { | ||||||
|                         'projectId': projectId, |                           'name': publisherName, | ||||||
|                         'id': appId, |                           'projectId': projectId, | ||||||
|                       }, |                           'id': appId, | ||||||
|                     ); |                         }, | ||||||
|                   }, |                       ); | ||||||
|  |                     }, | ||||||
|           ), |           ), | ||||||
|         ], |         ], | ||||||
|         bottom: TabBar( |         bottom: TabBar( | ||||||
|           controller: tabController, |           controller: tabController, | ||||||
|           tabs: [Tab(text: 'overview'.tr()), Tab(text: 'secrets'.tr())], |           tabs: [ | ||||||
|  |             Tab( | ||||||
|  |               child: Text( | ||||||
|  |                 'overview'.tr(), | ||||||
|  |                 textAlign: TextAlign.center, | ||||||
|  |                 style: TextStyle( | ||||||
|  |                   color: Theme.of(context).appBarTheme.foregroundColor!, | ||||||
|  |                 ), | ||||||
|  |               ), | ||||||
|  |             ), | ||||||
|  |             Tab( | ||||||
|  |               child: Text( | ||||||
|  |                 'secrets'.tr(), | ||||||
|  |                 textAlign: TextAlign.center, | ||||||
|  |                 style: TextStyle( | ||||||
|  |                   color: Theme.of(context).appBarTheme.foregroundColor!, | ||||||
|  |                 ), | ||||||
|  |               ), | ||||||
|  |             ), | ||||||
|  |           ], | ||||||
|         ), |         ), | ||||||
|       ), |       ), | ||||||
|       body: appData.when( |       body: appData.when( | ||||||
| @@ -70,12 +92,14 @@ class AppDetailScreen extends HookConsumerWidget { | |||||||
|           ); |           ); | ||||||
|         }, |         }, | ||||||
|         loading: () => const Center(child: CircularProgressIndicator()), |         loading: () => const Center(child: CircularProgressIndicator()), | ||||||
|         error: (err, stack) => ResponseErrorWidget( |         error: | ||||||
|           error: err, |             (err, stack) => ResponseErrorWidget( | ||||||
|           onRetry: () => ref.invalidate( |               error: err, | ||||||
|             customAppProvider(publisherName, projectId, appId), |               onRetry: | ||||||
|           ), |                   () => ref.invalidate( | ||||||
|         ), |                     customAppProvider(publisherName, projectId, appId), | ||||||
|  |                   ), | ||||||
|  |             ), | ||||||
|       ), |       ), | ||||||
|     ); |     ); | ||||||
|   } |   } | ||||||
| @@ -98,12 +122,13 @@ class _AppOverview extends StatelessWidget { | |||||||
|               children: [ |               children: [ | ||||||
|                 Container( |                 Container( | ||||||
|                   color: Theme.of(context).colorScheme.surfaceContainer, |                   color: Theme.of(context).colorScheme.surfaceContainer, | ||||||
|                   child: app.background != null |                   child: | ||||||
|                       ? CloudFileWidget( |                       app.background != null | ||||||
|                           item: app.background!, |                           ? CloudFileWidget( | ||||||
|                           fit: BoxFit.cover, |                             item: app.background!, | ||||||
|                         ) |                             fit: BoxFit.cover, | ||||||
|                       : const SizedBox.shrink(), |                           ) | ||||||
|  |                           : const SizedBox.shrink(), | ||||||
|                 ), |                 ), | ||||||
|                 Positioned( |                 Positioned( | ||||||
|                   left: 20, |                   left: 20, | ||||||
|   | |||||||
| @@ -52,7 +52,26 @@ class BotDetailScreen extends HookConsumerWidget { | |||||||
|         ], |         ], | ||||||
|         bottom: TabBar( |         bottom: TabBar( | ||||||
|           controller: tabController, |           controller: tabController, | ||||||
|           tabs: [Tab(text: 'overview'.tr()), Tab(text: 'keys'.tr())], |           tabs: [ | ||||||
|  |             Tab( | ||||||
|  |               child: Text( | ||||||
|  |                 'overview'.tr(), | ||||||
|  |                 textAlign: TextAlign.center, | ||||||
|  |                 style: TextStyle( | ||||||
|  |                   color: Theme.of(context).appBarTheme.foregroundColor!, | ||||||
|  |                 ), | ||||||
|  |               ), | ||||||
|  |             ), | ||||||
|  |             Tab( | ||||||
|  |               child: Text( | ||||||
|  |                 'keys'.tr(), | ||||||
|  |                 textAlign: TextAlign.center, | ||||||
|  |                 style: TextStyle( | ||||||
|  |                   color: Theme.of(context).appBarTheme.foregroundColor!, | ||||||
|  |                 ), | ||||||
|  |               ), | ||||||
|  |             ), | ||||||
|  |           ], | ||||||
|         ), |         ), | ||||||
|       ), |       ), | ||||||
|       body: botData.when( |       body: botData.when( | ||||||
|   | |||||||
| @@ -9,6 +9,7 @@ import 'package:island/widgets/content/cloud_files.dart'; | |||||||
| import 'package:island/widgets/response.dart'; | import 'package:island/widgets/response.dart'; | ||||||
| import 'package:material_symbols_icons/symbols.dart'; | import 'package:material_symbols_icons/symbols.dart'; | ||||||
| import 'package:riverpod_annotation/riverpod_annotation.dart'; | import 'package:riverpod_annotation/riverpod_annotation.dart'; | ||||||
|  | import 'package:island/widgets/extended_refresh_indicator.dart'; | ||||||
|  |  | ||||||
| part 'bots.g.dart'; | part 'bots.g.dart'; | ||||||
|  |  | ||||||
| @@ -60,7 +61,7 @@ class BotsScreen extends HookConsumerWidget { | |||||||
|             ), |             ), | ||||||
|           ); |           ); | ||||||
|         } |         } | ||||||
|         return RefreshIndicator( |         return ExtendedRefreshIndicator( | ||||||
|           onRefresh: |           onRefresh: | ||||||
|               () => ref.refresh(botsProvider(publisherName, projectId).future), |               () => ref.refresh(botsProvider(publisherName, projectId).future), | ||||||
|           child: ListView.builder( |           child: ListView.builder( | ||||||
|   | |||||||
| @@ -58,7 +58,26 @@ class ProjectDetailScreen extends HookConsumerWidget { | |||||||
|         ], |         ], | ||||||
|         bottom: TabBar( |         bottom: TabBar( | ||||||
|           controller: tabController, |           controller: tabController, | ||||||
|           tabs: [Tab(text: 'customApps'.tr()), Tab(text: 'bots'.tr())], |           tabs: [ | ||||||
|  |             Tab( | ||||||
|  |               child: Text( | ||||||
|  |                 'customApps'.tr(), | ||||||
|  |                 textAlign: TextAlign.center, | ||||||
|  |                 style: TextStyle( | ||||||
|  |                   color: Theme.of(context).appBarTheme.foregroundColor!, | ||||||
|  |                 ), | ||||||
|  |               ), | ||||||
|  |             ), | ||||||
|  |             Tab( | ||||||
|  |               child: Text( | ||||||
|  |                 'bots'.tr(), | ||||||
|  |                 textAlign: TextAlign.center, | ||||||
|  |                 style: TextStyle( | ||||||
|  |                   color: Theme.of(context).appBarTheme.foregroundColor!, | ||||||
|  |                 ), | ||||||
|  |               ), | ||||||
|  |             ), | ||||||
|  |           ], | ||||||
|         ), |         ), | ||||||
|       ), |       ), | ||||||
|       body: TabBarView( |       body: TabBarView( | ||||||
|   | |||||||
| @@ -143,8 +143,26 @@ class ArticlesScreen extends ConsumerWidget { | |||||||
|               bottom: TabBar( |               bottom: TabBar( | ||||||
|                 isScrollable: true, |                 isScrollable: true, | ||||||
|                 tabs: [ |                 tabs: [ | ||||||
|                   const Tab(text: 'All'), |                   Tab( | ||||||
|                   ...feeds.map((feed) => Tab(text: feed.title)), |                     child: Text( | ||||||
|  |                       'All', | ||||||
|  |                       textAlign: TextAlign.center, | ||||||
|  |                       style: TextStyle( | ||||||
|  |                         color: Theme.of(context).appBarTheme.foregroundColor!, | ||||||
|  |                       ), | ||||||
|  |                     ), | ||||||
|  |                   ), | ||||||
|  |                   ...feeds.map( | ||||||
|  |                     (feed) => Tab( | ||||||
|  |                       child: Text( | ||||||
|  |                         feed.title, | ||||||
|  |                         textAlign: TextAlign.center, | ||||||
|  |                         style: TextStyle( | ||||||
|  |                           color: Theme.of(context).appBarTheme.foregroundColor!, | ||||||
|  |                         ), | ||||||
|  |                       ), | ||||||
|  |                     ), | ||||||
|  |                   ), | ||||||
|                 ], |                 ], | ||||||
|               ), |               ), | ||||||
|             ), |             ), | ||||||
|   | |||||||
| @@ -72,7 +72,7 @@ class MarketplaceWebFeedsScreen extends HookConsumerWidget { | |||||||
|         searchController.clear(); |         searchController.clear(); | ||||||
|       } |       } | ||||||
|       return null; |       return null; | ||||||
|     }, [query.value]); |     }, [query]); | ||||||
|  |  | ||||||
|     // Clean up timer on dispose |     // Clean up timer on dispose | ||||||
|     useEffect(() { |     useEffect(() { | ||||||
|   | |||||||
| @@ -27,6 +27,7 @@ import 'package:island/pods/network.dart'; | |||||||
| import 'package:island/widgets/realm/realm_card.dart'; | import 'package:island/widgets/realm/realm_card.dart'; | ||||||
| import 'package:island/widgets/publisher/publisher_card.dart'; | import 'package:island/widgets/publisher/publisher_card.dart'; | ||||||
| import 'package:island/widgets/web_article_card.dart'; | import 'package:island/widgets/web_article_card.dart'; | ||||||
|  | import 'package:island/widgets/extended_refresh_indicator.dart'; | ||||||
| import 'package:styled_widget/styled_widget.dart'; | import 'package:styled_widget/styled_widget.dart'; | ||||||
|  |  | ||||||
| part 'explore.g.dart'; | part 'explore.g.dart'; | ||||||
| @@ -368,7 +369,7 @@ class ExploreScreen extends HookConsumerWidget { | |||||||
|  |  | ||||||
|     final isWide = isWideScreen(context); |     final isWide = isWideScreen(context); | ||||||
|  |  | ||||||
|     return RefreshIndicator( |     return ExtendedRefreshIndicator( | ||||||
|       onRefresh: () => Future.sync(activitiesNotifier.forceRefresh), |       onRefresh: () => Future.sync(activitiesNotifier.forceRefresh), | ||||||
|       child: PagingHelperView( |       child: PagingHelperView( | ||||||
|         provider: activityListNotifierProvider(filter), |         provider: activityListNotifierProvider(filter), | ||||||
| @@ -399,6 +400,69 @@ class _DiscoveryActivityItem extends StatelessWidget { | |||||||
|     final items = data['items'] as List; |     final items = data['items'] as List; | ||||||
|     final type = items.firstOrNull?['type'] ?? 'unknown'; |     final type = items.firstOrNull?['type'] ?? 'unknown'; | ||||||
|  |  | ||||||
|  |     var flexWeights = isWideScreen(context) ? <int>[3, 2, 1] : <int>[4, 1]; | ||||||
|  |     if (type == 'post') flexWeights = <int>[3, 2]; | ||||||
|  |  | ||||||
|  |     final height = type == 'post' ? 280.0 : 180.0; | ||||||
|  |  | ||||||
|  |     final contentWidget = switch (type) { | ||||||
|  |       'post' => ListView.separated( | ||||||
|  |         scrollDirection: Axis.horizontal, | ||||||
|  |         itemCount: items.length, | ||||||
|  |         separatorBuilder: (context, index) => const Gap(12), | ||||||
|  |         padding: EdgeInsets.symmetric(horizontal: 8, vertical: 4), | ||||||
|  |         itemBuilder: (context, index) { | ||||||
|  |           final item = items[index]; | ||||||
|  |           return Container( | ||||||
|  |             width: 320, | ||||||
|  |             decoration: BoxDecoration( | ||||||
|  |               border: Border.all( | ||||||
|  |                 width: 1 / MediaQuery.of(context).devicePixelRatio, | ||||||
|  |                 color: Theme.of(context).dividerColor.withOpacity(0.5), | ||||||
|  |               ), | ||||||
|  |               borderRadius: const BorderRadius.all(Radius.circular(8)), | ||||||
|  |             ), | ||||||
|  |             child: ClipRRect( | ||||||
|  |               borderRadius: const BorderRadius.all(Radius.circular(8)), | ||||||
|  |               child: SingleChildScrollView( | ||||||
|  |                 child: PostActionableItem( | ||||||
|  |                   item: SnPost.fromJson(item['data']), | ||||||
|  |                   isCompact: true, | ||||||
|  |                 ), | ||||||
|  |               ), | ||||||
|  |             ), | ||||||
|  |           ); | ||||||
|  |         }, | ||||||
|  |       ), | ||||||
|  |       _ => CarouselView.weighted( | ||||||
|  |         flexWeights: flexWeights, | ||||||
|  |         consumeMaxWeight: false, | ||||||
|  |         enableSplash: false, | ||||||
|  |         shape: const RoundedRectangleBorder( | ||||||
|  |           borderRadius: BorderRadius.all(Radius.circular(8)), | ||||||
|  |         ), | ||||||
|  |         itemSnapping: false, | ||||||
|  |         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, | ||||||
|  |               ), | ||||||
|  |               _ => const Placeholder(), | ||||||
|  |             }, | ||||||
|  |         ], | ||||||
|  |       ), | ||||||
|  |     }; | ||||||
|  |  | ||||||
|     return Card( |     return Card( | ||||||
|       margin: EdgeInsets.symmetric(horizontal: 8, vertical: 4), |       margin: EdgeInsets.symmetric(horizontal: 8, vertical: 4), | ||||||
|       child: Column( |       child: Column( | ||||||
| @@ -407,13 +471,20 @@ class _DiscoveryActivityItem extends StatelessWidget { | |||||||
|           Row( |           Row( | ||||||
|             crossAxisAlignment: CrossAxisAlignment.center, |             crossAxisAlignment: CrossAxisAlignment.center, | ||||||
|             children: [ |             children: [ | ||||||
|               const Icon(Symbols.explore, size: 19), |               Icon(switch (type) { | ||||||
|  |                 'realm' => Symbols.public, | ||||||
|  |                 'publisher' => Symbols.account_circle, | ||||||
|  |                 'article' => Symbols.auto_stories, | ||||||
|  |                 'post' => Symbols.shuffle, | ||||||
|  |                 _ => Symbols.explore, | ||||||
|  |               }, size: 19), | ||||||
|               const Gap(8), |               const Gap(8), | ||||||
|               Text( |               Text( | ||||||
|                 (switch (type) { |                 (switch (type) { | ||||||
|                   'realm' => 'discoverRealms', |                   'realm' => 'discoverRealms', | ||||||
|                   'publisher' => 'discoverPublishers', |                   'publisher' => 'discoverPublishers', | ||||||
|                   'article' => 'discoverWebArticles', |                   'article' => 'discoverWebArticles', | ||||||
|  |                   'post' => 'discoverShuffledPost', | ||||||
|                   _ => 'unknown', |                   _ => 'unknown', | ||||||
|                 }).tr(), |                 }).tr(), | ||||||
|                 style: Theme.of(context).textTheme.titleMedium, |                 style: Theme.of(context).textTheme.titleMedium, | ||||||
| @@ -421,37 +492,8 @@ class _DiscoveryActivityItem extends StatelessWidget { | |||||||
|             ], |             ], | ||||||
|           ).padding(horizontal: 20, top: 8, bottom: 4), |           ).padding(horizontal: 20, top: 8, bottom: 4), | ||||||
|           SizedBox( |           SizedBox( | ||||||
|             height: 180, |             height: height, | ||||||
|             child: ConstrainedBox( |             child: contentWidget, | ||||||
|               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), |           ).padding(bottom: 8, horizontal: 8), | ||||||
|         ], |         ], | ||||||
|       ), |       ), | ||||||
| @@ -569,7 +611,8 @@ class ActivityListNotifier extends _$ActivityListNotifier | |||||||
|       if (cursor != null) 'cursor': cursor, |       if (cursor != null) 'cursor': cursor, | ||||||
|       'take': take, |       'take': take, | ||||||
|       if (filter != null) 'filter': filter, |       if (filter != null) 'filter': filter, | ||||||
|       if (kDebugMode) 'debugInclude': 'realms,publishers,articles', |       if (kDebugMode) | ||||||
|  |         'debugInclude': 'realms,publishers,articles,shuffledPosts', | ||||||
|     }; |     }; | ||||||
|  |  | ||||||
|     final response = await client.get( |     final response = await client.get( | ||||||
| @@ -584,12 +627,13 @@ class ActivityListNotifier extends _$ActivityListNotifier | |||||||
|  |  | ||||||
|     final hasMore = (items.firstOrNull?.type ?? 'empty') != 'empty'; |     final hasMore = (items.firstOrNull?.type ?? 'empty') != 'empty'; | ||||||
|     final nextCursor = |     final nextCursor = | ||||||
|         items |         items.isNotEmpty | ||||||
|             .map((x) => x.createdAt) |             ? items | ||||||
|             .lastOrNull |                 .map((x) => x.createdAt) | ||||||
|             ?.toUtc() |                 .reduce((a, b) => a.isBefore(b) ? a : b) | ||||||
|             .toIso8601String() |                 .toUtc() | ||||||
|             .toString(); |                 .toIso8601String() | ||||||
|  |             : null; | ||||||
|  |  | ||||||
|     return CursorPagingData( |     return CursorPagingData( | ||||||
|       items: items, |       items: items, | ||||||
|   | |||||||
| @@ -7,7 +7,7 @@ part of 'explore.dart'; | |||||||
| // ************************************************************************** | // ************************************************************************** | ||||||
|  |  | ||||||
| String _$activityListNotifierHash() => | String _$activityListNotifierHash() => | ||||||
|     r'b75fd5c08d5f84ca433e16b7387d317ea72b91c9'; |     r'167021cada54da7c8d8437eef1ffb387a92ea2e3'; | ||||||
|  |  | ||||||
| /// Copied from Dart SDK | /// Copied from Dart SDK | ||||||
| class _SystemHash { | class _SystemHash { | ||||||
|   | |||||||
| @@ -151,7 +151,9 @@ class PostCategoryDetailScreen extends HookConsumerWidget { | |||||||
|                                       (error, _) => Text( |                                       (error, _) => Text( | ||||||
|                                         'Error loading subscription status', |                                         'Error loading subscription status', | ||||||
|                                       ), |                                       ), | ||||||
|                                   loading: () => CircularProgressIndicator(), |                                   loading: | ||||||
|  |                                       () => | ||||||
|  |                                           CircularProgressIndicator().center(), | ||||||
|                                 ), |                                 ), | ||||||
|                               ], |                               ], | ||||||
|                             ).padding(horizontal: 24, vertical: 16), |                             ).padding(horizontal: 24, vertical: 16), | ||||||
| @@ -220,7 +222,9 @@ class PostCategoryDetailScreen extends HookConsumerWidget { | |||||||
|                                       (error, _) => Text( |                                       (error, _) => Text( | ||||||
|                                         'Error loading subscription status', |                                         'Error loading subscription status', | ||||||
|                                       ), |                                       ), | ||||||
|                                   loading: () => CircularProgressIndicator(), |                                   loading: | ||||||
|  |                                       () => | ||||||
|  |                                           CircularProgressIndicator().center(), | ||||||
|                                 ), |                                 ), | ||||||
|                               ], |                               ], | ||||||
|                             ).padding(horizontal: 24, vertical: 16), |                             ).padding(horizontal: 24, vertical: 16), | ||||||
|   | |||||||
| @@ -1,15 +1,26 @@ | |||||||
| import 'package:easy_localization/easy_localization.dart'; | import 'package:easy_localization/easy_localization.dart'; | ||||||
|  | import 'package:flutter/foundation.dart'; | ||||||
| import 'package:flutter/material.dart'; | import 'package:flutter/material.dart'; | ||||||
|  | import 'package:flutter/services.dart'; | ||||||
| import 'package:gap/gap.dart'; | import 'package:gap/gap.dart'; | ||||||
|  | import 'package:go_router/go_router.dart'; | ||||||
| import 'package:hooks_riverpod/hooks_riverpod.dart'; | import 'package:hooks_riverpod/hooks_riverpod.dart'; | ||||||
| import 'package:island/models/post.dart'; | import 'package:island/models/post.dart'; | ||||||
| import 'package:island/pods/network.dart'; | import 'package:island/pods/network.dart'; | ||||||
| import 'package:island/pods/userinfo.dart'; | import 'package:island/pods/userinfo.dart'; | ||||||
|  | import 'package:island/screens/posts/compose.dart'; | ||||||
|  | import 'package:island/widgets/alert.dart'; | ||||||
| import 'package:island/widgets/app_scaffold.dart'; | import 'package:island/widgets/app_scaffold.dart'; | ||||||
|  | import 'package:island/widgets/extended_refresh_indicator.dart'; | ||||||
| import 'package:island/widgets/post/post_item.dart'; | import 'package:island/widgets/post/post_item.dart'; | ||||||
|  | import 'package:island/widgets/post/post_pin_sheet.dart'; | ||||||
| import 'package:island/widgets/post/post_quick_reply.dart'; | import 'package:island/widgets/post/post_quick_reply.dart'; | ||||||
| import 'package:island/widgets/post/post_replies.dart'; | import 'package:island/widgets/post/post_replies.dart'; | ||||||
| import 'package:island/widgets/response.dart'; | import 'package:island/widgets/response.dart'; | ||||||
|  | import 'package:island/utils/share_utils.dart'; | ||||||
|  | import 'package:island/widgets/safety/abuse_report_helper.dart'; | ||||||
|  | import 'package:island/widgets/share/share_sheet.dart'; | ||||||
|  | import 'package:material_symbols_icons/symbols.dart'; | ||||||
| import 'package:riverpod_annotation/riverpod_annotation.dart'; | import 'package:riverpod_annotation/riverpod_annotation.dart'; | ||||||
| import 'package:styled_widget/styled_widget.dart'; | import 'package:styled_widget/styled_widget.dart'; | ||||||
|  |  | ||||||
| @@ -46,6 +57,321 @@ class PostState extends StateNotifier<AsyncValue<SnPost?>> { | |||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | class PostActionButtons extends HookConsumerWidget { | ||||||
|  |   final SnPost post; | ||||||
|  |   final EdgeInsets renderingPadding; | ||||||
|  |   final VoidCallback? onRefresh; | ||||||
|  |   final Function(SnPost)? onUpdate; | ||||||
|  |  | ||||||
|  |   const PostActionButtons({ | ||||||
|  |     super.key, | ||||||
|  |     required this.post, | ||||||
|  |     this.renderingPadding = EdgeInsets.zero, | ||||||
|  |     this.onRefresh, | ||||||
|  |     this.onUpdate, | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Widget build(BuildContext context, WidgetRef ref) { | ||||||
|  |     final user = ref.watch(userInfoProvider); | ||||||
|  |     final isAuthor = | ||||||
|  |         user.value != null && user.value?.id == post.publisher.accountId; | ||||||
|  |  | ||||||
|  |     final actions = <Widget>[]; | ||||||
|  |  | ||||||
|  |     const kButtonHeight = 40.0; | ||||||
|  |     const kButtonRadius = 20.0; | ||||||
|  |  | ||||||
|  |     // 1. Author-only actions first | ||||||
|  |     if (isAuthor) { | ||||||
|  |       // Combined edit/delete actions using custom segmented-style buttons | ||||||
|  |       final editButtons = <Widget>[ | ||||||
|  |         FilledButton.tonal( | ||||||
|  |           onPressed: () { | ||||||
|  |             context.pushNamed('postEdit', pathParameters: {'id': post.id}).then( | ||||||
|  |               (value) { | ||||||
|  |                 if (value != null) { | ||||||
|  |                   onRefresh?.call(); | ||||||
|  |                 } | ||||||
|  |               }, | ||||||
|  |             ); | ||||||
|  |           }, | ||||||
|  |           style: FilledButton.styleFrom( | ||||||
|  |             shape: const RoundedRectangleBorder( | ||||||
|  |               borderRadius: BorderRadius.only( | ||||||
|  |                 topLeft: Radius.circular(kButtonRadius), | ||||||
|  |                 bottomLeft: Radius.circular(kButtonRadius), | ||||||
|  |               ), | ||||||
|  |             ), | ||||||
|  |           ), | ||||||
|  |           child: Row( | ||||||
|  |             mainAxisSize: MainAxisSize.min, | ||||||
|  |             children: [ | ||||||
|  |               const Icon(Symbols.edit, size: 18), | ||||||
|  |               const Gap(4), | ||||||
|  |               Text('edit'.tr()), | ||||||
|  |             ], | ||||||
|  |           ), | ||||||
|  |         ), | ||||||
|  |         Tooltip( | ||||||
|  |           message: 'delete'.tr(), | ||||||
|  |           child: FilledButton.tonal( | ||||||
|  |             onPressed: () { | ||||||
|  |               showConfirmAlert('deletePostHint'.tr(), 'deletePost'.tr()).then(( | ||||||
|  |                 confirm, | ||||||
|  |               ) { | ||||||
|  |                 if (confirm) { | ||||||
|  |                   final client = ref.watch(apiClientProvider); | ||||||
|  |                   client | ||||||
|  |                       .delete('/sphere/posts/${post.id}') | ||||||
|  |                       .catchError((err) { | ||||||
|  |                         showErrorAlert(err); | ||||||
|  |                         return err; | ||||||
|  |                       }) | ||||||
|  |                       .then((_) { | ||||||
|  |                         onRefresh?.call(); | ||||||
|  |                       }); | ||||||
|  |                 } | ||||||
|  |               }); | ||||||
|  |             }, | ||||||
|  |             style: FilledButton.styleFrom( | ||||||
|  |               shape: const RoundedRectangleBorder( | ||||||
|  |                 borderRadius: BorderRadius.only( | ||||||
|  |                   topRight: Radius.circular(kButtonRadius), | ||||||
|  |                   bottomRight: Radius.circular(kButtonRadius), | ||||||
|  |                 ), | ||||||
|  |               ), | ||||||
|  |             ), | ||||||
|  |             child: const Icon(Symbols.delete, size: 18), | ||||||
|  |           ), | ||||||
|  |         ), | ||||||
|  |       ]; | ||||||
|  |  | ||||||
|  |       actions.add( | ||||||
|  |         Row( | ||||||
|  |           mainAxisSize: MainAxisSize.min, | ||||||
|  |           children: | ||||||
|  |               editButtons | ||||||
|  |                   .map((e) => SizedBox(height: kButtonHeight, child: e)) | ||||||
|  |                   .expand((widget) => [widget, const VerticalDivider(width: 1)]) | ||||||
|  |                   .toList() | ||||||
|  |                 ..removeLast(), | ||||||
|  |         ), | ||||||
|  |       ); | ||||||
|  |  | ||||||
|  |       // Pin/Unpin actions (also author-only) | ||||||
|  |       if (post.pinMode == null) { | ||||||
|  |         actions.add( | ||||||
|  |           FilledButton.tonalIcon( | ||||||
|  |             onPressed: () { | ||||||
|  |               showModalBottomSheet( | ||||||
|  |                 context: context, | ||||||
|  |                 isScrollControlled: true, | ||||||
|  |                 builder: (context) => PostPinSheet(post: post), | ||||||
|  |               ).then((value) { | ||||||
|  |                 if (value is int) { | ||||||
|  |                   onUpdate?.call(post.copyWith(pinMode: value)); | ||||||
|  |                 } | ||||||
|  |               }); | ||||||
|  |             }, | ||||||
|  |             icon: const Icon(Symbols.keep), | ||||||
|  |             label: Text('pinPost'.tr()), | ||||||
|  |           ), | ||||||
|  |         ); | ||||||
|  |       } else { | ||||||
|  |         actions.add( | ||||||
|  |           FilledButton.tonalIcon( | ||||||
|  |             onPressed: () { | ||||||
|  |               showConfirmAlert('unpinPostHint'.tr(), 'unpinPost'.tr()).then(( | ||||||
|  |                 confirm, | ||||||
|  |               ) async { | ||||||
|  |                 if (confirm) { | ||||||
|  |                   final client = ref.watch(apiClientProvider); | ||||||
|  |                   try { | ||||||
|  |                     if (context.mounted) showLoadingModal(context); | ||||||
|  |                     await client.delete('/sphere/posts/${post.id}/pin'); | ||||||
|  |                     onUpdate?.call(post.copyWith(pinMode: null)); | ||||||
|  |                   } catch (err) { | ||||||
|  |                     showErrorAlert(err); | ||||||
|  |                   } finally { | ||||||
|  |                     if (context.mounted) hideLoadingModal(context); | ||||||
|  |                   } | ||||||
|  |                 } | ||||||
|  |               }); | ||||||
|  |             }, | ||||||
|  |             icon: const Icon(Symbols.keep_off), | ||||||
|  |             label: Text('unpinPost'.tr()), | ||||||
|  |           ), | ||||||
|  |         ); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     // 2. Replies and forwards | ||||||
|  |     final replyButtons = <Widget>[ | ||||||
|  |       FilledButton.tonal( | ||||||
|  |         onPressed: () { | ||||||
|  |           context.pushNamed( | ||||||
|  |             'postCompose', | ||||||
|  |             extra: PostComposeInitialState(replyingTo: post), | ||||||
|  |           ); | ||||||
|  |         }, | ||||||
|  |         style: FilledButton.styleFrom( | ||||||
|  |           shape: const RoundedRectangleBorder( | ||||||
|  |             borderRadius: BorderRadius.only( | ||||||
|  |               topLeft: Radius.circular(kButtonRadius), | ||||||
|  |               bottomLeft: Radius.circular(kButtonRadius), | ||||||
|  |             ), | ||||||
|  |           ), | ||||||
|  |         ), | ||||||
|  |         child: Row( | ||||||
|  |           mainAxisSize: MainAxisSize.min, | ||||||
|  |           children: [ | ||||||
|  |             const Icon(Symbols.reply, size: 18), | ||||||
|  |             const Gap(4), | ||||||
|  |             Text('reply'.tr()), | ||||||
|  |           ], | ||||||
|  |         ), | ||||||
|  |       ), | ||||||
|  |       Tooltip( | ||||||
|  |         message: 'forward'.tr(), | ||||||
|  |         child: FilledButton.tonal( | ||||||
|  |           onPressed: () { | ||||||
|  |             context.pushNamed( | ||||||
|  |               'postCompose', | ||||||
|  |               extra: PostComposeInitialState(forwardingTo: post), | ||||||
|  |             ); | ||||||
|  |           }, | ||||||
|  |           style: FilledButton.styleFrom( | ||||||
|  |             shape: const RoundedRectangleBorder( | ||||||
|  |               borderRadius: BorderRadius.only( | ||||||
|  |                 topRight: Radius.circular(kButtonRadius), | ||||||
|  |                 bottomRight: Radius.circular(kButtonRadius), | ||||||
|  |               ), | ||||||
|  |             ), | ||||||
|  |           ), | ||||||
|  |           child: const Icon(Symbols.forward, size: 18), | ||||||
|  |         ), | ||||||
|  |       ), | ||||||
|  |     ]; | ||||||
|  |  | ||||||
|  |     actions.add( | ||||||
|  |       Row( | ||||||
|  |         mainAxisSize: MainAxisSize.min, | ||||||
|  |         children: | ||||||
|  |             replyButtons | ||||||
|  |                 .map((e) => SizedBox(height: kButtonHeight, child: e)) | ||||||
|  |                 .toList(), | ||||||
|  |       ), | ||||||
|  |     ); | ||||||
|  |  | ||||||
|  |     // 3. Share, copy link, and report | ||||||
|  |     final shareButtons = <Widget>[ | ||||||
|  |       FilledButton.tonal( | ||||||
|  |         onPressed: () { | ||||||
|  |           showShareSheetLink( | ||||||
|  |             context: context, | ||||||
|  |             link: 'https://solian.app/posts/${post.id}', | ||||||
|  |             title: 'sharePost'.tr(), | ||||||
|  |             toSystem: true, | ||||||
|  |           ); | ||||||
|  |         }, | ||||||
|  |         style: FilledButton.styleFrom( | ||||||
|  |           shape: const RoundedRectangleBorder( | ||||||
|  |             borderRadius: BorderRadius.only( | ||||||
|  |               topLeft: Radius.circular(kButtonRadius), | ||||||
|  |               bottomLeft: Radius.circular(kButtonRadius), | ||||||
|  |             ), | ||||||
|  |           ), | ||||||
|  |         ), | ||||||
|  |         child: Row( | ||||||
|  |           mainAxisSize: MainAxisSize.min, | ||||||
|  |           children: [ | ||||||
|  |             const Icon(Symbols.share, size: 18), | ||||||
|  |             const Gap(4), | ||||||
|  |             Text('share'.tr()), | ||||||
|  |           ], | ||||||
|  |         ), | ||||||
|  |       ), | ||||||
|  |     ]; | ||||||
|  |  | ||||||
|  |     if (!kIsWeb) { | ||||||
|  |       shareButtons.add( | ||||||
|  |         Tooltip( | ||||||
|  |           message: 'sharePostPhoto'.tr(), | ||||||
|  |           child: FilledButton.tonal( | ||||||
|  |             onPressed: () => sharePostAsScreenshot(context, ref, post), | ||||||
|  |             style: FilledButton.styleFrom( | ||||||
|  |               shape: const RoundedRectangleBorder( | ||||||
|  |                 borderRadius: BorderRadius.only( | ||||||
|  |                   topRight: Radius.circular(kButtonRadius), | ||||||
|  |                   bottomRight: Radius.circular(kButtonRadius), | ||||||
|  |                 ), | ||||||
|  |               ), | ||||||
|  |             ), | ||||||
|  |             child: const Icon(Symbols.share_reviews, size: 18), | ||||||
|  |           ), | ||||||
|  |         ), | ||||||
|  |       ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     actions.add( | ||||||
|  |       Row( | ||||||
|  |         mainAxisSize: MainAxisSize.min, | ||||||
|  |         children: | ||||||
|  |             shareButtons | ||||||
|  |                 .map((e) => SizedBox(height: kButtonHeight, child: e)) | ||||||
|  |                 .expand((widget) => [widget, const VerticalDivider(width: 1)]) | ||||||
|  |                 .toList() | ||||||
|  |               ..removeLast(), | ||||||
|  |       ), | ||||||
|  |     ); | ||||||
|  |  | ||||||
|  |     actions.add( | ||||||
|  |       FilledButton.tonalIcon( | ||||||
|  |         onPressed: () { | ||||||
|  |           Clipboard.setData( | ||||||
|  |             ClipboardData(text: 'https://solian.app/posts/${post.id}'), | ||||||
|  |           ); | ||||||
|  |         }, | ||||||
|  |         icon: const Icon(Symbols.link), | ||||||
|  |         label: Text('copyLink'.tr()), | ||||||
|  |       ), | ||||||
|  |     ); | ||||||
|  |  | ||||||
|  |     actions.add( | ||||||
|  |       FilledButton.tonalIcon( | ||||||
|  |         onPressed: () { | ||||||
|  |           showAbuseReportSheet(context, resourceIdentifier: 'post/${post.id}'); | ||||||
|  |         }, | ||||||
|  |         icon: const Icon(Symbols.flag), | ||||||
|  |         label: Text('abuseReport'.tr()), | ||||||
|  |       ), | ||||||
|  |     ); | ||||||
|  |  | ||||||
|  |     // Add gaps between actions (excluding first one) using FP style | ||||||
|  |     final children = | ||||||
|  |         actions.asMap().entries.expand((entry) { | ||||||
|  |           final index = entry.key; | ||||||
|  |           final action = entry.value; | ||||||
|  |           if (index == 0) { | ||||||
|  |             return [action]; | ||||||
|  |           } else { | ||||||
|  |             return [const Gap(8), action]; | ||||||
|  |           } | ||||||
|  |         }).toList(); | ||||||
|  |  | ||||||
|  |     return Container( | ||||||
|  |       height: kButtonHeight, | ||||||
|  |       margin: const EdgeInsets.only(bottom: 12), | ||||||
|  |       child: ListView( | ||||||
|  |         scrollDirection: Axis.horizontal, | ||||||
|  |         padding: EdgeInsets.symmetric(horizontal: renderingPadding.horizontal), | ||||||
|  |         children: children, | ||||||
|  |       ), | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
| class PostDetailScreen extends HookConsumerWidget { | class PostDetailScreen extends HookConsumerWidget { | ||||||
|   final String id; |   final String id; | ||||||
|   const PostDetailScreen({super.key, required this.id}); |   const PostDetailScreen({super.key, required this.id}); | ||||||
| @@ -66,29 +392,58 @@ class PostDetailScreen extends HookConsumerWidget { | |||||||
|           return Stack( |           return Stack( | ||||||
|             fit: StackFit.expand, |             fit: StackFit.expand, | ||||||
|             children: [ |             children: [ | ||||||
|               CustomScrollView( |               ExtendedRefreshIndicator( | ||||||
|                 slivers: [ |                 onRefresh: () async { | ||||||
|                   SliverToBoxAdapter( |                   ref.invalidate(postProvider(id)); | ||||||
|                     child: Center( |                   ref.invalidate(postRepliesNotifierProvider(id)); | ||||||
|                       child: ConstrainedBox( |                 }, | ||||||
|                         constraints: BoxConstraints(maxWidth: 600), |                 child: CustomScrollView( | ||||||
|                         child: PostItem( |                   physics: const AlwaysScrollableScrollPhysics(), | ||||||
|                           item: post!, |                   slivers: [ | ||||||
|                           isFullPost: true, |                     SliverToBoxAdapter( | ||||||
|                           isEmbedReply: false, |                       child: Center( | ||||||
|                           onUpdate: (newItem) { |                         child: ConstrainedBox( | ||||||
|                             // Update the local state with the new post data |                           constraints: BoxConstraints(maxWidth: 600), | ||||||
|                             ref |                           child: PostItem( | ||||||
|                                 .read(postStateProvider(id).notifier) |                             item: post!, | ||||||
|                                 .updatePost(newItem); |                             isFullPost: true, | ||||||
|                           }, |                             isEmbedReply: false, | ||||||
|  |                             onUpdate: (newItem) { | ||||||
|  |                               // Update the local state with the new post data | ||||||
|  |                               ref | ||||||
|  |                                   .read(postStateProvider(id).notifier) | ||||||
|  |                                   .updatePost(newItem); | ||||||
|  |                             }, | ||||||
|  |                           ), | ||||||
|                         ), |                         ), | ||||||
|                       ), |                       ), | ||||||
|                     ), |                     ), | ||||||
|                   ), |                     SliverToBoxAdapter( | ||||||
|                   PostRepliesList(postId: id, maxWidth: 600), |                       child: Center( | ||||||
|                   SliverGap(MediaQuery.of(context).padding.bottom + 80), |                         child: ConstrainedBox( | ||||||
|                 ], |                           constraints: BoxConstraints(maxWidth: 600), | ||||||
|  |                           child: PostActionButtons( | ||||||
|  |                             post: post, | ||||||
|  |                             renderingPadding: const EdgeInsets.symmetric( | ||||||
|  |                               horizontal: 8, | ||||||
|  |                             ), | ||||||
|  |                             onRefresh: () { | ||||||
|  |                               ref.invalidate(postProvider(id)); | ||||||
|  |                               ref.invalidate(postRepliesNotifierProvider(id)); | ||||||
|  |                             }, | ||||||
|  |                             onUpdate: (newItem) { | ||||||
|  |                               ref | ||||||
|  |                                   .read(postStateProvider(id).notifier) | ||||||
|  |                                   .updatePost(newItem); | ||||||
|  |                             }, | ||||||
|  |                           ), | ||||||
|  |                         ), | ||||||
|  |                       ), | ||||||
|  |                     ), | ||||||
|  |                     PostRepliesList(postId: id, maxWidth: 600), | ||||||
|  |                     SliverGap(MediaQuery.of(context).padding.bottom + 80), | ||||||
|  |                   ], | ||||||
|  |                 ), | ||||||
|               ), |               ), | ||||||
|               if (user.value != null) |               if (user.value != null) | ||||||
|                 Positioned( |                 Positioned( | ||||||
| @@ -126,7 +481,7 @@ class PostDetailScreen extends HookConsumerWidget { | |||||||
|         error: |         error: | ||||||
|             (e, _) => ResponseErrorWidget( |             (e, _) => ResponseErrorWidget( | ||||||
|               error: e, |               error: e, | ||||||
|               onRetry: () => ref.invalidate(postStateProvider(id)), |               onRetry: () => ref.invalidate(postProvider(id)), | ||||||
|             ), |             ), | ||||||
|       ), |       ), | ||||||
|     ); |     ); | ||||||
|   | |||||||
| @@ -1,5 +1,7 @@ | |||||||
| import 'dart:async'; | import 'dart:async'; | ||||||
|  | import 'package:easy_localization/easy_localization.dart'; | ||||||
| import 'package:flutter/material.dart'; | import 'package:flutter/material.dart'; | ||||||
|  | import 'package:flutter_hooks/flutter_hooks.dart'; | ||||||
| import 'package:hooks_riverpod/hooks_riverpod.dart'; | import 'package:hooks_riverpod/hooks_riverpod.dart'; | ||||||
| import 'package:island/models/post.dart'; | import 'package:island/models/post.dart'; | ||||||
| import 'package:island/pods/network.dart'; | import 'package:island/pods/network.dart'; | ||||||
| @@ -7,6 +9,7 @@ import 'package:island/widgets/app_scaffold.dart'; | |||||||
| import 'package:island/widgets/post/post_item.dart'; | import 'package:island/widgets/post/post_item.dart'; | ||||||
| import 'package:island/widgets/response.dart'; | import 'package:island/widgets/response.dart'; | ||||||
| import 'package:riverpod_paging_utils/riverpod_paging_utils.dart'; | import 'package:riverpod_paging_utils/riverpod_paging_utils.dart'; | ||||||
|  | import 'package:styled_widget/styled_widget.dart'; | ||||||
|  |  | ||||||
| final postSearchNotifierProvider = StateNotifierProvider.autoDispose< | final postSearchNotifierProvider = StateNotifierProvider.autoDispose< | ||||||
|   PostSearchNotifier, |   PostSearchNotifier, | ||||||
| @@ -18,6 +21,13 @@ class PostSearchNotifier | |||||||
|   final AutoDisposeRef ref; |   final AutoDisposeRef ref; | ||||||
|   static const int _pageSize = 20; |   static const int _pageSize = 20; | ||||||
|   String _currentQuery = ''; |   String _currentQuery = ''; | ||||||
|  |   String? _pubName; | ||||||
|  |   String? _realm; | ||||||
|  |   int? _type; | ||||||
|  |   List<String>? _categories; | ||||||
|  |   List<String>? _tags; | ||||||
|  |   bool _shuffle = false; | ||||||
|  |   bool? _pinned; | ||||||
|   bool _isLoading = false; |   bool _isLoading = false; | ||||||
|  |  | ||||||
|   PostSearchNotifier(this.ref) : super(const AsyncValue.loading()) { |   PostSearchNotifier(this.ref) : super(const AsyncValue.loading()) { | ||||||
| @@ -26,11 +36,38 @@ class PostSearchNotifier | |||||||
|     ); |     ); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   Future<void> search(String query) async { |   Future<void> search( | ||||||
|  |     String query, { | ||||||
|  |     String? pubName, | ||||||
|  |     String? realm, | ||||||
|  |     int? type, | ||||||
|  |     List<String>? categories, | ||||||
|  |     List<String>? tags, | ||||||
|  |     bool shuffle = false, | ||||||
|  |     bool? pinned, | ||||||
|  |   }) async { | ||||||
|     if (_isLoading) return; |     if (_isLoading) return; | ||||||
|  |  | ||||||
|     _currentQuery = query.trim(); |     _currentQuery = query.trim(); | ||||||
|     if (_currentQuery.isEmpty) { |     _pubName = pubName; | ||||||
|  |     _realm = realm; | ||||||
|  |     _type = type; | ||||||
|  |     _categories = categories; | ||||||
|  |     _tags = tags; | ||||||
|  |     _shuffle = shuffle; | ||||||
|  |     _pinned = pinned; | ||||||
|  |  | ||||||
|  |     // Allow search even with empty query if any filters are applied | ||||||
|  |     final hasFilters = | ||||||
|  |         pubName != null || | ||||||
|  |         realm != null || | ||||||
|  |         type != null || | ||||||
|  |         categories != null || | ||||||
|  |         tags != null || | ||||||
|  |         shuffle || | ||||||
|  |         pinned != null; | ||||||
|  |  | ||||||
|  |     if (_currentQuery.isEmpty && !hasFilters) { | ||||||
|       state = AsyncValue.data( |       state = AsyncValue.data( | ||||||
|         CursorPagingData(items: [], hasMore: false, nextCursor: null), |         CursorPagingData(items: [], hasMore: false, nextCursor: null), | ||||||
|       ); |       ); | ||||||
| @@ -57,6 +94,13 @@ class PostSearchNotifier | |||||||
|           'offset': offset, |           'offset': offset, | ||||||
|           'take': _pageSize, |           'take': _pageSize, | ||||||
|           'vector': false, |           'vector': false, | ||||||
|  |           if (_pubName != null) 'pub': _pubName, | ||||||
|  |           if (_realm != null) 'realm': _realm, | ||||||
|  |           if (_type != null) 'type': _type, | ||||||
|  |           if (_tags != null) 'tags': _tags, | ||||||
|  |           if (_categories != null) 'categories': _categories, | ||||||
|  |           if (_shuffle) 'shuffle': true, | ||||||
|  |           if (_pinned != null) 'pinned': _pinned, | ||||||
|         }, |         }, | ||||||
|       ); |       ); | ||||||
|  |  | ||||||
| @@ -80,100 +124,269 @@ class PostSearchNotifier | |||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
| class PostSearchScreen extends ConsumerStatefulWidget { | class PostSearchScreen extends HookConsumerWidget { | ||||||
|   const PostSearchScreen({super.key}); |   const PostSearchScreen({super.key}); | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   ConsumerState<PostSearchScreen> createState() => _PostSearchScreenState(); |   Widget build(BuildContext context, WidgetRef ref) { | ||||||
| } |     final searchController = useTextEditingController(); | ||||||
|  |     final debounce = useMemoized(() => Duration(milliseconds: 500)); | ||||||
|  |     final debounceTimer = useRef<Timer?>(null); | ||||||
|  |     final showFilters = useState(false); | ||||||
|  |     final pubNameController = useTextEditingController(); | ||||||
|  |     final realmController = useTextEditingController(); | ||||||
|  |     final typeValue = useState<int?>(null); | ||||||
|  |     final selectedCategories = useState<List<String>>([]); | ||||||
|  |     final selectedTags = useState<List<String>>([]); | ||||||
|  |     final shuffleValue = useState(false); | ||||||
|  |     final pinnedValue = useState<bool?>(null); | ||||||
|  |  | ||||||
| class _PostSearchScreenState extends ConsumerState<PostSearchScreen> { |     useEffect(() { | ||||||
|   final _searchController = TextEditingController(); |       return () { | ||||||
|   final _debounce = Duration(milliseconds: 500); |         searchController.dispose(); | ||||||
|   Timer? _debounceTimer; |         pubNameController.dispose(); | ||||||
|  |         realmController.dispose(); | ||||||
|  |         debounceTimer.value?.cancel(); | ||||||
|  |       }; | ||||||
|  |     }, []); | ||||||
|  |  | ||||||
|   @override |     void onSearchChanged(String query) { | ||||||
|   void dispose() { |       if (debounceTimer.value?.isActive ?? false) debounceTimer.value!.cancel(); | ||||||
|     _searchController.dispose(); |  | ||||||
|     _debounceTimer?.cancel(); |  | ||||||
|     super.dispose(); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   void _onSearchChanged(String query) { |       debounceTimer.value = Timer(debounce, () { | ||||||
|     if (_debounceTimer?.isActive ?? false) _debounceTimer!.cancel(); |         ref.read(postSearchNotifierProvider.notifier).search(query); | ||||||
|  |       }); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     _debounceTimer = Timer(_debounce, () { |     void onSearchWithFilters(String query) { | ||||||
|       ref.read(postSearchNotifierProvider.notifier).search(query); |       if (debounceTimer.value?.isActive ?? false) debounceTimer.value!.cancel(); | ||||||
|     }); |  | ||||||
|   } |       debounceTimer.value = Timer(debounce, () { | ||||||
|  |         ref | ||||||
|  |             .read(postSearchNotifierProvider.notifier) | ||||||
|  |             .search( | ||||||
|  |               query, | ||||||
|  |               pubName: | ||||||
|  |                   pubNameController.text.isNotEmpty | ||||||
|  |                       ? pubNameController.text | ||||||
|  |                       : null, | ||||||
|  |               realm: | ||||||
|  |                   realmController.text.isNotEmpty ? realmController.text : null, | ||||||
|  |               type: typeValue.value, | ||||||
|  |               categories: | ||||||
|  |                   selectedCategories.value.isNotEmpty | ||||||
|  |                       ? selectedCategories.value | ||||||
|  |                       : null, | ||||||
|  |               tags: selectedTags.value.isNotEmpty ? selectedTags.value : null, | ||||||
|  |               shuffle: shuffleValue.value, | ||||||
|  |               pinned: pinnedValue.value, | ||||||
|  |             ); | ||||||
|  |       }); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     void toggleFilters() { | ||||||
|  |       showFilters.value = !showFilters.value; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     void applyFilters() { | ||||||
|  |       onSearchWithFilters(searchController.text); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     void clearFilters() { | ||||||
|  |       pubNameController.clear(); | ||||||
|  |       realmController.clear(); | ||||||
|  |       typeValue.value = null; | ||||||
|  |       selectedCategories.value = []; | ||||||
|  |       selectedTags.value = []; | ||||||
|  |       shuffleValue.value = false; | ||||||
|  |       pinnedValue.value = null; | ||||||
|  |       onSearchChanged(searchController.text); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     Widget buildFilterPanel() { | ||||||
|  |       return Card( | ||||||
|  |         margin: EdgeInsets.symmetric(vertical: 8, horizontal: 8), | ||||||
|  |         child: Padding( | ||||||
|  |           padding: EdgeInsets.all(16), | ||||||
|  |           child: Column( | ||||||
|  |             crossAxisAlignment: CrossAxisAlignment.start, | ||||||
|  |             children: [ | ||||||
|  |               Row( | ||||||
|  |                 mainAxisAlignment: MainAxisAlignment.spaceBetween, | ||||||
|  |                 children: [ | ||||||
|  |                   Text( | ||||||
|  |                     'filters'.tr(), | ||||||
|  |                     style: Theme.of(context).textTheme.titleMedium, | ||||||
|  |                   ).padding(left: 4), | ||||||
|  |                   Row( | ||||||
|  |                     children: [ | ||||||
|  |                       TextButton( | ||||||
|  |                         onPressed: applyFilters, | ||||||
|  |                         child: Text('apply'.tr()), | ||||||
|  |                       ), | ||||||
|  |                       TextButton( | ||||||
|  |                         onPressed: clearFilters, | ||||||
|  |                         child: Text('clear'.tr()), | ||||||
|  |                       ), | ||||||
|  |                     ], | ||||||
|  |                   ), | ||||||
|  |                 ], | ||||||
|  |               ), | ||||||
|  |               SizedBox(height: 16), | ||||||
|  |               TextField( | ||||||
|  |                 controller: pubNameController, | ||||||
|  |                 decoration: InputDecoration( | ||||||
|  |                   labelText: 'pubName'.tr(), | ||||||
|  |                   border: OutlineInputBorder(), | ||||||
|  |                 ), | ||||||
|  |                 onChanged: | ||||||
|  |                     (value) => onSearchWithFilters(searchController.text), | ||||||
|  |               ), | ||||||
|  |               SizedBox(height: 8), | ||||||
|  |               TextField( | ||||||
|  |                 controller: realmController, | ||||||
|  |                 decoration: InputDecoration( | ||||||
|  |                   labelText: 'realm'.tr(), | ||||||
|  |                   border: OutlineInputBorder(), | ||||||
|  |                 ), | ||||||
|  |                 onChanged: | ||||||
|  |                     (value) => onSearchWithFilters(searchController.text), | ||||||
|  |               ), | ||||||
|  |               SizedBox(height: 8), | ||||||
|  |               Row( | ||||||
|  |                 children: [ | ||||||
|  |                   Checkbox( | ||||||
|  |                     value: shuffleValue.value, | ||||||
|  |                     onChanged: (value) { | ||||||
|  |                       shuffleValue.value = value ?? false; | ||||||
|  |                       onSearchWithFilters(searchController.text); | ||||||
|  |                     }, | ||||||
|  |                   ), | ||||||
|  |                   Text('shuffle'.tr()), | ||||||
|  |                 ], | ||||||
|  |               ), | ||||||
|  |               Row( | ||||||
|  |                 children: [ | ||||||
|  |                   Checkbox( | ||||||
|  |                     value: pinnedValue.value ?? false, | ||||||
|  |                     onChanged: (value) { | ||||||
|  |                       pinnedValue.value = value; | ||||||
|  |                       onSearchWithFilters(searchController.text); | ||||||
|  |                     }, | ||||||
|  |                   ), | ||||||
|  |                   Text('pinned'.tr()), | ||||||
|  |                 ], | ||||||
|  |               ), | ||||||
|  |               // TODO: Add dropdown for type selection | ||||||
|  |               // TODO: Add multi-select for categories and tags | ||||||
|  |             ], | ||||||
|  |           ), | ||||||
|  |         ), | ||||||
|  |       ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|   @override |  | ||||||
|   Widget build(BuildContext context) { |  | ||||||
|     return AppScaffold( |     return AppScaffold( | ||||||
|       isNoBackground: false, |       isNoBackground: false, | ||||||
|       appBar: AppBar( |       appBar: AppBar( | ||||||
|         title: TextField( |         title: Row( | ||||||
|           controller: _searchController, |           children: [ | ||||||
|           decoration: InputDecoration( |             Expanded( | ||||||
|             hintText: 'Search posts...', |               child: TextField( | ||||||
|             border: InputBorder.none, |                 controller: searchController, | ||||||
|             hintStyle: TextStyle( |                 decoration: InputDecoration( | ||||||
|               color: Theme.of(context).appBarTheme.foregroundColor, |                   hintText: 'search'.tr(), | ||||||
|  |                   border: InputBorder.none, | ||||||
|  |                   hintStyle: TextStyle( | ||||||
|  |                     color: Theme.of(context).appBarTheme.foregroundColor, | ||||||
|  |                   ), | ||||||
|  |                 ), | ||||||
|  |                 style: TextStyle( | ||||||
|  |                   color: Theme.of(context).appBarTheme.foregroundColor, | ||||||
|  |                 ), | ||||||
|  |                 onChanged: onSearchChanged, | ||||||
|  |                 onSubmitted: (value) { | ||||||
|  |                   onSearchWithFilters(value); | ||||||
|  |                 }, | ||||||
|  |                 autofocus: true, | ||||||
|  |               ), | ||||||
|             ), |             ), | ||||||
|           ), |             IconButton( | ||||||
|           style: TextStyle( |               icon: Icon( | ||||||
|             color: Theme.of(context).appBarTheme.foregroundColor, |                 showFilters.value | ||||||
|           ), |                     ? Icons.filter_alt | ||||||
|           onChanged: _onSearchChanged, |                     : Icons.filter_alt_outlined, | ||||||
|           onSubmitted: (value) { |               ), | ||||||
|             ref.read(postSearchNotifierProvider.notifier).search(value); |               onPressed: toggleFilters, | ||||||
|           }, |               tooltip: 'toggleFilters'.tr(), | ||||||
|           autofocus: true, |             ), | ||||||
|  |           ], | ||||||
|         ), |         ), | ||||||
|       ), |       ), | ||||||
|       body: Consumer( |       body: Consumer( | ||||||
|         builder: (context, ref, child) { |         builder: (context, ref, child) { | ||||||
|           final searchState = ref.watch(postSearchNotifierProvider); |           final searchState = ref.watch(postSearchNotifierProvider); | ||||||
|  |  | ||||||
|           return searchState.when( |           return CustomScrollView( | ||||||
|             data: (data) { |             slivers: [ | ||||||
|               if (data.items.isEmpty && _searchController.text.isNotEmpty) { |               if (showFilters.value) | ||||||
|                 return const Center(child: Text('No results found')); |                 SliverToBoxAdapter( | ||||||
|               } |                   child: Center( | ||||||
|  |                     child: ConstrainedBox( | ||||||
|               return ListView.builder( |                       constraints: const BoxConstraints(maxWidth: 600), | ||||||
|                 padding: EdgeInsets.zero, |                       child: buildFilterPanel(), | ||||||
|                 itemCount: data.items.length + (data.hasMore ? 1 : 0), |                     ), | ||||||
|                 itemBuilder: (context, index) { |                   ), | ||||||
|                   if (index >= data.items.length) { |                 ), | ||||||
|                     ref |               searchState.when( | ||||||
|                         .read(postSearchNotifierProvider.notifier) |                 data: (data) { | ||||||
|                         .fetch(cursor: data.nextCursor); |                   if (data.items.isEmpty && searchController.text.isNotEmpty) { | ||||||
|                     return const Center(child: CircularProgressIndicator()); |                     return SliverFillRemaining( | ||||||
|  |                       child: Center(child: Text('noResultsFound'.tr())), | ||||||
|  |                     ); | ||||||
|                   } |                   } | ||||||
|  |  | ||||||
|                   final post = data.items[index]; |                   return SliverList( | ||||||
|                   return Center( |                     delegate: SliverChildBuilderDelegate((context, index) { | ||||||
|                     child: ConstrainedBox( |                       if (index >= data.items.length) { | ||||||
|                       constraints: BoxConstraints(maxWidth: 600), |                         ref | ||||||
|                       child: Card( |                             .read(postSearchNotifierProvider.notifier) | ||||||
|                         margin: EdgeInsets.symmetric( |                             .fetch(cursor: data.nextCursor); | ||||||
|                           horizontal: 8, |                         return Center(child: CircularProgressIndicator()); | ||||||
|                           vertical: 4, |                       } | ||||||
|  |  | ||||||
|  |                       final post = data.items[index]; | ||||||
|  |                       return Center( | ||||||
|  |                         child: ConstrainedBox( | ||||||
|  |                           constraints: BoxConstraints(maxWidth: 600), | ||||||
|  |                           child: Card( | ||||||
|  |                             margin: EdgeInsets.symmetric( | ||||||
|  |                               horizontal: 8, | ||||||
|  |                               vertical: 4, | ||||||
|  |                             ), | ||||||
|  |                             child: PostActionableItem( | ||||||
|  |                               item: post, | ||||||
|  |                               borderRadius: 8, | ||||||
|  |                             ), | ||||||
|  |                           ), | ||||||
|                         ), |                         ), | ||||||
|                         child: PostActionableItem(item: post, borderRadius: 8), |                       ); | ||||||
|                       ), |                     }, childCount: data.items.length + (data.hasMore ? 1 : 0)), | ||||||
|                     ), |  | ||||||
|                   ); |                   ); | ||||||
|                 }, |                 }, | ||||||
|               ); |                 loading: | ||||||
|             }, |                     () => SliverFillRemaining( | ||||||
|             loading: () => const Center(child: CircularProgressIndicator()), |                       child: Center(child: CircularProgressIndicator()), | ||||||
|             error: |                     ), | ||||||
|                 (error, stack) => ResponseErrorWidget( |                 error: | ||||||
|                   error: error, |                     (error, stack) => SliverFillRemaining( | ||||||
|                   onRetry: () => ref.invalidate(postSearchNotifierProvider), |                       child: ResponseErrorWidget( | ||||||
|                 ), |                         error: error, | ||||||
|  |                         onRetry: | ||||||
|  |                             () => ref.invalidate(postSearchNotifierProvider), | ||||||
|  |                       ), | ||||||
|  |                     ), | ||||||
|  |               ), | ||||||
|  |             ], | ||||||
|           ); |           ); | ||||||
|         }, |         }, | ||||||
|       ), |       ), | ||||||
|   | |||||||
| @@ -27,6 +27,224 @@ import 'package:styled_widget/styled_widget.dart'; | |||||||
|  |  | ||||||
| part 'pub_profile.g.dart'; | part 'pub_profile.g.dart'; | ||||||
|  |  | ||||||
|  | class _PublisherBasisWidget extends StatelessWidget { | ||||||
|  |   final SnPublisher data; | ||||||
|  |   final AsyncValue<SnSubscriptionStatus> subStatus; | ||||||
|  |   final ValueNotifier<bool> subscribing; | ||||||
|  |   final VoidCallback subscribe; | ||||||
|  |   final VoidCallback unsubscribe; | ||||||
|  |  | ||||||
|  |   const _PublisherBasisWidget({ | ||||||
|  |     required this.data, | ||||||
|  |     required this.subStatus, | ||||||
|  |     required this.subscribing, | ||||||
|  |     required this.subscribe, | ||||||
|  |     required this.unsubscribe, | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Widget build(BuildContext context) { | ||||||
|  |     return 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, | ||||||
|  |               borderRadius: data.type == 0 ? null : 12, | ||||||
|  |             ), | ||||||
|  |           ), | ||||||
|  |           onTap: () { | ||||||
|  |             if (data.account?.name != null) { | ||||||
|  |               Navigator.pop(context, true); | ||||||
|  |               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), | ||||||
|  |               const Gap(4), | ||||||
|  |               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); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class _PublisherBadgesWidget extends StatelessWidget { | ||||||
|  |   final SnPublisher data; | ||||||
|  |   final AsyncValue<List<SnAccountBadge>> badges; | ||||||
|  |  | ||||||
|  |   const _PublisherBadgesWidget({required this.data, required this.badges}); | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Widget build(BuildContext context) { | ||||||
|  |     return (badges.value?.isNotEmpty ?? false) | ||||||
|  |         ? Card( | ||||||
|  |           child: BadgeList( | ||||||
|  |             badges: badges.value!, | ||||||
|  |           ).padding(horizontal: 26, vertical: 20), | ||||||
|  |         ).padding(horizontal: 4) | ||||||
|  |         : const SizedBox.shrink(); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class _PublisherVerificationWidget extends StatelessWidget { | ||||||
|  |   final SnPublisher data; | ||||||
|  |  | ||||||
|  |   const _PublisherVerificationWidget({required this.data}); | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Widget build(BuildContext context) { | ||||||
|  |     return (data.verification != null) | ||||||
|  |         ? Card( | ||||||
|  |           margin: EdgeInsets.symmetric(horizontal: 8, vertical: 4), | ||||||
|  |           child: VerificationStatusCard(mark: data.verification!), | ||||||
|  |         ) | ||||||
|  |         : const SizedBox.shrink(); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class _PublisherBioWidget extends StatelessWidget { | ||||||
|  |   final SnPublisher data; | ||||||
|  |  | ||||||
|  |   const _PublisherBioWidget({required this.data}); | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Widget build(BuildContext context) { | ||||||
|  |     return 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), | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class _PublisherCategoryTabWidget extends StatelessWidget { | ||||||
|  |   final TabController categoryTabController; | ||||||
|  |  | ||||||
|  |   const _PublisherCategoryTabWidget({required this.categoryTabController}); | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Widget build(BuildContext context) { | ||||||
|  |     return 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'.tr()), | ||||||
|  |           Tab(text: 'postTypePost'.tr()), | ||||||
|  |           Tab(text: 'postArticle'.tr()), | ||||||
|  |         ], | ||||||
|  |       ), | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
| @riverpod | @riverpod | ||||||
| Future<SnPublisher> publisher(Ref ref, String uname) async { | Future<SnPublisher> publisher(Ref ref, String uname) async { | ||||||
|   final apiClient = ref.watch(apiClientProvider); |   final apiClient = ref.watch(apiClientProvider); | ||||||
| @@ -132,166 +350,6 @@ class PublisherProfileScreen extends HookConsumerWidget { | |||||||
|       offset: Offset(1.0, 1.0), |       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, |  | ||||||
|               borderRadius: data.type == 0 ? null : 12, |  | ||||||
|             ), |  | ||||||
|           ), |  | ||||||
|           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), |  | ||||||
|               const Gap(4), |  | ||||||
|               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( |     return publisher.when( | ||||||
|       data: |       data: | ||||||
|           (data) => AppScaffold( |           (data) => AppScaffold( | ||||||
| @@ -345,12 +403,16 @@ class PublisherProfileScreen extends HookConsumerWidget { | |||||||
|                           child: CustomScrollView( |                           child: CustomScrollView( | ||||||
|                             slivers: [ |                             slivers: [ | ||||||
|                               SliverGap(16), |                               SliverGap(16), | ||||||
|  |                               SliverPostList(pubName: name, pinned: true), | ||||||
|                               SliverToBoxAdapter( |                               SliverToBoxAdapter( | ||||||
|                                 child: publisherCategoryTabWidget(), |                                 child: _PublisherCategoryTabWidget( | ||||||
|  |                                   categoryTabController: categoryTabController, | ||||||
|  |                                 ), | ||||||
|                               ), |                               ), | ||||||
|                               SliverPostList( |                               SliverPostList( | ||||||
|                                 key: ValueKey(categoryTab.value), |                                 key: ValueKey(categoryTab.value), | ||||||
|                                 pubName: name, |                                 pubName: name, | ||||||
|  |                                 pinned: false, | ||||||
|                                 type: switch (categoryTab.value) { |                                 type: switch (categoryTab.value) { | ||||||
|                                   1 => 0, |                                   1 => 0, | ||||||
|                                   2 => 1, |                                   2 => 1, | ||||||
| @@ -371,10 +433,19 @@ class PublisherProfileScreen extends HookConsumerWidget { | |||||||
|                               child: Column( |                               child: Column( | ||||||
|                                 crossAxisAlignment: CrossAxisAlignment.stretch, |                                 crossAxisAlignment: CrossAxisAlignment.stretch, | ||||||
|                                 children: [ |                                 children: [ | ||||||
|                                   publisherBasisWidget(data).padding(bottom: 8), |                                   _PublisherBasisWidget( | ||||||
|                                   publisherBadgesWidget(data), |                                     data: data, | ||||||
|                                   publisherVerificationWidget(data), |                                     subStatus: subStatus, | ||||||
|                                   publisherBioWidget(data), |                                     subscribing: subscribing, | ||||||
|  |                                     subscribe: subscribe, | ||||||
|  |                                     unsubscribe: unsubscribe, | ||||||
|  |                                   ).padding(bottom: 8), | ||||||
|  |                                   _PublisherBadgesWidget( | ||||||
|  |                                     data: data, | ||||||
|  |                                     badges: badges, | ||||||
|  |                                   ), | ||||||
|  |                                   _PublisherVerificationWidget(data: data), | ||||||
|  |                                   _PublisherBioWidget(data: data), | ||||||
|                                 ], |                                 ], | ||||||
|                               ), |                               ), | ||||||
|                             ), |                             ), | ||||||
| @@ -426,17 +497,36 @@ class PublisherProfileScreen extends HookConsumerWidget { | |||||||
|                           ), |                           ), | ||||||
|                         ), |                         ), | ||||||
|                         SliverToBoxAdapter( |                         SliverToBoxAdapter( | ||||||
|                           child: publisherBasisWidget(data).padding(bottom: 8), |                           child: _PublisherBasisWidget( | ||||||
|  |                             data: data, | ||||||
|  |                             subStatus: subStatus, | ||||||
|  |                             subscribing: subscribing, | ||||||
|  |                             subscribe: subscribe, | ||||||
|  |                             unsubscribe: unsubscribe, | ||||||
|  |                           ).padding(bottom: 8), | ||||||
|                         ), |                         ), | ||||||
|                         SliverToBoxAdapter(child: publisherBadgesWidget(data)), |  | ||||||
|                         SliverToBoxAdapter( |                         SliverToBoxAdapter( | ||||||
|                           child: publisherVerificationWidget(data), |                           child: _PublisherBadgesWidget( | ||||||
|  |                             data: data, | ||||||
|  |                             badges: badges, | ||||||
|  |                           ), | ||||||
|  |                         ), | ||||||
|  |                         SliverToBoxAdapter( | ||||||
|  |                           child: _PublisherVerificationWidget(data: data), | ||||||
|  |                         ), | ||||||
|  |                         SliverToBoxAdapter( | ||||||
|  |                           child: _PublisherBioWidget(data: data), | ||||||
|  |                         ), | ||||||
|  |                         SliverPostList(pubName: name, pinned: true), | ||||||
|  |                         SliverToBoxAdapter( | ||||||
|  |                           child: _PublisherCategoryTabWidget( | ||||||
|  |                             categoryTabController: categoryTabController, | ||||||
|  |                           ), | ||||||
|                         ), |                         ), | ||||||
|                         SliverToBoxAdapter(child: publisherBioWidget(data)), |  | ||||||
|                         SliverToBoxAdapter(child: publisherCategoryTabWidget()), |  | ||||||
|                         SliverPostList( |                         SliverPostList( | ||||||
|                           key: ValueKey(categoryTab.value), |                           key: ValueKey(categoryTab.value), | ||||||
|                           pubName: name, |                           pubName: name, | ||||||
|  |                           pinned: false, | ||||||
|                           type: switch (categoryTab.value) { |                           type: switch (categoryTab.value) { | ||||||
|                             1 => 0, |                             1 => 0, | ||||||
|                             2 => 1, |                             2 => 1, | ||||||
|   | |||||||
| @@ -5,6 +5,7 @@ import 'package:flutter/material.dart'; | |||||||
| import 'package:island/models/chat.dart'; | import 'package:island/models/chat.dart'; | ||||||
| import 'package:island/services/color.dart'; | import 'package:island/services/color.dart'; | ||||||
| import 'package:island/services/responsive.dart'; | import 'package:island/services/responsive.dart'; | ||||||
|  | import 'package:island/widgets/account/account_pfc.dart'; | ||||||
| import 'package:island/widgets/account/status.dart'; | import 'package:island/widgets/account/status.dart'; | ||||||
| import 'package:island/widgets/post/post_list.dart'; | import 'package:island/widgets/post/post_list.dart'; | ||||||
| import 'package:palette_generator/palette_generator.dart'; | import 'package:palette_generator/palette_generator.dart'; | ||||||
| @@ -244,7 +245,10 @@ class RealmDetailScreen extends HookConsumerWidget { | |||||||
|                         Flexible( |                         Flexible( | ||||||
|                           flex: 3, |                           flex: 3, | ||||||
|                           child: CustomScrollView( |                           child: CustomScrollView( | ||||||
|                             slivers: [SliverPostList(realm: slug)], |                             slivers: [ | ||||||
|  |                               SliverPostList(realm: slug, pinned: true), | ||||||
|  |                               SliverPostList(realm: slug, pinned: false), | ||||||
|  |                             ], | ||||||
|                           ), |                           ), | ||||||
|                         ), |                         ), | ||||||
|                         Flexible( |                         Flexible( | ||||||
| @@ -359,7 +363,8 @@ class RealmDetailScreen extends HookConsumerWidget { | |||||||
|                         SliverToBoxAdapter( |                         SliverToBoxAdapter( | ||||||
|                           child: realmChatRoomListWidget(realm), |                           child: realmChatRoomListWidget(realm), | ||||||
|                         ), |                         ), | ||||||
|                         SliverPostList(realm: slug), |                         SliverPostList(realm: slug, pinned: true), | ||||||
|  |                         SliverPostList(realm: slug, pinned: false), | ||||||
|                       ], |                       ], | ||||||
|                     ), |                     ), | ||||||
|       ), |       ), | ||||||
| @@ -654,8 +659,11 @@ class _RealmMemberListSheet extends HookConsumerWidget { | |||||||
|                 final member = data.items[index]; |                 final member = data.items[index]; | ||||||
|                 return ListTile( |                 return ListTile( | ||||||
|                   contentPadding: EdgeInsets.only(left: 16, right: 12), |                   contentPadding: EdgeInsets.only(left: 16, right: 12), | ||||||
|                   leading: ProfilePictureWidget( |                   leading: AccountPfcGestureDetector( | ||||||
|                     fileId: member.account!.profile.picture?.id, |                     uname: member.account!.name, | ||||||
|  |                     child: ProfilePictureWidget( | ||||||
|  |                       fileId: member.account!.profile.picture?.id, | ||||||
|  |                     ), | ||||||
|                   ), |                   ), | ||||||
|                   title: Row( |                   title: Row( | ||||||
|                     spacing: 6, |                     spacing: 6, | ||||||
|   | |||||||
| @@ -23,6 +23,7 @@ import 'package:material_symbols_icons/symbols.dart'; | |||||||
| import 'package:riverpod_annotation/riverpod_annotation.dart'; | import 'package:riverpod_annotation/riverpod_annotation.dart'; | ||||||
| import 'package:styled_widget/styled_widget.dart'; | import 'package:styled_widget/styled_widget.dart'; | ||||||
| import 'package:island/widgets/realm/realm_list_tile.dart'; | import 'package:island/widgets/realm/realm_list_tile.dart'; | ||||||
|  | import 'package:island/widgets/extended_refresh_indicator.dart'; | ||||||
|  |  | ||||||
| part 'realms.g.dart'; | part 'realms.g.dart'; | ||||||
|  |  | ||||||
| @@ -90,7 +91,7 @@ class RealmListScreen extends HookConsumerWidget { | |||||||
|         }, |         }, | ||||||
|       ), |       ), | ||||||
|       floatingActionButtonLocation: TabbedFabLocation(context), |       floatingActionButtonLocation: TabbedFabLocation(context), | ||||||
|       body: RefreshIndicator( |       body: ExtendedRefreshIndicator( | ||||||
|         child: realms.when( |         child: realms.when( | ||||||
|           data: |           data: | ||||||
|               (value) => Column( |               (value) => Column( | ||||||
|   | |||||||
| @@ -219,6 +219,33 @@ class SettingsScreen extends HookConsumerWidget { | |||||||
|           }, |           }, | ||||||
|         ), |         ), | ||||||
|  |  | ||||||
|  |       // Background image enabled | ||||||
|  |       if (!kIsWeb && docBasepath.value != null) | ||||||
|  |         FutureBuilder<bool>( | ||||||
|  |           future: | ||||||
|  |               File('${docBasepath.value}/$kAppBackgroundImagePath').exists(), | ||||||
|  |           builder: (context, snapshot) { | ||||||
|  |             if (!snapshot.hasData || !snapshot.data!) { | ||||||
|  |               return const SizedBox.shrink(); | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             return ListTile( | ||||||
|  |               minLeadingWidth: 48, | ||||||
|  |               title: Text('settingsBackgroundImageEnable').tr(), | ||||||
|  |               contentPadding: const EdgeInsets.only(left: 24, right: 17), | ||||||
|  |               leading: const Icon(Symbols.image), | ||||||
|  |               trailing: Switch( | ||||||
|  |                 value: settings.showBackgroundImage, | ||||||
|  |                 onChanged: (value) { | ||||||
|  |                   ref | ||||||
|  |                       .read(appSettingsNotifierProvider.notifier) | ||||||
|  |                       .setShowBackgroundImage(value); | ||||||
|  |                 }, | ||||||
|  |               ), | ||||||
|  |             ); | ||||||
|  |           }, | ||||||
|  |         ), | ||||||
|  |  | ||||||
|       // Clear background image option |       // Clear background image option | ||||||
|       if (!kIsWeb && docBasepath.value != null) |       if (!kIsWeb && docBasepath.value != null) | ||||||
|         FutureBuilder<bool>( |         FutureBuilder<bool>( | ||||||
| @@ -423,66 +450,25 @@ class SettingsScreen extends HookConsumerWidget { | |||||||
|           }, |           }, | ||||||
|         ), |         ), | ||||||
|       ), |       ), | ||||||
|  |       ListTile( | ||||||
|  |         minLeadingWidth: 48, | ||||||
|  |         title: Text('settingsDataSavingMode').tr(), | ||||||
|  |         contentPadding: const EdgeInsets.only(left: 24, right: 17), | ||||||
|  |         leading: const Icon(Symbols.data_saver_on_rounded), | ||||||
|  |         trailing: Switch( | ||||||
|  |           value: settings.dataSavingMode, | ||||||
|  |           onChanged: (value) { | ||||||
|  |             ref | ||||||
|  |                 .read(appSettingsNotifierProvider.notifier) | ||||||
|  |                 .setDataSavingMode(value); | ||||||
|  |           }, | ||||||
|  |         ), | ||||||
|  |       ), | ||||||
|     ]; |     ]; | ||||||
|  |  | ||||||
|     // Desktop-specific settings |     // Desktop-specific settings | ||||||
|     final desktopSettings = |     // But nothing for now | ||||||
|         !isDesktop |     final desktopSettings = !isDesktop ? <Widget>[] : <Widget>[]; | ||||||
|             ? <Widget>[] |  | ||||||
|             : <Widget>[ |  | ||||||
|               ListTile( |  | ||||||
|                 minLeadingWidth: 48, |  | ||||||
|                 title: Text('settingsKeyboardShortcuts').tr(), |  | ||||||
|                 contentPadding: const EdgeInsets.only(left: 24, right: 17), |  | ||||||
|                 leading: const Icon(Symbols.keyboard), |  | ||||||
|                 onTap: () { |  | ||||||
|                   showDialog( |  | ||||||
|                     context: context, |  | ||||||
|                     builder: |  | ||||||
|                         (context) => AlertDialog( |  | ||||||
|                           title: Text('settingsKeyboardShortcuts').tr(), |  | ||||||
|                           content: SingleChildScrollView( |  | ||||||
|                             child: Column( |  | ||||||
|                               mainAxisSize: MainAxisSize.min, |  | ||||||
|                               crossAxisAlignment: CrossAxisAlignment.start, |  | ||||||
|                               children: [ |  | ||||||
|                                 _ShortcutRow( |  | ||||||
|                                   shortcut: 'Ctrl+F', |  | ||||||
|                                   description: |  | ||||||
|                                       'settingsKeyboardShortcutSearch'.tr(), |  | ||||||
|                                 ), |  | ||||||
|                                 _ShortcutRow( |  | ||||||
|                                   shortcut: 'Ctrl+,', |  | ||||||
|                                   description: |  | ||||||
|                                       'settingsKeyboardShortcutSettings'.tr(), |  | ||||||
|                                 ), |  | ||||||
|                                 _ShortcutRow( |  | ||||||
|                                   shortcut: 'Ctrl+N', |  | ||||||
|                                   description: |  | ||||||
|                                       'settingsKeyboardShortcutNewMessage'.tr(), |  | ||||||
|                                 ), |  | ||||||
|                                 _ShortcutRow( |  | ||||||
|                                   shortcut: 'Esc', |  | ||||||
|                                   description: |  | ||||||
|                                       'settingsKeyboardShortcutCloseDialog' |  | ||||||
|                                           .tr(), |  | ||||||
|                                 ), |  | ||||||
|                                 // Add more shortcuts as needed |  | ||||||
|                               ], |  | ||||||
|                             ), |  | ||||||
|                           ), |  | ||||||
|                           actions: [ |  | ||||||
|                             TextButton( |  | ||||||
|                               onPressed: () => Navigator.of(context).pop(), |  | ||||||
|                               child: Text('close').tr(), |  | ||||||
|                             ), |  | ||||||
|                           ], |  | ||||||
|                         ), |  | ||||||
|                   ); |  | ||||||
|                 }, |  | ||||||
|                 trailing: const Icon(Symbols.chevron_right), |  | ||||||
|               ), |  | ||||||
|             ]; |  | ||||||
|  |  | ||||||
|     // Create a responsive layout based on screen width |     // Create a responsive layout based on screen width | ||||||
|     Widget buildSettingsList() { |     Widget buildSettingsList() { | ||||||
| @@ -553,34 +539,7 @@ class SettingsScreen extends HookConsumerWidget { | |||||||
|  |  | ||||||
|     return AppScaffold( |     return AppScaffold( | ||||||
|       isNoBackground: false, |       isNoBackground: false, | ||||||
|       appBar: AppBar( |       appBar: AppBar(title: Text('settings').tr()), | ||||||
|         title: Text('settings').tr(), |  | ||||||
|         actions: |  | ||||||
|             isDesktop |  | ||||||
|                 ? [ |  | ||||||
|                   IconButton( |  | ||||||
|                     icon: const Icon(Symbols.help_outline), |  | ||||||
|                     onPressed: () { |  | ||||||
|                       // Show help dialog |  | ||||||
|                       showDialog( |  | ||||||
|                         context: context, |  | ||||||
|                         builder: |  | ||||||
|                             (context) => AlertDialog( |  | ||||||
|                               title: Text('settingsHelp').tr(), |  | ||||||
|                               content: Text('settingsHelpContent').tr(), |  | ||||||
|                               actions: [ |  | ||||||
|                                 TextButton( |  | ||||||
|                                   onPressed: () => Navigator.of(context).pop(), |  | ||||||
|                                   child: Text('close').tr(), |  | ||||||
|                                 ), |  | ||||||
|                               ], |  | ||||||
|                             ), |  | ||||||
|                       ); |  | ||||||
|                     }, |  | ||||||
|                   ), |  | ||||||
|                 ] |  | ||||||
|                 : null, |  | ||||||
|       ), |  | ||||||
|       body: Focus( |       body: Focus( | ||||||
|         autofocus: true, |         autofocus: true, | ||||||
|         onKeyEvent: (node, event) { |         onKeyEvent: (node, event) { | ||||||
| @@ -630,35 +589,3 @@ class _SettingsSection extends StatelessWidget { | |||||||
|     ); |     ); | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
| // Helper widget for displaying keyboard shortcuts |  | ||||||
| class _ShortcutRow extends StatelessWidget { |  | ||||||
|   final String shortcut; |  | ||||||
|   final String description; |  | ||||||
|  |  | ||||||
|   const _ShortcutRow({required this.shortcut, required this.description}); |  | ||||||
|  |  | ||||||
|   @override |  | ||||||
|   Widget build(BuildContext context) { |  | ||||||
|     return Padding( |  | ||||||
|       padding: const EdgeInsets.symmetric(vertical: 8.0), |  | ||||||
|       child: Row( |  | ||||||
|         children: [ |  | ||||||
|           Container( |  | ||||||
|             padding: EdgeInsets.symmetric(horizontal: 8, vertical: 4), |  | ||||||
|             decoration: BoxDecoration( |  | ||||||
|               color: Theme.of(context).colorScheme.surfaceVariant, |  | ||||||
|               borderRadius: BorderRadius.circular(4), |  | ||||||
|               border: Border.all( |  | ||||||
|                 color: Theme.of(context).colorScheme.outline.withOpacity(0.5), |  | ||||||
|               ), |  | ||||||
|             ), |  | ||||||
|             child: Text(shortcut, style: TextStyle(fontFamily: 'monospace')), |  | ||||||
|           ), |  | ||||||
|           SizedBox(width: 16), |  | ||||||
|           Text(description), |  | ||||||
|         ], |  | ||||||
|       ), |  | ||||||
|     ); |  | ||||||
|   } |  | ||||||
| } |  | ||||||
|   | |||||||
| @@ -77,7 +77,7 @@ class MarketplaceStickersScreen extends HookConsumerWidget { | |||||||
|         searchController.clear(); |         searchController.clear(); | ||||||
|       } |       } | ||||||
|       return null; |       return null; | ||||||
|     }, [query.value]); |     }, [query]); | ||||||
|  |  | ||||||
|     // Clean up timer on dispose |     // Clean up timer on dispose | ||||||
|     useEffect(() { |     useEffect(() { | ||||||
|   | |||||||
							
								
								
									
										62
									
								
								lib/screens/tray_manager.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										62
									
								
								lib/screens/tray_manager.dart
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,62 @@ | |||||||
|  | import 'dart:io'; | ||||||
|  |  | ||||||
|  | import 'package:bitsdojo_window/bitsdojo_window.dart'; | ||||||
|  | import 'package:flutter/foundation.dart'; | ||||||
|  | import 'package:tray_manager/tray_manager.dart'; | ||||||
|  |  | ||||||
|  | class TrayService { | ||||||
|  |   TrayService._(); | ||||||
|  |  | ||||||
|  |   static final TrayService _instance = TrayService._(); | ||||||
|  |  | ||||||
|  |   static TrayService get instance => _instance; | ||||||
|  |  | ||||||
|  |   bool _checkPlatformAvalability() { | ||||||
|  |     if (kIsWeb) return false; | ||||||
|  |     if (Platform.isAndroid || Platform.isIOS) return false; | ||||||
|  |     return true; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   Future<void> initialize(TrayListener listener) async { | ||||||
|  |     if (!_checkPlatformAvalability()) return; | ||||||
|  |  | ||||||
|  |     await trayManager.setIcon( | ||||||
|  |       Platform.isWindows | ||||||
|  |           ? 'assets/icons/icon.ico' | ||||||
|  |           : 'assets/icons/icon-outline.svg', | ||||||
|  |     ); | ||||||
|  |  | ||||||
|  |     final menu = Menu( | ||||||
|  |       items: [ | ||||||
|  |         MenuItem(key: 'show_window', label: 'Show Window'), | ||||||
|  |         MenuItem.separator(), | ||||||
|  |         MenuItem(key: 'exit_app', label: 'Exit App'), | ||||||
|  |       ], | ||||||
|  |     ); | ||||||
|  |     await trayManager.setContextMenu(menu); | ||||||
|  |  | ||||||
|  |     trayManager.addListener(listener); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   Future<void> dispose(TrayListener listener) async { | ||||||
|  |     if (!_checkPlatformAvalability()) return; | ||||||
|  |  | ||||||
|  |     trayManager.removeListener(listener); | ||||||
|  |     await trayManager.destroy(); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   void handleAction(MenuItem item) { | ||||||
|  |     switch (item.key) { | ||||||
|  |       case 'show_window': | ||||||
|  |         if (appWindow.isVisible) { | ||||||
|  |           appWindow.restore(); | ||||||
|  |         } else { | ||||||
|  |           appWindow.show(); | ||||||
|  |         } | ||||||
|  |         break; | ||||||
|  |       case 'exit_app': | ||||||
|  |         appWindow.close(); | ||||||
|  |         break; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | } | ||||||
| @@ -6,6 +6,7 @@ import 'package:dio/dio.dart'; | |||||||
| import 'package:firebase_messaging/firebase_messaging.dart'; | import 'package:firebase_messaging/firebase_messaging.dart'; | ||||||
| import 'package:flutter/foundation.dart'; | import 'package:flutter/foundation.dart'; | ||||||
| import 'package:flutter/material.dart'; | import 'package:flutter/material.dart'; | ||||||
|  | import 'package:flutter_local_notifications/flutter_local_notifications.dart'; | ||||||
| import 'package:flutter_riverpod/flutter_riverpod.dart'; | import 'package:flutter_riverpod/flutter_riverpod.dart'; | ||||||
| import 'package:go_router/go_router.dart'; | import 'package:go_router/go_router.dart'; | ||||||
| import 'package:island/main.dart'; | import 'package:island/main.dart'; | ||||||
| @@ -16,54 +17,159 @@ import 'package:island/widgets/app_notification.dart'; | |||||||
| import 'package:top_snackbar_flutter/top_snack_bar.dart'; | import 'package:top_snackbar_flutter/top_snack_bar.dart'; | ||||||
| import 'package:url_launcher/url_launcher_string.dart'; | import 'package:url_launcher/url_launcher_string.dart'; | ||||||
|  |  | ||||||
|  | final FlutterLocalNotificationsPlugin flutterLocalNotificationsPlugin = | ||||||
|  |     FlutterLocalNotificationsPlugin(); | ||||||
|  |  | ||||||
|  | AppLifecycleState _appLifecycleState = AppLifecycleState.resumed; | ||||||
|  |  | ||||||
|  | void _onAppLifecycleChanged(AppLifecycleState state) { | ||||||
|  |   _appLifecycleState = state; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | Future<void> initializeLocalNotifications() async { | ||||||
|  |   const AndroidInitializationSettings initializationSettingsAndroid = | ||||||
|  |       AndroidInitializationSettings('@mipmap/ic_launcher'); | ||||||
|  |  | ||||||
|  |   const DarwinInitializationSettings initializationSettingsIOS = | ||||||
|  |       DarwinInitializationSettings(); | ||||||
|  |  | ||||||
|  |   const DarwinInitializationSettings initializationSettingsMacOS = | ||||||
|  |       DarwinInitializationSettings(); | ||||||
|  |  | ||||||
|  |   const LinuxInitializationSettings initializationSettingsLinux = | ||||||
|  |       LinuxInitializationSettings(defaultActionName: 'Open notification'); | ||||||
|  |  | ||||||
|  |   const WindowsInitializationSettings initializationSettingsWindows = | ||||||
|  |       WindowsInitializationSettings( | ||||||
|  |         appName: 'Island', | ||||||
|  |         appUserModelId: 'dev.solsynth.solian', | ||||||
|  |         guid: 'dev.solsynth.solian', | ||||||
|  |       ); | ||||||
|  |  | ||||||
|  |   const InitializationSettings initializationSettings = InitializationSettings( | ||||||
|  |     android: initializationSettingsAndroid, | ||||||
|  |     iOS: initializationSettingsIOS, | ||||||
|  |     macOS: initializationSettingsMacOS, | ||||||
|  |     linux: initializationSettingsLinux, | ||||||
|  |     windows: initializationSettingsWindows, | ||||||
|  |   ); | ||||||
|  |  | ||||||
|  |   await flutterLocalNotificationsPlugin.initialize( | ||||||
|  |     initializationSettings, | ||||||
|  |     onDidReceiveNotificationResponse: (NotificationResponse response) async { | ||||||
|  |       final payload = response.payload; | ||||||
|  |       if (payload != null) { | ||||||
|  |         if (payload.startsWith('/')) { | ||||||
|  |           // In-app routes | ||||||
|  |           rootNavigatorKey.currentContext?.push(payload); | ||||||
|  |         } else { | ||||||
|  |           // External URLs | ||||||
|  |           launchUrlString(payload); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |   ); | ||||||
|  |  | ||||||
|  |   WidgetsBinding.instance.addObserver( | ||||||
|  |     LifecycleEventHandler(onAppLifecycleChanged: _onAppLifecycleChanged), | ||||||
|  |   ); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class LifecycleEventHandler extends WidgetsBindingObserver { | ||||||
|  |   final void Function(AppLifecycleState) onAppLifecycleChanged; | ||||||
|  |  | ||||||
|  |   LifecycleEventHandler({required this.onAppLifecycleChanged}); | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   void didChangeAppLifecycleState(AppLifecycleState state) { | ||||||
|  |     onAppLifecycleChanged(state); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
| StreamSubscription<WebSocketPacket> setupNotificationListener( | StreamSubscription<WebSocketPacket> setupNotificationListener( | ||||||
|   BuildContext context, |   BuildContext context, | ||||||
|   WidgetRef ref, |   WidgetRef ref, | ||||||
| ) { | ) { | ||||||
|   final ws = ref.watch(websocketProvider); |   final ws = ref.watch(websocketProvider); | ||||||
|   return ws.dataStream.listen((pkt) { |   return ws.dataStream.listen((pkt) async { | ||||||
|     if (pkt.type == "notifications.new") { |     if (pkt.type == "notifications.new") { | ||||||
|       final notification = SnNotification.fromJson(pkt.data!); |       final notification = SnNotification.fromJson(pkt.data!); | ||||||
|       showTopSnackBar( |       if (_appLifecycleState == AppLifecycleState.resumed) { | ||||||
|         globalOverlay.currentState!, |         // App is focused, show in-app notification | ||||||
|         Center( |         log( | ||||||
|           child: ConstrainedBox( |           '[Notification] Showing in-app notification: ${notification.title}', | ||||||
|             constraints: const BoxConstraints(maxWidth: 480), |         ); | ||||||
|             child: NotificationCard(notification: notification), |         showTopSnackBar( | ||||||
|  |           globalOverlay.currentState!, | ||||||
|  |           Center( | ||||||
|  |             child: ConstrainedBox( | ||||||
|  |               constraints: const BoxConstraints(maxWidth: 480), | ||||||
|  |               child: NotificationCard(notification: notification), | ||||||
|  |             ), | ||||||
|           ), |           ), | ||||||
|         ), |           onTap: () { | ||||||
|         onTap: () { |             if (notification.meta['action_uri'] != null) { | ||||||
|           if (notification.meta['action_uri'] != null) { |               var uri = notification.meta['action_uri'] as String; | ||||||
|             var uri = notification.meta['action_uri'] as String; |               if (uri.startsWith('/')) { | ||||||
|             if (uri.startsWith('/')) { |                 // In-app routes | ||||||
|               // In-app routes |                 rootNavigatorKey.currentContext?.push( | ||||||
|               rootNavigatorKey.currentContext?.push( |                   notification.meta['action_uri'], | ||||||
|                 notification.meta['action_uri'], |                 ); | ||||||
|               ); |               } else { | ||||||
|             } else { |                 // External URLs | ||||||
|               // External URLs |                 launchUrlString(uri); | ||||||
|               launchUrlString(uri); |               } | ||||||
|             } |             } | ||||||
|           } |           }, | ||||||
|         }, |           onDismissed: () {}, | ||||||
|         onDismissed: () {}, |           dismissType: DismissType.onSwipe, | ||||||
|         dismissType: DismissType.onSwipe, |           displayDuration: const Duration(seconds: 5), | ||||||
|         displayDuration: const Duration(seconds: 5), |           snackBarPosition: SnackBarPosition.top, | ||||||
|         snackBarPosition: SnackBarPosition.top, |           padding: EdgeInsets.only( | ||||||
|         padding: EdgeInsets.only( |             left: 16, | ||||||
|           left: 16, |             right: 16, | ||||||
|           right: 16, |             top: | ||||||
|           top: |                 (!kIsWeb && | ||||||
|               (!kIsWeb && |                         (Platform.isMacOS || | ||||||
|                       (Platform.isMacOS || |                             Platform.isWindows || | ||||||
|                           Platform.isWindows || |                             Platform.isLinux)) | ||||||
|                           Platform.isLinux)) |                     ? 28 | ||||||
|                   ? 28 |                     // ignore: use_build_context_synchronously | ||||||
|                   // ignore: use_build_context_synchronously |                     : MediaQuery.of(context).padding.top + 16, | ||||||
|                   : MediaQuery.of(context).padding.top + 16, |             bottom: 16, | ||||||
|           bottom: 16, |           ), | ||||||
|         ), |         ); | ||||||
|       ); |       } else { | ||||||
|  |         // App is in background, show system notification (only on supported platforms) | ||||||
|  |         if (!kIsWeb && !Platform.isIOS) { | ||||||
|  |           log( | ||||||
|  |             '[Notification] Showing system notification: ${notification.title}', | ||||||
|  |           ); | ||||||
|  |           const AndroidNotificationDetails androidNotificationDetails = | ||||||
|  |               AndroidNotificationDetails( | ||||||
|  |                 'channel_id', | ||||||
|  |                 'channel_name', | ||||||
|  |                 channelDescription: 'channel_description', | ||||||
|  |                 importance: Importance.max, | ||||||
|  |                 priority: Priority.high, | ||||||
|  |                 ticker: 'ticker', | ||||||
|  |               ); | ||||||
|  |           const NotificationDetails notificationDetails = NotificationDetails( | ||||||
|  |             android: androidNotificationDetails, | ||||||
|  |           ); | ||||||
|  |           await flutterLocalNotificationsPlugin.show( | ||||||
|  |             0, | ||||||
|  |             notification.title, | ||||||
|  |             notification.content, | ||||||
|  |             notificationDetails, | ||||||
|  |             payload: notification.meta['action_uri'] as String?, | ||||||
|  |           ); | ||||||
|  |         } else { | ||||||
|  |           log( | ||||||
|  |             '[Notification] Skipping system notification for unsupported platform: ${notification.title}', | ||||||
|  |           ); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|     } |     } | ||||||
|   }); |   }); | ||||||
| } | } | ||||||
| @@ -72,7 +178,7 @@ Future<void> subscribePushNotification( | |||||||
|   Dio apiClient, { |   Dio apiClient, { | ||||||
|   bool detailedErrors = false, |   bool detailedErrors = false, | ||||||
| }) async { | }) async { | ||||||
|   if (Platform.isLinux) { |   if (!kIsWeb && Platform.isLinux) { | ||||||
|     return; |     return; | ||||||
|   } |   } | ||||||
|   await FirebaseMessaging.instance.requestPermission( |   await FirebaseMessaging.instance.requestPermission( | ||||||
|   | |||||||
| @@ -1,5 +1,11 @@ | |||||||
| import 'package:flutter_udid/flutter_udid.dart'; | import 'package:flutter_udid/flutter_udid.dart'; | ||||||
|  |  | ||||||
|  | String? _cachedUdid; | ||||||
|  |  | ||||||
| Future<String> getUdid() async { | Future<String> getUdid() async { | ||||||
|   return await FlutterUdid.consistentUdid; |   if (_cachedUdid != null) { | ||||||
|  |     return _cachedUdid!; | ||||||
|  |   } | ||||||
|  |   _cachedUdid = await FlutterUdid.consistentUdid; | ||||||
|  |   return _cachedUdid!; | ||||||
| } | } | ||||||
|   | |||||||
| @@ -2,6 +2,7 @@ import 'dart:async'; | |||||||
| import 'dart:developer'; | import 'dart:developer'; | ||||||
| import 'dart:io'; | import 'dart:io'; | ||||||
|  |  | ||||||
|  | import 'package:archive/archive.dart'; | ||||||
| import 'package:dio/dio.dart'; | import 'package:dio/dio.dart'; | ||||||
| import 'package:flutter/foundation.dart'; | import 'package:flutter/foundation.dart'; | ||||||
| import 'package:flutter/material.dart'; | import 'package:flutter/material.dart'; | ||||||
| @@ -10,6 +11,9 @@ import 'package:flutter_app_update/update_model.dart'; | |||||||
| import 'package:island/widgets/content/markdown.dart'; | import 'package:island/widgets/content/markdown.dart'; | ||||||
| import 'package:material_symbols_icons/symbols.dart'; | import 'package:material_symbols_icons/symbols.dart'; | ||||||
| import 'package:package_info_plus/package_info_plus.dart'; | import 'package:package_info_plus/package_info_plus.dart'; | ||||||
|  | import 'package:path_provider/path_provider.dart'; | ||||||
|  | import 'package:path/path.dart' as path; | ||||||
|  | import 'package:process_run/process_run.dart'; | ||||||
| import 'package:collection/collection.dart'; // Added for firstWhereOrNull | import 'package:collection/collection.dart'; // Added for firstWhereOrNull | ||||||
| import 'package:styled_widget/styled_widget.dart'; | import 'package:styled_widget/styled_widget.dart'; | ||||||
| import 'package:url_launcher/url_launcher.dart'; | import 'package:url_launcher/url_launcher.dart'; | ||||||
| @@ -180,9 +184,13 @@ class UpdateService { | |||||||
|       useRootNavigator: true, |       useRootNavigator: true, | ||||||
|       builder: (ctx) { |       builder: (ctx) { | ||||||
|         String? androidUpdateUrl; |         String? androidUpdateUrl; | ||||||
|  |         String? windowsUpdateUrl; | ||||||
|         if (Platform.isAndroid) { |         if (Platform.isAndroid) { | ||||||
|           androidUpdateUrl = _getAndroidUpdateUrl(release.assets); |           androidUpdateUrl = _getAndroidUpdateUrl(release.assets); | ||||||
|         } |         } | ||||||
|  |         if (Platform.isWindows) { | ||||||
|  |           windowsUpdateUrl = _getWindowsUpdateUrl(); | ||||||
|  |         } | ||||||
|         return _UpdateSheet( |         return _UpdateSheet( | ||||||
|           release: release, |           release: release, | ||||||
|           onOpen: () async { |           onOpen: () async { | ||||||
| @@ -192,6 +200,7 @@ class UpdateService { | |||||||
|             } |             } | ||||||
|           }, |           }, | ||||||
|           androidUpdateUrl: androidUpdateUrl, |           androidUpdateUrl: androidUpdateUrl, | ||||||
|  |           windowsUpdateUrl: windowsUpdateUrl, | ||||||
|           useProxy: useProxy, // Pass the useProxy flag |           useProxy: useProxy, // Pass the useProxy flag | ||||||
|         ); |         ); | ||||||
|       }, |       }, | ||||||
| @@ -211,15 +220,270 @@ class UpdateService { | |||||||
|  |  | ||||||
|     // Prioritize arm64, then armeabi, then x86_64 |     // Prioritize arm64, then armeabi, then x86_64 | ||||||
|     if (arm64 != null) { |     if (arm64 != null) { | ||||||
|       return arm64.browserDownloadUrl; |       return 'https://fs.solsynth.dev/d/official/solian/${arm64.name}'; | ||||||
|     } else if (armeabi != null) { |     } else if (armeabi != null) { | ||||||
|       return armeabi.browserDownloadUrl; |       return 'https://fs.solsynth.dev/d/official/solian/${armeabi.name}'; | ||||||
|     } else if (x86_64 != null) { |     } else if (x86_64 != null) { | ||||||
|       return x86_64.browserDownloadUrl; |       return 'https://fs.solsynth.dev/d/official/solian/${x86_64.name}'; | ||||||
|     } |     } | ||||||
|     return null; |     return null; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   String _getWindowsUpdateUrl() { | ||||||
|  |     return 'https://fs.solsynth.dev/d/official/solian/build-output-windows-installer.zip'; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   /// Downloads the Windows installer ZIP file | ||||||
|  |   Future<String?> _downloadWindowsInstaller(String url) async { | ||||||
|  |     try { | ||||||
|  |       log('[Update] Starting Windows installer download from: $url'); | ||||||
|  |  | ||||||
|  |       final tempDir = await getTemporaryDirectory(); | ||||||
|  |       final fileName = | ||||||
|  |           'solian-installer-${DateTime.now().millisecondsSinceEpoch}.zip'; | ||||||
|  |       final filePath = path.join(tempDir.path, fileName); | ||||||
|  |  | ||||||
|  |       final response = await _dio.download( | ||||||
|  |         url, | ||||||
|  |         filePath, | ||||||
|  |         onReceiveProgress: (received, total) { | ||||||
|  |           if (total != -1) { | ||||||
|  |             log( | ||||||
|  |               '[Update] Download progress: ${(received / total * 100).toStringAsFixed(1)}%', | ||||||
|  |             ); | ||||||
|  |           } | ||||||
|  |         }, | ||||||
|  |       ); | ||||||
|  |  | ||||||
|  |       if (response.statusCode == 200) { | ||||||
|  |         log('[Update] Windows installer downloaded successfully to: $filePath'); | ||||||
|  |         return filePath; | ||||||
|  |       } else { | ||||||
|  |         log( | ||||||
|  |           '[Update] Failed to download Windows installer. Status: ${response.statusCode}', | ||||||
|  |         ); | ||||||
|  |         return null; | ||||||
|  |       } | ||||||
|  |     } catch (e) { | ||||||
|  |       log('[Update] Error downloading Windows installer: $e'); | ||||||
|  |       return null; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   /// Extracts the ZIP file to a temporary directory | ||||||
|  |   Future<String?> _extractWindowsInstaller(String zipPath) async { | ||||||
|  |     try { | ||||||
|  |       log('[Update] Extracting Windows installer from: $zipPath'); | ||||||
|  |  | ||||||
|  |       final tempDir = await getTemporaryDirectory(); | ||||||
|  |       final extractDir = path.join( | ||||||
|  |         tempDir.path, | ||||||
|  |         'solian-installer-${DateTime.now().millisecondsSinceEpoch}', | ||||||
|  |       ); | ||||||
|  |  | ||||||
|  |       final zipFile = File(zipPath); | ||||||
|  |       final bytes = await zipFile.readAsBytes(); | ||||||
|  |       final archive = ZipDecoder().decodeBytes(bytes); | ||||||
|  |  | ||||||
|  |       for (final file in archive) { | ||||||
|  |         final filename = file.name; | ||||||
|  |         if (file.isFile) { | ||||||
|  |           final data = file.content as List<int>; | ||||||
|  |           final filePath = path.join(extractDir, filename); | ||||||
|  |           await Directory(path.dirname(filePath)).create(recursive: true); | ||||||
|  |           await File(filePath).writeAsBytes(data); | ||||||
|  |         } else { | ||||||
|  |           final dirPath = path.join(extractDir, filename); | ||||||
|  |           await Directory(dirPath).create(recursive: true); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       log('[Update] Windows installer extracted successfully to: $extractDir'); | ||||||
|  |       return extractDir; | ||||||
|  |     } catch (e) { | ||||||
|  |       log('[Update] Error extracting Windows installer: $e'); | ||||||
|  |       return null; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   /// Runs the setup.exe file | ||||||
|  |   Future<bool> _runWindowsInstaller(String extractDir) async { | ||||||
|  |     try { | ||||||
|  |       log('[Update] Running Windows installer from: $extractDir'); | ||||||
|  |  | ||||||
|  |       final setupExePath = path.join(extractDir, 'setup.exe'); | ||||||
|  |  | ||||||
|  |       if (!await File(setupExePath).exists()) { | ||||||
|  |         log('[Update] setup.exe not found in extracted directory'); | ||||||
|  |         return false; | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       final shell = Shell(); | ||||||
|  |       final results = await shell.run(setupExePath); | ||||||
|  |       final result = results.first; | ||||||
|  |  | ||||||
|  |       if (result.exitCode == 0) { | ||||||
|  |         log('[Update] Windows installer completed successfully'); | ||||||
|  |         return true; | ||||||
|  |       } else { | ||||||
|  |         log( | ||||||
|  |           '[Update] Windows installer failed with exit code: ${result.exitCode}', | ||||||
|  |         ); | ||||||
|  |         log('[Update] Installer output: ${result.stdout}'); | ||||||
|  |         log('[Update] Installer errors: ${result.stderr}'); | ||||||
|  |         return false; | ||||||
|  |       } | ||||||
|  |     } catch (e) { | ||||||
|  |       log('[Update] Error running Windows installer: $e'); | ||||||
|  |       return false; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   /// Performs automatic Windows update: download, extract, and install | ||||||
|  |   Future<void> _performAutomaticWindowsUpdate( | ||||||
|  |     BuildContext context, | ||||||
|  |     String url, | ||||||
|  |   ) async { | ||||||
|  |     if (!context.mounted) return; | ||||||
|  |  | ||||||
|  |     // Show progress dialog | ||||||
|  |     showDialog( | ||||||
|  |       context: context, | ||||||
|  |       barrierDismissible: false, | ||||||
|  |       builder: | ||||||
|  |           (context) => const AlertDialog( | ||||||
|  |             title: Text('Installing Update'), | ||||||
|  |             content: Column( | ||||||
|  |               mainAxisSize: MainAxisSize.min, | ||||||
|  |               children: [ | ||||||
|  |                 CircularProgressIndicator(), | ||||||
|  |                 SizedBox(height: 16), | ||||||
|  |                 Text('Downloading installer...'), | ||||||
|  |               ], | ||||||
|  |             ), | ||||||
|  |           ), | ||||||
|  |     ); | ||||||
|  |  | ||||||
|  |     try { | ||||||
|  |       // Step 1: Download | ||||||
|  |       if (!context.mounted) return; | ||||||
|  |       Navigator.of(context).pop(); // Close progress dialog | ||||||
|  |       showDialog( | ||||||
|  |         context: context, | ||||||
|  |         barrierDismissible: false, | ||||||
|  |         builder: | ||||||
|  |             (context) => const AlertDialog( | ||||||
|  |               title: Text('Installing Update'), | ||||||
|  |               content: Column( | ||||||
|  |                 mainAxisSize: MainAxisSize.min, | ||||||
|  |                 children: [ | ||||||
|  |                   CircularProgressIndicator(), | ||||||
|  |                   SizedBox(height: 16), | ||||||
|  |                   Text('Extracting installer...'), | ||||||
|  |                 ], | ||||||
|  |               ), | ||||||
|  |             ), | ||||||
|  |       ); | ||||||
|  |  | ||||||
|  |       final zipPath = await _downloadWindowsInstaller(url); | ||||||
|  |       if (zipPath == null) { | ||||||
|  |         if (!context.mounted) return; | ||||||
|  |         Navigator.of(context).pop(); | ||||||
|  |         _showErrorDialog(context, 'Failed to download installer'); | ||||||
|  |         return; | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       // Step 2: Extract | ||||||
|  |       if (!context.mounted) return; | ||||||
|  |       Navigator.of(context).pop(); // Close progress dialog | ||||||
|  |       showDialog( | ||||||
|  |         context: context, | ||||||
|  |         barrierDismissible: false, | ||||||
|  |         builder: | ||||||
|  |             (context) => const AlertDialog( | ||||||
|  |               title: Text('Installing Update'), | ||||||
|  |               content: Column( | ||||||
|  |                 mainAxisSize: MainAxisSize.min, | ||||||
|  |                 children: [ | ||||||
|  |                   CircularProgressIndicator(), | ||||||
|  |                   SizedBox(height: 16), | ||||||
|  |                   Text('Running installer...'), | ||||||
|  |                 ], | ||||||
|  |               ), | ||||||
|  |             ), | ||||||
|  |       ); | ||||||
|  |  | ||||||
|  |       final extractDir = await _extractWindowsInstaller(zipPath); | ||||||
|  |       if (extractDir == null) { | ||||||
|  |         if (!context.mounted) return; | ||||||
|  |         Navigator.of(context).pop(); | ||||||
|  |         _showErrorDialog(context, 'Failed to extract installer'); | ||||||
|  |         return; | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       // Step 3: Run installer | ||||||
|  |       if (!context.mounted) return; | ||||||
|  |       Navigator.of(context).pop(); // Close progress dialog | ||||||
|  |  | ||||||
|  |       final success = await _runWindowsInstaller(extractDir); | ||||||
|  |       if (!context.mounted) return; | ||||||
|  |  | ||||||
|  |       if (success) { | ||||||
|  |         showDialog( | ||||||
|  |           context: context, | ||||||
|  |           builder: | ||||||
|  |               (context) => AlertDialog( | ||||||
|  |                 title: const Text('Update Complete'), | ||||||
|  |                 content: const Text( | ||||||
|  |                   'The application has been updated successfully. Please restart the application.', | ||||||
|  |                 ), | ||||||
|  |                 actions: [ | ||||||
|  |                   TextButton( | ||||||
|  |                     onPressed: () { | ||||||
|  |                       Navigator.of(context).pop(); | ||||||
|  |                       // Close the update sheet | ||||||
|  |                       Navigator.of(context).pop(); | ||||||
|  |                     }, | ||||||
|  |                     child: const Text('OK'), | ||||||
|  |                   ), | ||||||
|  |                 ], | ||||||
|  |               ), | ||||||
|  |         ); | ||||||
|  |       } else { | ||||||
|  |         _showErrorDialog(context, 'Failed to run installer'); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       // Cleanup | ||||||
|  |       try { | ||||||
|  |         await File(zipPath).delete(); | ||||||
|  |         await Directory(extractDir).delete(recursive: true); | ||||||
|  |       } catch (e) { | ||||||
|  |         log('[Update] Error cleaning up temporary files: $e'); | ||||||
|  |       } | ||||||
|  |     } catch (e) { | ||||||
|  |       if (!context.mounted) return; | ||||||
|  |       Navigator.of(context).pop(); // Close any open dialogs | ||||||
|  |       _showErrorDialog(context, 'Update failed: $e'); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   void _showErrorDialog(BuildContext context, String message) { | ||||||
|  |     showDialog( | ||||||
|  |       context: context, | ||||||
|  |       builder: | ||||||
|  |           (context) => AlertDialog( | ||||||
|  |             title: const Text('Update Failed'), | ||||||
|  |             content: Text(message), | ||||||
|  |             actions: [ | ||||||
|  |               TextButton( | ||||||
|  |                 onPressed: () => Navigator.of(context).pop(), | ||||||
|  |                 child: const Text('OK'), | ||||||
|  |               ), | ||||||
|  |             ], | ||||||
|  |           ), | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  |  | ||||||
|   /// Fetch the latest release info from GitHub. |   /// Fetch the latest release info from GitHub. | ||||||
|   /// Public so other screens (e.g., About) can manually trigger update checks. |   /// Public so other screens (e.g., About) can manually trigger update checks. | ||||||
|   Future<GithubReleaseInfo?> fetchLatestRelease() async { |   Future<GithubReleaseInfo?> fetchLatestRelease() async { | ||||||
| @@ -277,10 +541,12 @@ class _UpdateSheet extends StatefulWidget { | |||||||
|     required this.release, |     required this.release, | ||||||
|     required this.onOpen, |     required this.onOpen, | ||||||
|     this.androidUpdateUrl, |     this.androidUpdateUrl, | ||||||
|  |     this.windowsUpdateUrl, | ||||||
|     this.useProxy = false, |     this.useProxy = false, | ||||||
|   }); |   }); | ||||||
|  |  | ||||||
|   final String? androidUpdateUrl; |   final String? androidUpdateUrl; | ||||||
|  |   final String? windowsUpdateUrl; | ||||||
|   final bool useProxy; |   final bool useProxy; | ||||||
|   final GithubReleaseInfo release; |   final GithubReleaseInfo release; | ||||||
|   final VoidCallback onOpen; |   final VoidCallback onOpen; | ||||||
| @@ -299,8 +565,11 @@ class _UpdateSheetState extends State<_UpdateSheet> { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   Future<void> _installUpdate(String url) async { |   Future<void> _installUpdate(String url) async { | ||||||
|     final downloadUrl = |     String downloadUrl = url; | ||||||
|         _useProxy ? 'https://ghfast.top/${Uri.encodeComponent(url)}' : url; |     if (_useProxy) { | ||||||
|  |       final fileName = url.split('/').last; | ||||||
|  |       downloadUrl = 'https://fs.solsynth.dev/d/rainyun02/solian/$fileName'; | ||||||
|  |     } | ||||||
|  |  | ||||||
|     UpdateModel model = UpdateModel( |     UpdateModel model = UpdateModel( | ||||||
|       downloadUrl, |       downloadUrl, | ||||||
| @@ -350,7 +619,7 @@ class _UpdateSheetState extends State<_UpdateSheet> { | |||||||
|             ), |             ), | ||||||
|             if (!kIsWeb && Platform.isAndroid) |             if (!kIsWeb && Platform.isAndroid) | ||||||
|               SwitchListTile( |               SwitchListTile( | ||||||
|                 title: const Text('Use GitHub Proxy for Download'), |                 title: const Text('Use secondary source for download'), | ||||||
|                 value: _useProxy, |                 value: _useProxy, | ||||||
|                 onChanged: (value) { |                 onChanged: (value) { | ||||||
|                   setState(() { |                   setState(() { | ||||||
| @@ -376,6 +645,25 @@ class _UpdateSheetState extends State<_UpdateSheet> { | |||||||
|                           label: const Text('Install update'), |                           label: const Text('Install update'), | ||||||
|                         ), |                         ), | ||||||
|                       ), |                       ), | ||||||
|  |                     if (!kIsWeb && | ||||||
|  |                         Platform.isWindows && | ||||||
|  |                         widget.windowsUpdateUrl != null) | ||||||
|  |                       Expanded( | ||||||
|  |                         child: FilledButton.icon( | ||||||
|  |                           onPressed: () { | ||||||
|  |                             // Access the UpdateService instance to call the automatic update method | ||||||
|  |                             final updateService = UpdateService( | ||||||
|  |                               useProxy: widget.useProxy, | ||||||
|  |                             ); | ||||||
|  |                             updateService._performAutomaticWindowsUpdate( | ||||||
|  |                               context, | ||||||
|  |                               widget.windowsUpdateUrl!, | ||||||
|  |                             ); | ||||||
|  |                           }, | ||||||
|  |                           icon: const Icon(Symbols.update), | ||||||
|  |                           label: const Text('Install update'), | ||||||
|  |                         ), | ||||||
|  |                       ), | ||||||
|                     Expanded( |                     Expanded( | ||||||
|                       child: FilledButton.icon( |                       child: FilledButton.icon( | ||||||
|                         onPressed: widget.onOpen, |                         onPressed: widget.onOpen, | ||||||
|   | |||||||
							
								
								
									
										9
									
								
								lib/utils/format.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								lib/utils/format.dart
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,9 @@ | |||||||
|  | String formatFileSize(int bytes) { | ||||||
|  |   if (bytes <= 0) return '0 B'; | ||||||
|  |   if (bytes < 1024) return '$bytes B'; | ||||||
|  |   if (bytes < 1024 * 1024) return '${(bytes / 1024).toStringAsFixed(2)} KB'; | ||||||
|  |   if (bytes < 1024 * 1024 * 1024) { | ||||||
|  |     return '${(bytes / (1024 * 1024)).toStringAsFixed(2)} MB'; | ||||||
|  |   } | ||||||
|  |   return '${(bytes / (1024 * 1024 * 1024)).toStringAsFixed(2)} GB'; | ||||||
|  | } | ||||||
							
								
								
									
										62
									
								
								lib/utils/share_utils.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										62
									
								
								lib/utils/share_utils.dart
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,62 @@ | |||||||
|  | import 'dart:io'; | ||||||
|  |  | ||||||
|  | import 'package:flutter/foundation.dart'; | ||||||
|  | import 'package:flutter/material.dart'; | ||||||
|  | import 'package:hooks_riverpod/hooks_riverpod.dart'; | ||||||
|  | import 'package:island/models/post.dart'; | ||||||
|  | import 'package:island/pods/config.dart'; | ||||||
|  | import 'package:island/widgets/alert.dart'; | ||||||
|  | import 'package:island/widgets/post/post_item_screenshot.dart'; | ||||||
|  | import 'package:path_provider/path_provider.dart' show getTemporaryDirectory; | ||||||
|  | import 'package:screenshot/screenshot.dart'; | ||||||
|  | import 'package:share_plus/share_plus.dart'; | ||||||
|  |  | ||||||
|  | /// Shares a post as a screenshot image | ||||||
|  | Future<void> sharePostAsScreenshot( | ||||||
|  |   BuildContext context, | ||||||
|  |   WidgetRef ref, | ||||||
|  |   SnPost post, | ||||||
|  | ) async { | ||||||
|  |   if (kIsWeb) return; | ||||||
|  |  | ||||||
|  |   final screenshotController = ScreenshotController(); | ||||||
|  |  | ||||||
|  |   showLoadingModal(context); | ||||||
|  |   await screenshotController | ||||||
|  |       .captureFromWidget( | ||||||
|  |         ProviderScope( | ||||||
|  |           overrides: [ | ||||||
|  |             sharedPreferencesProvider.overrideWithValue( | ||||||
|  |               ref.watch(sharedPreferencesProvider), | ||||||
|  |             ), | ||||||
|  |           ], | ||||||
|  |           child: Directionality( | ||||||
|  |             textDirection: TextDirection.ltr, | ||||||
|  |             child: SizedBox( | ||||||
|  |               width: 520, | ||||||
|  |               child: PostItemScreenshot(item: post, isFullPost: true), | ||||||
|  |             ), | ||||||
|  |           ), | ||||||
|  |         ), | ||||||
|  |         context: context, | ||||||
|  |         pixelRatio: MediaQuery.of(context).devicePixelRatio, | ||||||
|  |         delay: const Duration(seconds: 1), | ||||||
|  |       ) | ||||||
|  |       .then((Uint8List? image) async { | ||||||
|  |         if (image == null) return; | ||||||
|  |         final directory = await getTemporaryDirectory(); | ||||||
|  |         final imagePath = await File('${directory.path}/image.png').create(); | ||||||
|  |         await imagePath.writeAsBytes(image); | ||||||
|  |  | ||||||
|  |         if (!context.mounted) return; | ||||||
|  |         hideLoadingModal(context); | ||||||
|  |         final box = context.findRenderObject() as RenderBox?; | ||||||
|  |         await Share.shareXFiles([ | ||||||
|  |           XFile(imagePath.path), | ||||||
|  |         ], sharePositionOrigin: box!.localToGlobal(Offset.zero) & box.size); | ||||||
|  |       }) | ||||||
|  |       .catchError((err) { | ||||||
|  |         if (context.mounted) hideLoadingModal(context); | ||||||
|  |         showErrorAlert(err); | ||||||
|  |       }); | ||||||
|  | } | ||||||
| @@ -12,6 +12,7 @@ import 'package:island/widgets/content/sheet.dart'; | |||||||
| import 'package:island/widgets/response.dart'; | import 'package:island/widgets/response.dart'; | ||||||
| import 'package:riverpod_annotation/riverpod_annotation.dart'; | import 'package:riverpod_annotation/riverpod_annotation.dart'; | ||||||
| import 'package:styled_widget/styled_widget.dart'; | import 'package:styled_widget/styled_widget.dart'; | ||||||
|  | import 'package:island/widgets/extended_refresh_indicator.dart'; | ||||||
|  |  | ||||||
| part 'account_devices.g.dart'; | part 'account_devices.g.dart'; | ||||||
|  |  | ||||||
| @@ -177,7 +178,7 @@ class AccountSessionSheet extends HookConsumerWidget { | |||||||
|       titleText: 'authSessions'.tr(), |       titleText: 'authSessions'.tr(), | ||||||
|       child: authDevices.when( |       child: authDevices.when( | ||||||
|         data: |         data: | ||||||
|             (data) => RefreshIndicator( |             (data) => ExtendedRefreshIndicator( | ||||||
|               onRefresh: |               onRefresh: | ||||||
|                   () => Future.sync(() => ref.invalidate(authDevicesProvider)), |                   () => Future.sync(() => ref.invalidate(authDevicesProvider)), | ||||||
|               child: ListView.builder( |               child: ListView.builder( | ||||||
|   | |||||||
| @@ -37,11 +37,28 @@ class AccountName extends StatelessWidget { | |||||||
|       mainAxisSize: MainAxisSize.min, |       mainAxisSize: MainAxisSize.min, | ||||||
|       spacing: 4, |       spacing: 4, | ||||||
|       children: [ |       children: [ | ||||||
|         Flexible(child: Text(account.nick, style: nameStyle)), |         Flexible( | ||||||
|  |           child: Text( | ||||||
|  |             account.nick, | ||||||
|  |             style: nameStyle, | ||||||
|  |             maxLines: 1, | ||||||
|  |             overflow: TextOverflow.ellipsis, | ||||||
|  |           ), | ||||||
|  |         ), | ||||||
|         if (account.perkSubscription != null) |         if (account.perkSubscription != null) | ||||||
|           StellarMembershipMark(membership: account.perkSubscription!), |           StellarMembershipMark(membership: account.perkSubscription!), | ||||||
|         if (account.profile.verification != null) |         if (account.profile.verification != null) | ||||||
|           VerificationMark(mark: account.profile.verification!), |           VerificationMark(mark: account.profile.verification!), | ||||||
|  |         if (account.automatedId != null) | ||||||
|  |           Tooltip( | ||||||
|  |             message: 'accountAutomated'.tr(), | ||||||
|  |             child: Icon( | ||||||
|  |               Symbols.smart_toy, | ||||||
|  |               size: 16, | ||||||
|  |               color: nameStyle.color, | ||||||
|  |               fill: 1, | ||||||
|  |             ), | ||||||
|  |           ), | ||||||
|       ], |       ], | ||||||
|     ); |     ); | ||||||
|   } |   } | ||||||
| @@ -141,7 +158,7 @@ class VerificationStatusCard extends StatelessWidget { | |||||||
|   @override |   @override | ||||||
|   Widget build(BuildContext context) { |   Widget build(BuildContext context) { | ||||||
|     return Column( |     return Column( | ||||||
|       crossAxisAlignment: CrossAxisAlignment.start, |       crossAxisAlignment: CrossAxisAlignment.stretch, | ||||||
|       children: [ |       children: [ | ||||||
|         Icon( |         Icon( | ||||||
|           mark.type == 4 |           mark.type == 4 | ||||||
| @@ -152,7 +169,7 @@ class VerificationStatusCard extends StatelessWidget { | |||||||
|           size: 32, |           size: 32, | ||||||
|           color: kVerificationMarkColors[mark.type], |           color: kVerificationMarkColors[mark.type], | ||||||
|           fill: 1, |           fill: 1, | ||||||
|         ), |         ).alignment(Alignment.centerLeft), | ||||||
|         const Gap(8), |         const Gap(8), | ||||||
|         Text(mark.title ?? 'No title').bold(), |         Text(mark.title ?? 'No title').bold(), | ||||||
|         Text(mark.description ?? 'descriptionNone'.tr()), |         Text(mark.description ?? 'descriptionNone'.tr()), | ||||||
|   | |||||||
| @@ -1,6 +1,7 @@ | |||||||
| import 'dart:math' as math; | import 'dart:math' as math; | ||||||
|  |  | ||||||
| import 'package:easy_localization/easy_localization.dart'; | import 'package:easy_localization/easy_localization.dart'; | ||||||
|  | import 'package:flutter/foundation.dart'; | ||||||
| import 'package:flutter/material.dart'; | import 'package:flutter/material.dart'; | ||||||
| import 'package:go_router/go_router.dart'; | import 'package:go_router/go_router.dart'; | ||||||
| import 'package:flutter_popup_card/flutter_popup_card.dart'; | import 'package:flutter_popup_card/flutter_popup_card.dart'; | ||||||
| @@ -74,27 +75,75 @@ class AccountProfileCard extends HookConsumerWidget { | |||||||
|                         uname: data.name, |                         uname: data.name, | ||||||
|                         padding: EdgeInsets.zero, |                         padding: EdgeInsets.zero, | ||||||
|                       ), |                       ), | ||||||
|                       if (data.profile.timeZone.isNotEmpty) |                       Tooltip( | ||||||
|                         Row( |                         message: 'creditsStatus'.tr(), | ||||||
|  |                         child: Row( | ||||||
|                           spacing: 6, |                           spacing: 6, | ||||||
|                           children: [ |                           children: [ | ||||||
|                             Icon( |                             Icon( | ||||||
|                               Symbols.alarm, |                               Symbols.star, | ||||||
|                               size: 17, |                               size: 17, | ||||||
|                               fill: 1, |                               fill: 1, | ||||||
|                             ).padding(right: 2), |                             ).padding(right: 2), | ||||||
|                             Text( |                             Text( | ||||||
|                               getTzInfo( |                               '${data.profile.socialCredits.toStringAsFixed(2)} pts', | ||||||
|                                 data.profile.timeZone, |  | ||||||
|                               ).$2.formatCustomGlobal('HH:mm'), |  | ||||||
|                             ).fontSize(12), |  | ||||||
|                             Text( |  | ||||||
|                               getTzInfo( |  | ||||||
|                                 data.profile.timeZone, |  | ||||||
|                               ).$1.formatOffsetLocal(), |  | ||||||
|                             ).fontSize(12), |                             ).fontSize(12), | ||||||
|  |                             switch (data.profile.socialCreditsLevel) { | ||||||
|  |                               -1 => Text('socialCreditsLevelPoor').tr(), | ||||||
|  |                               0 => Text('socialCreditsLevelNormal').tr(), | ||||||
|  |                               1 => Text('socialCreditsLevelGood').tr(), | ||||||
|  |                               2 => Text('socialCreditsLevelExcellent').tr(), | ||||||
|  |                               _ => Text('unknown').tr(), | ||||||
|  |                             }.fontSize(12), | ||||||
|                           ], |                           ], | ||||||
|                         ).padding(top: 2), |                         ), | ||||||
|  |                       ), | ||||||
|  |                       if (data.automatedId != null) | ||||||
|  |                         Row( | ||||||
|  |                           spacing: 6, | ||||||
|  |                           children: [ | ||||||
|  |                             Icon( | ||||||
|  |                               Symbols.smart_toy, | ||||||
|  |                               size: 17, | ||||||
|  |                               fill: 1, | ||||||
|  |                             ).padding(right: 2), | ||||||
|  |                             Text('accountAutomated').tr().fontSize(12), | ||||||
|  |                           ], | ||||||
|  |                         ), | ||||||
|  |                       if (data.profile.timeZone.isNotEmpty && !kIsWeb) | ||||||
|  |                         () { | ||||||
|  |                           try { | ||||||
|  |                             final tzInfo = getTzInfo(data.profile.timeZone); | ||||||
|  |                             return Row( | ||||||
|  |                               spacing: 6, | ||||||
|  |                               children: [ | ||||||
|  |                                 Icon( | ||||||
|  |                                   Symbols.alarm, | ||||||
|  |                                   size: 17, | ||||||
|  |                                   fill: 1, | ||||||
|  |                                 ).padding(right: 2), | ||||||
|  |                                 Text( | ||||||
|  |                                   tzInfo.$2.formatCustomGlobal('HH:mm'), | ||||||
|  |                                 ).fontSize(12), | ||||||
|  |                                 Text( | ||||||
|  |                                   tzInfo.$1.formatOffsetLocal(), | ||||||
|  |                                 ).fontSize(12), | ||||||
|  |                               ], | ||||||
|  |                             ).padding(top: 2); | ||||||
|  |                           } catch (e) { | ||||||
|  |                             return Row( | ||||||
|  |                               spacing: 6, | ||||||
|  |                               children: [ | ||||||
|  |                                 Icon( | ||||||
|  |                                   Symbols.alarm, | ||||||
|  |                                   size: 17, | ||||||
|  |                                   fill: 1, | ||||||
|  |                                 ).padding(right: 2), | ||||||
|  |                                 Text('timezoneNotFound'.tr()).fontSize(12), | ||||||
|  |                               ], | ||||||
|  |                             ).padding(top: 2); | ||||||
|  |                           } | ||||||
|  |                         }(), | ||||||
|                       if (data.badges.isNotEmpty) |                       if (data.badges.isNotEmpty) | ||||||
|                         BadgeList(badges: data.badges).padding(top: 12), |                         BadgeList(badges: data.badges).padding(top: 12), | ||||||
|                       LevelingProgressCard( |                       LevelingProgressCard( | ||||||
|   | |||||||
| @@ -2,7 +2,9 @@ import 'package:easy_localization/easy_localization.dart'; | |||||||
| import 'package:flutter/material.dart'; | import 'package:flutter/material.dart'; | ||||||
| import 'package:flutter_hooks/flutter_hooks.dart'; | import 'package:flutter_hooks/flutter_hooks.dart'; | ||||||
| import 'package:hooks_riverpod/hooks_riverpod.dart'; | import 'package:hooks_riverpod/hooks_riverpod.dart'; | ||||||
|  | import 'package:island/models/account.dart'; | ||||||
| import 'package:island/models/activity.dart'; | import 'package:island/models/activity.dart'; | ||||||
|  | import 'package:material_symbols_icons/symbols.dart'; | ||||||
| import 'package:styled_widget/styled_widget.dart'; | import 'package:styled_widget/styled_widget.dart'; | ||||||
| import 'package:island/widgets/account/event_details_widget.dart'; | import 'package:island/widgets/account/event_details_widget.dart'; | ||||||
| import 'package:table_calendar/table_calendar.dart'; | import 'package:table_calendar/table_calendar.dart'; | ||||||
| @@ -87,24 +89,56 @@ class EventCalendarWidget extends HookConsumerWidget { | |||||||
|               return Center(child: Text(text)); |               return Center(child: Text(text)); | ||||||
|             }, |             }, | ||||||
|             markerBuilder: (context, day, events) { |             markerBuilder: (context, day, events) { | ||||||
|               var checkInResult = |               final checkInResult = | ||||||
|                   events.whereType<SnCheckInResult>().firstOrNull; |                   events.whereType<SnCheckInResult>().firstOrNull; | ||||||
|  |               final statuses = events.whereType<SnAccountStatus>().toList(); | ||||||
|  |  | ||||||
|  |               final textColor = | ||||||
|  |                   isSameDay(selectedDay.value, day) | ||||||
|  |                       ? Colors.white | ||||||
|  |                       : isSameDay(DateTime.now(), day) | ||||||
|  |                       ? Colors.white | ||||||
|  |                       : Theme.of(context).colorScheme.onSurface; | ||||||
|  |  | ||||||
|  |               final shadow = | ||||||
|  |                   isSameDay(selectedDay.value, day) || | ||||||
|  |                           isSameDay(DateTime.now(), day) | ||||||
|  |                       ? [ | ||||||
|  |                         Shadow( | ||||||
|  |                           color: Colors.black.withOpacity(0.5), | ||||||
|  |                           offset: const Offset(0, 1), | ||||||
|  |                           blurRadius: 4, | ||||||
|  |                         ), | ||||||
|  |                       ] | ||||||
|  |                       : null; | ||||||
|  |  | ||||||
|               if (checkInResult != null) { |               if (checkInResult != null) { | ||||||
|                 return Positioned( |                 return Positioned( | ||||||
|                   top: 32, |                   top: 32, | ||||||
|                   child: Text( |                   child: Row( | ||||||
|                     'checkInResultT${checkInResult.level}'.tr(), |                     spacing: 2, | ||||||
|                     style: TextStyle( |                     children: [ | ||||||
|                       fontSize: 9, |                       Text( | ||||||
|                       color: |                         'checkInResultT${checkInResult.level}'.tr(), | ||||||
|                           isSameDay(selectedDay.value, day) |                         style: TextStyle( | ||||||
|                               ? Theme.of(context).colorScheme.onPrimaryContainer |                           fontSize: 9, | ||||||
|                               : isSameDay(DateTime.now(), day) |                           color: textColor, | ||||||
|                               ? Theme.of( |                           shadows: shadow, | ||||||
|                                 context, |                         ), | ||||||
|                               ).colorScheme.onSecondaryContainer |                       ), | ||||||
|                               : Theme.of(context).colorScheme.onSurface, |                       if (statuses.isNotEmpty) ...[ | ||||||
|                     ), |                         Icon( | ||||||
|  |                           switch (statuses.first.attitude) { | ||||||
|  |                             0 => Symbols.sentiment_satisfied, | ||||||
|  |                             2 => Symbols.sentiment_dissatisfied, | ||||||
|  |                             _ => Symbols.sentiment_neutral, | ||||||
|  |                           }, | ||||||
|  |                           size: 12, | ||||||
|  |                           color: textColor, | ||||||
|  |                           shadows: shadow, | ||||||
|  |                         ), | ||||||
|  |                       ], | ||||||
|  |                     ], | ||||||
|                   ), |                   ), | ||||||
|                 ); |                 ); | ||||||
|               } |               } | ||||||
|   | |||||||
| @@ -2,6 +2,7 @@ import 'package:easy_localization/easy_localization.dart'; | |||||||
| import 'package:flutter/material.dart'; | import 'package:flutter/material.dart'; | ||||||
| import 'package:gap/gap.dart'; | import 'package:gap/gap.dart'; | ||||||
| import 'package:island/models/activity.dart'; | import 'package:island/models/activity.dart'; | ||||||
|  | import 'package:island/services/time.dart'; | ||||||
| import 'package:material_symbols_icons/symbols.dart'; | import 'package:material_symbols_icons/symbols.dart'; | ||||||
| import 'package:styled_widget/styled_widget.dart'; | import 'package:styled_widget/styled_widget.dart'; | ||||||
|  |  | ||||||
| @@ -45,6 +46,10 @@ class EventDetailsWidget extends StatelessWidget { | |||||||
|                       size: 12, |                       size: 12, | ||||||
|                       fill: 1, |                       fill: 1, | ||||||
|                     ).padding(top: 4, right: 4), |                     ).padding(top: 4, right: 4), | ||||||
|  |                     Icon( | ||||||
|  |                       tip.isPositive ? Symbols.thumb_up : Symbols.thumb_down, | ||||||
|  |                       size: 14, | ||||||
|  |                     ).padding(top: 2.5), | ||||||
|                     Expanded( |                     Expanded( | ||||||
|                       child: Column( |                       child: Column( | ||||||
|                         crossAxisAlignment: CrossAxisAlignment.start, |                         crossAxisAlignment: CrossAxisAlignment.start, | ||||||
| @@ -53,6 +58,33 @@ class EventDetailsWidget extends StatelessWidget { | |||||||
|                     ), |                     ), | ||||||
|                   ], |                   ], | ||||||
|                 ).padding(top: 8), |                 ).padding(top: 8), | ||||||
|  |               if (event!.statuses.isNotEmpty) ...[ | ||||||
|  |                 const Gap(16), | ||||||
|  |                 Text('statusLabel').tr().fontSize(16).bold(), | ||||||
|  |               ], | ||||||
|  |               for (final status in event!.statuses) ...[ | ||||||
|  |                 Row( | ||||||
|  |                   spacing: 8, | ||||||
|  |                   children: [ | ||||||
|  |                     Icon(switch (status.attitude) { | ||||||
|  |                       0 => Symbols.sentiment_satisfied, | ||||||
|  |                       2 => Symbols.sentiment_dissatisfied, | ||||||
|  |                       _ => Symbols.sentiment_neutral, | ||||||
|  |                     }), | ||||||
|  |                     Expanded( | ||||||
|  |                       child: Column( | ||||||
|  |                         crossAxisAlignment: CrossAxisAlignment.start, | ||||||
|  |                         children: [ | ||||||
|  |                           Text(status.label), | ||||||
|  |                           Text( | ||||||
|  |                             '${status.createdAt.formatSystem()} - ${status.clearedAt?.formatSystem() ?? 'present'.tr()}', | ||||||
|  |                           ).fontSize(11).opacity(0.8), | ||||||
|  |                         ], | ||||||
|  |                       ), | ||||||
|  |                     ), | ||||||
|  |                   ], | ||||||
|  |                 ).padding(vertical: 8), | ||||||
|  |               ], | ||||||
|             ], |             ], | ||||||
|           ), |           ), | ||||||
|         if (event?.checkInResult == null && (event?.statuses.isEmpty ?? true)) |         if (event?.checkInResult == null && (event?.statuses.isEmpty ?? true)) | ||||||
|   | |||||||
| @@ -60,7 +60,9 @@ class AccountStatusCreationWidget extends HookConsumerWidget { | |||||||
|                         spacing: 4, |                         spacing: 4, | ||||||
|                         children: [ |                         children: [ | ||||||
|                           Icon(Symbols.keyboard_arrow_up), |                           Icon(Symbols.keyboard_arrow_up), | ||||||
|                           Text('statusCreateHint').tr(), |                           Expanded( | ||||||
|  |                             child: Text('statusCreateHint', maxLines: 1).tr(), | ||||||
|  |                           ), | ||||||
|                         ], |                         ], | ||||||
|                       ), |                       ), | ||||||
|                     ).opacity(0.85), |                     ).opacity(0.85), | ||||||
|   | |||||||
| @@ -17,8 +17,8 @@ class NotificationCard extends HookConsumerWidget { | |||||||
|     return Card( |     return Card( | ||||||
|       elevation: 4, |       elevation: 4, | ||||||
|       margin: const EdgeInsets.only(bottom: 8), |       margin: const EdgeInsets.only(bottom: 8), | ||||||
|       shape: RoundedRectangleBorder( |       shape: const RoundedRectangleBorder( | ||||||
|         borderRadius: BorderRadius.vertical(bottom: Radius.circular(8)), |         borderRadius: BorderRadius.all(Radius.circular(8)), | ||||||
|       ), |       ), | ||||||
|       child: Column( |       child: Column( | ||||||
|         crossAxisAlignment: CrossAxisAlignment.start, |         crossAxisAlignment: CrossAxisAlignment.start, | ||||||
|   | |||||||
| @@ -1,8 +1,10 @@ | |||||||
| import 'dart:io'; | import 'dart:io'; | ||||||
|  | import 'dart:ui'; | ||||||
| import 'package:bitsdojo_window/bitsdojo_window.dart'; | import 'package:bitsdojo_window/bitsdojo_window.dart'; | ||||||
| import 'package:easy_localization/easy_localization.dart'; | import 'package:easy_localization/easy_localization.dart'; | ||||||
| import 'package:flutter/foundation.dart'; | import 'package:flutter/foundation.dart'; | ||||||
| import 'package:flutter/material.dart'; | import 'package:flutter/material.dart'; | ||||||
|  | import 'package:flutter/services.dart'; | ||||||
| import 'package:go_router/go_router.dart'; | import 'package:go_router/go_router.dart'; | ||||||
| import 'package:flutter_hooks/flutter_hooks.dart'; | import 'package:flutter_hooks/flutter_hooks.dart'; | ||||||
| import 'package:hooks_riverpod/hooks_riverpod.dart'; | import 'package:hooks_riverpod/hooks_riverpod.dart'; | ||||||
| @@ -14,6 +16,15 @@ import 'package:material_symbols_icons/material_symbols_icons.dart'; | |||||||
| import 'package:path_provider/path_provider.dart'; | import 'package:path_provider/path_provider.dart'; | ||||||
| import 'package:styled_widget/styled_widget.dart'; | import 'package:styled_widget/styled_widget.dart'; | ||||||
|  |  | ||||||
|  | class AppScrollBehavior extends MaterialScrollBehavior { | ||||||
|  |   @override | ||||||
|  |   Set<PointerDeviceKind> get dragDevices => { | ||||||
|  |     PointerDeviceKind.touch, // default | ||||||
|  |     PointerDeviceKind.trackpad, // default | ||||||
|  |     PointerDeviceKind.mouse, // add mouse dragging | ||||||
|  |   }; | ||||||
|  | } | ||||||
|  |  | ||||||
| class WindowScaffold extends HookConsumerWidget { | class WindowScaffold extends HookConsumerWidget { | ||||||
|   final Widget child; |   final Widget child; | ||||||
|   const WindowScaffold({super.key, required this.child}); |   const WindowScaffold({super.key, required this.child}); | ||||||
| @@ -153,7 +164,7 @@ class _WindowSizeObserver extends WidgetsBindingObserver { | |||||||
|  |  | ||||||
| final rootScaffoldKey = GlobalKey<ScaffoldState>(); | final rootScaffoldKey = GlobalKey<ScaffoldState>(); | ||||||
|  |  | ||||||
| class AppScaffold extends StatelessWidget { | class AppScaffold extends HookConsumerWidget { | ||||||
|   final Widget? body; |   final Widget? body; | ||||||
|   final PreferredSizeWidget? bottomNavigationBar; |   final PreferredSizeWidget? bottomNavigationBar; | ||||||
|   final PreferredSizeWidget? bottomSheet; |   final PreferredSizeWidget? bottomSheet; | ||||||
| @@ -186,7 +197,14 @@ class AppScaffold extends StatelessWidget { | |||||||
|   }); |   }); | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   Widget build(BuildContext context) { |   Widget build(BuildContext context, WidgetRef ref) { | ||||||
|  |     final focusNode = useFocusNode(); | ||||||
|  |  | ||||||
|  |     useEffect(() { | ||||||
|  |       focusNode.requestFocus(); | ||||||
|  |       return null; | ||||||
|  |     }, []); | ||||||
|  |  | ||||||
|     final appBarHeight = appBar?.preferredSize.height ?? 0; |     final appBarHeight = appBar?.preferredSize.height ?? 0; | ||||||
|     final safeTop = MediaQuery.of(context).padding.top; |     final safeTop = MediaQuery.of(context).padding.top; | ||||||
|  |  | ||||||
| @@ -201,29 +219,59 @@ class AppScaffold extends StatelessWidget { | |||||||
|       ], |       ], | ||||||
|     ); |     ); | ||||||
|  |  | ||||||
|     return Scaffold( |     return Shortcuts( | ||||||
|       extendBody: extendBody ?? true, |       shortcuts: <LogicalKeySet, Intent>{ | ||||||
|       extendBodyBehindAppBar: true, |         LogicalKeySet(LogicalKeyboardKey.escape): const PopIntent(), | ||||||
|       backgroundColor: |       }, | ||||||
|           noBackground |       child: Actions( | ||||||
|               ? Colors.transparent |         actions: <Type, Action<Intent>>{PopIntent: PopAction(context)}, | ||||||
|               : Theme.of(context).scaffoldBackgroundColor, |         child: Focus( | ||||||
|       body: |           focusNode: focusNode, | ||||||
|           noBackground ? content : AppBackground(isRoot: true, child: content), |           child: Scaffold( | ||||||
|       appBar: appBar, |             extendBody: extendBody ?? true, | ||||||
|       bottomNavigationBar: bottomNavigationBar, |             extendBodyBehindAppBar: true, | ||||||
|       bottomSheet: bottomSheet, |             backgroundColor: | ||||||
|       drawer: drawer, |                 noBackground | ||||||
|       endDrawer: endDrawer, |                     ? Colors.transparent | ||||||
|       floatingActionButton: floatingActionButton, |                     : Theme.of(context).scaffoldBackgroundColor, | ||||||
|       floatingActionButtonAnimator: floatingActionButtonAnimator, |             body: | ||||||
|       floatingActionButtonLocation: floatingActionButtonLocation, |                 noBackground | ||||||
|       onDrawerChanged: onDrawerChanged, |                     ? content | ||||||
|       onEndDrawerChanged: onEndDrawerChanged, |                     : AppBackground(isRoot: true, child: content), | ||||||
|  |             appBar: appBar, | ||||||
|  |             bottomNavigationBar: bottomNavigationBar, | ||||||
|  |             bottomSheet: bottomSheet, | ||||||
|  |             drawer: drawer, | ||||||
|  |             endDrawer: endDrawer, | ||||||
|  |             floatingActionButton: floatingActionButton, | ||||||
|  |             floatingActionButtonAnimator: floatingActionButtonAnimator, | ||||||
|  |             floatingActionButtonLocation: floatingActionButtonLocation, | ||||||
|  |             onDrawerChanged: onDrawerChanged, | ||||||
|  |             onEndDrawerChanged: onEndDrawerChanged, | ||||||
|  |           ), | ||||||
|  |         ), | ||||||
|  |       ), | ||||||
|     ); |     ); | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | class PopIntent extends Intent { | ||||||
|  |   const PopIntent(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class PopAction extends Action<PopIntent> { | ||||||
|  |   final BuildContext context; | ||||||
|  |  | ||||||
|  |   PopAction(this.context); | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   void invoke(PopIntent intent) { | ||||||
|  |     if (context.canPop()) { | ||||||
|  |       context.pop(); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
| class PageBackButton extends StatelessWidget { | class PageBackButton extends StatelessWidget { | ||||||
|   final Color? color; |   final Color? color; | ||||||
|   final List<Shadow>? shadows; |   final List<Shadow>? shadows; | ||||||
| @@ -271,11 +319,12 @@ class AppBackground extends ConsumerWidget { | |||||||
|   @override |   @override | ||||||
|   Widget build(BuildContext context, WidgetRef ref) { |   Widget build(BuildContext context, WidgetRef ref) { | ||||||
|     final imageFileAsync = ref.watch(backgroundImageFileProvider); |     final imageFileAsync = ref.watch(backgroundImageFileProvider); | ||||||
|  |     final settings = ref.watch(appSettingsNotifierProvider); | ||||||
|  |  | ||||||
|     if (isRoot || !isWideScreen(context)) { |     if (isRoot || !isWideScreen(context)) { | ||||||
|       return imageFileAsync.when( |       return imageFileAsync.when( | ||||||
|         data: (file) { |         data: (file) { | ||||||
|           if (file != null) { |           if (file != null && settings.showBackgroundImage) { | ||||||
|             return Container( |             return Container( | ||||||
|               color: Theme.of(context).colorScheme.surface, |               color: Theme.of(context).colorScheme.surface, | ||||||
|               child: Container( |               child: Container( | ||||||
|   | |||||||
| @@ -1,15 +1,18 @@ | |||||||
| import 'dart:async'; | import 'dart:async'; | ||||||
|  | import 'package:bitsdojo_window/bitsdojo_window.dart'; | ||||||
| import 'package:flutter/material.dart'; | import 'package:flutter/material.dart'; | ||||||
| import 'package:flutter_hooks/flutter_hooks.dart'; | import 'package:flutter_hooks/flutter_hooks.dart'; | ||||||
| import 'package:hooks_riverpod/hooks_riverpod.dart'; | import 'package:hooks_riverpod/hooks_riverpod.dart'; | ||||||
| import 'package:island/pods/websocket.dart'; | import 'package:island/pods/websocket.dart'; | ||||||
|  | import 'package:island/screens/tray_manager.dart'; | ||||||
| import 'package:island/services/notify.dart'; | import 'package:island/services/notify.dart'; | ||||||
| import 'package:island/services/sharing_intent.dart'; | import 'package:island/services/sharing_intent.dart'; | ||||||
| import 'package:island/services/update_service.dart'; | import 'package:island/services/update_service.dart'; | ||||||
| import 'package:island/widgets/content/network_status_sheet.dart'; | import 'package:island/widgets/content/network_status_sheet.dart'; | ||||||
| import 'package:island/widgets/tour/tour.dart'; | import 'package:island/widgets/tour/tour.dart'; | ||||||
|  | import 'package:tray_manager/tray_manager.dart'; | ||||||
|  |  | ||||||
| class AppWrapper extends HookConsumerWidget { | class AppWrapper extends HookConsumerWidget with TrayListener { | ||||||
|   final Widget child; |   final Widget child; | ||||||
|   const AppWrapper({super.key, required this.child}); |   const AppWrapper({super.key, required this.child}); | ||||||
|  |  | ||||||
| @@ -20,10 +23,16 @@ class AppWrapper extends HookConsumerWidget { | |||||||
|       Future(() { |       Future(() { | ||||||
|         if (context.mounted) ntySubs = setupNotificationListener(context, ref); |         if (context.mounted) ntySubs = setupNotificationListener(context, ref); | ||||||
|       }); |       }); | ||||||
|  |  | ||||||
|       final sharingService = SharingIntentService(); |       final sharingService = SharingIntentService(); | ||||||
|       sharingService.initialize(context); |       sharingService.initialize(context); | ||||||
|  |  | ||||||
|       UpdateService().checkForUpdates(context); |       UpdateService().checkForUpdates(context); | ||||||
|  |  | ||||||
|  |       TrayService.instance.initialize(this); | ||||||
|  |  | ||||||
|       return () { |       return () { | ||||||
|  |         TrayService.instance.dispose(this); | ||||||
|         sharingService.dispose(); |         sharingService.dispose(); | ||||||
|         ntySubs?.cancel(); |         ntySubs?.cancel(); | ||||||
|       }; |       }; | ||||||
| @@ -52,4 +61,31 @@ class AppWrapper extends HookConsumerWidget { | |||||||
|  |  | ||||||
|     return TourTriggerWidget(key: UniqueKey(), child: child); |     return TourTriggerWidget(key: UniqueKey(), child: child); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   void _trayIconPrimaryAction() { | ||||||
|  |     if (appWindow.isVisible) { | ||||||
|  |       appWindow.restore(); | ||||||
|  |     } else { | ||||||
|  |       appWindow.show(); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   void _trayIconSecondaryAction() { | ||||||
|  |     trayManager.popUpContextMenu(); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   void onTrayIconMouseUp() { | ||||||
|  |     _trayIconPrimaryAction(); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   void onTrayIconRightMouseDown() { | ||||||
|  |     _trayIconSecondaryAction(); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   void onTrayMenuItemClick(MenuItem menuItem) { | ||||||
|  |     TrayService.instance.handleAction(menuItem); | ||||||
|  |   } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -3,6 +3,7 @@ import 'dart:convert'; | |||||||
| import 'package:dio/dio.dart'; | import 'package:dio/dio.dart'; | ||||||
| import 'package:easy_localization/easy_localization.dart'; | import 'package:easy_localization/easy_localization.dart'; | ||||||
| import 'package:flutter/material.dart'; | import 'package:flutter/material.dart'; | ||||||
|  | import 'package:flutter_hooks/flutter_hooks.dart'; | ||||||
| import 'package:go_router/go_router.dart'; | import 'package:go_router/go_router.dart'; | ||||||
| import 'package:gap/gap.dart'; | import 'package:gap/gap.dart'; | ||||||
| import 'package:hooks_riverpod/hooks_riverpod.dart'; | import 'package:hooks_riverpod/hooks_riverpod.dart'; | ||||||
| @@ -14,6 +15,7 @@ import 'package:island/widgets/alert.dart'; | |||||||
| import 'package:island/widgets/content/cloud_files.dart'; | import 'package:island/widgets/content/cloud_files.dart'; | ||||||
| import 'package:material_symbols_icons/symbols.dart'; | import 'package:material_symbols_icons/symbols.dart'; | ||||||
| import 'package:riverpod_annotation/riverpod_annotation.dart'; | import 'package:riverpod_annotation/riverpod_annotation.dart'; | ||||||
|  | import 'package:slide_countdown/slide_countdown.dart'; | ||||||
| import 'package:styled_widget/styled_widget.dart'; | import 'package:styled_widget/styled_widget.dart'; | ||||||
|  |  | ||||||
| part 'check_in.g.dart'; | part 'check_in.g.dart'; | ||||||
| @@ -34,6 +36,17 @@ Future<SnCheckInResult?> checkInResultToday(Ref ref) async { | |||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @riverpod | ||||||
|  | Future<SnNotableDay?> nextNotableDay(Ref ref) async { | ||||||
|  |   final client = ref.watch(apiClientProvider); | ||||||
|  |   try { | ||||||
|  |     final resp = await client.get('/id/notable/me/next'); | ||||||
|  |     return SnNotableDay.fromJson(resp.data); | ||||||
|  |   } catch (err) { | ||||||
|  |     return null; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
| class CheckInWidget extends HookConsumerWidget { | class CheckInWidget extends HookConsumerWidget { | ||||||
|   final EdgeInsets? margin; |   final EdgeInsets? margin; | ||||||
|   final VoidCallback? onChecked; |   final VoidCallback? onChecked; | ||||||
| @@ -42,6 +55,22 @@ class CheckInWidget extends HookConsumerWidget { | |||||||
|   @override |   @override | ||||||
|   Widget build(BuildContext context, WidgetRef ref) { |   Widget build(BuildContext context, WidgetRef ref) { | ||||||
|     final todayResult = ref.watch(checkInResultTodayProvider); |     final todayResult = ref.watch(checkInResultTodayProvider); | ||||||
|  |     final nextNotableDay = ref.watch(nextNotableDayProvider); | ||||||
|  |  | ||||||
|  |     final userinfo = ref.watch(userInfoProvider); | ||||||
|  |     final isAdult = useMemoized(() { | ||||||
|  |       final birthday = userinfo.value?.profile.birthday; | ||||||
|  |       if (birthday == null) return false; | ||||||
|  |       final now = DateTime.now(); | ||||||
|  |       final age = | ||||||
|  |           now.year - | ||||||
|  |           birthday.year - | ||||||
|  |           ((now.month < birthday.month || | ||||||
|  |                   (now.month == birthday.month && now.day < birthday.day)) | ||||||
|  |               ? 1 | ||||||
|  |               : 0); | ||||||
|  |       return age >= 18; | ||||||
|  |     }, [userinfo]); | ||||||
|  |  | ||||||
|     Future<void> checkIn({String? captchatTk}) async { |     Future<void> checkIn({String? captchatTk}) async { | ||||||
|       final client = ref.read(apiClientProvider); |       final client = ref.read(apiClientProvider); | ||||||
| @@ -71,98 +100,147 @@ class CheckInWidget extends HookConsumerWidget { | |||||||
|     return Card( |     return Card( | ||||||
|       margin: |       margin: | ||||||
|           margin ?? EdgeInsets.only(left: 16, right: 16, top: 16, bottom: 8), |           margin ?? EdgeInsets.only(left: 16, right: 16, top: 16, bottom: 8), | ||||||
|       child: Row( |       child: Column( | ||||||
|         crossAxisAlignment: CrossAxisAlignment.center, |         crossAxisAlignment: CrossAxisAlignment.center, | ||||||
|         spacing: 16, |         spacing: 8, | ||||||
|         children: [ |         children: [ | ||||||
|           ClipRRect( |           Column( | ||||||
|             borderRadius: BorderRadius.circular(8), |             crossAxisAlignment: CrossAxisAlignment.start, | ||||||
|             child: Container( |             children: [ | ||||||
|               color: Theme.of(context).colorScheme.secondaryContainer, |               Row( | ||||||
|               width: 56, |                 spacing: 8, | ||||||
|               height: 56, |                 mainAxisSize: MainAxisSize.min, | ||||||
|               child: |                 crossAxisAlignment: CrossAxisAlignment.center, | ||||||
|                   Column( |                 children: [ | ||||||
|                     mainAxisSize: MainAxisSize.min, |                   Icon( | ||||||
|                     crossAxisAlignment: CrossAxisAlignment.center, |                     switch (DateTime.now().weekday) { | ||||||
|                     children: [ |                       6 || 7 => Symbols.weekend, | ||||||
|                       Text(DateFormat('EEE').format(DateTime.now())) |                       _ => isAdult ? Symbols.work : Symbols.school, | ||||||
|                           .fontSize(16) |                     }, | ||||||
|                           .bold() |                     fill: 1, | ||||||
|                           .textColor( |                     size: 16, | ||||||
|                             Theme.of(context).colorScheme.onSecondaryContainer, |                   ).padding(right: 2), | ||||||
|  |                   Text(DateFormat('EEE').format(DateTime.now())) | ||||||
|  |                       .fontSize(16) | ||||||
|  |                       .bold() | ||||||
|  |                       .textColor( | ||||||
|  |                         Theme.of(context).colorScheme.onSecondaryContainer, | ||||||
|  |                       ), | ||||||
|  |                   Text(DateFormat('MM/dd').format(DateTime.now())) | ||||||
|  |                       .fontSize(12) | ||||||
|  |                       .textColor( | ||||||
|  |                         Theme.of(context).colorScheme.onSecondaryContainer, | ||||||
|  |                       ) | ||||||
|  |                       .padding(top: 2), | ||||||
|  |                 ], | ||||||
|  |               ), | ||||||
|  |               Row( | ||||||
|  |                 spacing: 5, | ||||||
|  |                 children: [ | ||||||
|  |                   Text('notableDayNext') | ||||||
|  |                       .tr(args: [nextNotableDay.value?.localName ?? 'idk']) | ||||||
|  |                       .fontSize(12), | ||||||
|  |                   SlideCountdown( | ||||||
|  |                     decoration: const BoxDecoration(), | ||||||
|  |                     style: const TextStyle(fontSize: 12), | ||||||
|  |                     separatorStyle: const TextStyle(fontSize: 12), | ||||||
|  |                     padding: EdgeInsets.zero, | ||||||
|  |                     duration: nextNotableDay.value?.date.difference( | ||||||
|  |                       DateTime.now(), | ||||||
|  |                     ), | ||||||
|  |                   ), | ||||||
|  |                 ], | ||||||
|  |               ), | ||||||
|  |             ], | ||||||
|  |           ).padding(horizontal: 16, top: 8), | ||||||
|  |           const Divider(height: 1), | ||||||
|  |           Row( | ||||||
|  |             children: [ | ||||||
|  |               Expanded( | ||||||
|  |                 child: AnimatedSwitcher( | ||||||
|  |                   duration: const Duration(milliseconds: 300), | ||||||
|  |                   child: todayResult.when( | ||||||
|  |                     data: (result) { | ||||||
|  |                       if (result == null) return _CheckInNoneWidget(); | ||||||
|  |                       return Column( | ||||||
|  |                         crossAxisAlignment: CrossAxisAlignment.stretch, | ||||||
|  |                         children: [ | ||||||
|  |                           Text( | ||||||
|  |                             'checkInResultLevel${result.level}', | ||||||
|  |                           ).tr().fontSize(15).bold(), | ||||||
|  |                           Wrap( | ||||||
|  |                             children: | ||||||
|  |                                 result.tips | ||||||
|  |                                     .map((e) { | ||||||
|  |                                       return Row( | ||||||
|  |                                         mainAxisSize: MainAxisSize.min, | ||||||
|  |                                         children: [ | ||||||
|  |                                           Icon( | ||||||
|  |                                             e.isPositive | ||||||
|  |                                                 ? Symbols.thumb_up | ||||||
|  |                                                 : Symbols.thumb_down, | ||||||
|  |                                             size: 12, | ||||||
|  |                                           ), | ||||||
|  |                                           const Gap(4), | ||||||
|  |                                           Text(e.title).fontSize(11), | ||||||
|  |                                         ], | ||||||
|  |                                       ); | ||||||
|  |                                     }) | ||||||
|  |                                     .toList() | ||||||
|  |                                     .expand( | ||||||
|  |                                       (widget) => [ | ||||||
|  |                                         widget, | ||||||
|  |                                         Text('  ·  ').fontSize(11), | ||||||
|  |                                       ], | ||||||
|  |                                     ) | ||||||
|  |                                     .toList() | ||||||
|  |                                   ..removeLast(), | ||||||
|                           ), |                           ), | ||||||
|                       Text(DateFormat('MM/dd').format(DateTime.now())) |                         ], | ||||||
|                           .fontSize(12) |                       ); | ||||||
|                           .textColor( |                     }, | ||||||
|                             Theme.of(context).colorScheme.onSecondaryContainer, |                     loading: () => _CheckInNoneWidget(), | ||||||
|                           ), |                     error: | ||||||
|                     ], |                         (err, stack) => Column( | ||||||
|                   ).center(), |                           crossAxisAlignment: CrossAxisAlignment.start, | ||||||
|             ), |                           children: [ | ||||||
|           ), |                             Text('error').tr().fontSize(15).bold(), | ||||||
|           Expanded( |                             Text(err.toString()).fontSize(11), | ||||||
|             child: AnimatedSwitcher( |                           ], | ||||||
|               duration: const Duration(milliseconds: 300), |                         ), | ||||||
|               child: todayResult.when( |                   ), | ||||||
|                 data: (result) { |                 ), | ||||||
|                   if (result == null) return _CheckInNoneWidget(); |               ), | ||||||
|                   return Column( |               IconButton.outlined( | ||||||
|                     crossAxisAlignment: CrossAxisAlignment.stretch, |                 onPressed: () { | ||||||
|                     children: [ |                   if (todayResult.valueOrNull == null) { | ||||||
|                       Text( |                     checkIn(); | ||||||
|                         'checkInResultLevel${result.level}', |                   } else { | ||||||
|                       ).tr().fontSize(15).bold(), |                     context.pushNamed( | ||||||
|                       Text( |                       'accountCalendar', | ||||||
|                         result.tips |                       pathParameters: {'name': 'me'}, | ||||||
|                             .map( |                     ); | ||||||
|                               (e) => '${e.isPositive ? '宜' : '忌'} ${e.title}', |                   } | ||||||
|                             ) |  | ||||||
|                             .join('  ·  '), |  | ||||||
|                       ).fontSize(11), |  | ||||||
|                     ], |  | ||||||
|                   ); |  | ||||||
|                 }, |                 }, | ||||||
|                 loading: () => _CheckInNoneWidget(), |                 icon: AnimatedSwitcher( | ||||||
|                 error: |                   duration: const Duration(milliseconds: 300), | ||||||
|                     (err, stack) => Column( |                   child: todayResult.when( | ||||||
|                       crossAxisAlignment: CrossAxisAlignment.start, |                     data: | ||||||
|                       children: [ |                         (result) => Icon( | ||||||
|                         Text('error').tr().fontSize(15).bold(), |                           result == null | ||||||
|                         Text(err.toString()).fontSize(11), |                               ? Symbols.local_fire_department | ||||||
|                       ], |                               : Symbols.event, | ||||||
|                     ), |                           key: ValueKey(result != null), | ||||||
|  |                         ), | ||||||
|  |                     loading: () => const Icon(Symbols.refresh), | ||||||
|  |                     error: (_, _) => const Icon(Symbols.error), | ||||||
|  |                   ), | ||||||
|  |                 ), | ||||||
|               ), |               ), | ||||||
|             ), |             ], | ||||||
|           ), |           ).padding(horizontal: 16, bottom: 12, top: 4), | ||||||
|           IconButton.outlined( |  | ||||||
|             onPressed: () { |  | ||||||
|               if (todayResult.valueOrNull == null) { |  | ||||||
|                 checkIn(); |  | ||||||
|               } else { |  | ||||||
|                 context.pushNamed( |  | ||||||
|                   'accountCalendar', |  | ||||||
|                   pathParameters: {'name': 'me'}, |  | ||||||
|                 ); |  | ||||||
|               } |  | ||||||
|             }, |  | ||||||
|             icon: AnimatedSwitcher( |  | ||||||
|               duration: const Duration(milliseconds: 300), |  | ||||||
|               child: todayResult.when( |  | ||||||
|                 data: |  | ||||||
|                     (result) => Icon( |  | ||||||
|                       result == null |  | ||||||
|                           ? Symbols.local_fire_department |  | ||||||
|                           : Symbols.event, |  | ||||||
|                       key: ValueKey(result != null), |  | ||||||
|                     ), |  | ||||||
|                 loading: () => const Icon(Symbols.refresh), |  | ||||||
|                 error: (_, _) => const Icon(Symbols.error), |  | ||||||
|               ), |  | ||||||
|             ), |  | ||||||
|           ), |  | ||||||
|         ], |         ], | ||||||
|       ).padding(horizontal: 16, vertical: 12), |       ), | ||||||
|     ); |     ); | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -26,5 +26,24 @@ final checkInResultTodayProvider = | |||||||
| @Deprecated('Will be removed in 3.0. Use Ref instead') | @Deprecated('Will be removed in 3.0. Use Ref instead') | ||||||
| // ignore: unused_element | // ignore: unused_element | ||||||
| typedef CheckInResultTodayRef = AutoDisposeFutureProviderRef<SnCheckInResult?>; | typedef CheckInResultTodayRef = AutoDisposeFutureProviderRef<SnCheckInResult?>; | ||||||
|  | String _$nextNotableDayHash() => r'698370bec4be28774d332412c5a701f914064c90'; | ||||||
|  |  | ||||||
|  | /// See also [nextNotableDay]. | ||||||
|  | @ProviderFor(nextNotableDay) | ||||||
|  | final nextNotableDayProvider = | ||||||
|  |     AutoDisposeFutureProvider<SnNotableDay?>.internal( | ||||||
|  |       nextNotableDay, | ||||||
|  |       name: r'nextNotableDayProvider', | ||||||
|  |       debugGetCreateSourceHash: | ||||||
|  |           const bool.fromEnvironment('dart.vm.product') | ||||||
|  |               ? null | ||||||
|  |               : _$nextNotableDayHash, | ||||||
|  |       dependencies: null, | ||||||
|  |       allTransitiveDependencies: null, | ||||||
|  |     ); | ||||||
|  |  | ||||||
|  | @Deprecated('Will be removed in 3.0. Use Ref instead') | ||||||
|  | // ignore: unused_element | ||||||
|  | typedef NextNotableDayRef = AutoDisposeFutureProviderRef<SnNotableDay?>; | ||||||
| // ignore_for_file: type=lint | // 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 | // 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 | ||||||
|   | |||||||
| @@ -10,6 +10,7 @@ import 'package:hooks_riverpod/hooks_riverpod.dart'; | |||||||
| import 'package:island/models/file.dart'; | import 'package:island/models/file.dart'; | ||||||
| import 'package:island/pods/network.dart'; | import 'package:island/pods/network.dart'; | ||||||
| import 'package:island/services/file.dart'; | import 'package:island/services/file.dart'; | ||||||
|  | import 'package:island/utils/format.dart'; | ||||||
| import 'package:island/widgets/alert.dart'; | import 'package:island/widgets/alert.dart'; | ||||||
| import 'package:island/widgets/content/cloud_files.dart'; | import 'package:island/widgets/content/cloud_files.dart'; | ||||||
| import 'package:island/widgets/content/sheet.dart'; | import 'package:island/widgets/content/sheet.dart'; | ||||||
| @@ -284,6 +285,13 @@ class AttachmentPreview extends HookConsumerWidget { | |||||||
|                   Builder( |                   Builder( | ||||||
|                     key: ValueKey(item.hashCode), |                     key: ValueKey(item.hashCode), | ||||||
|                     builder: (context) { |                     builder: (context) { | ||||||
|  |                       final fallbackIcon = switch (item.type) { | ||||||
|  |                         UniversalFileType.video => Symbols.video_file, | ||||||
|  |                         UniversalFileType.audio => Symbols.audio_file, | ||||||
|  |                         UniversalFileType.image => Symbols.image, | ||||||
|  |                         _ => Symbols.insert_drive_file, | ||||||
|  |                       }; | ||||||
|  |  | ||||||
|                       if (item.isOnCloud) { |                       if (item.isOnCloud) { | ||||||
|                         return CloudFileWidget(item: item.data); |                         return CloudFileWidget(item: item.data); | ||||||
|                       } else if (item.data is XFile) { |                       } else if (item.data is XFile) { | ||||||
| @@ -309,9 +317,23 @@ class AttachmentPreview extends HookConsumerWidget { | |||||||
|                                 : Image.file(File(file.path)); |                                 : Image.file(File(file.path)); | ||||||
|                           default: |                           default: | ||||||
|                             return Column( |                             return Column( | ||||||
|  |                               mainAxisAlignment: MainAxisAlignment.center, | ||||||
|                               children: [ |                               children: [ | ||||||
|                                 const Icon(Symbols.document_scanner), |                                 Icon(fallbackIcon), | ||||||
|                                 Text(file.name), |                                 const Gap(6), | ||||||
|  |                                 Text(file.name, textAlign: TextAlign.center), | ||||||
|  |                                 FutureBuilder( | ||||||
|  |                                   future: file.length(), | ||||||
|  |                                   builder: (context, snapshot) { | ||||||
|  |                                     if (snapshot.hasData) { | ||||||
|  |                                       final size = snapshot.data as int; | ||||||
|  |                                       return Text( | ||||||
|  |                                         formatFileSize(size), | ||||||
|  |                                       ).fontSize(11); | ||||||
|  |                                     } | ||||||
|  |                                     return const SizedBox.shrink(); | ||||||
|  |                                   }, | ||||||
|  |                                 ), | ||||||
|                               ], |                               ], | ||||||
|                             ); |                             ); | ||||||
|                         } |                         } | ||||||
| @@ -321,7 +343,14 @@ class AttachmentPreview extends HookConsumerWidget { | |||||||
|                             return Image.memory(item.data); |                             return Image.memory(item.data); | ||||||
|                           default: |                           default: | ||||||
|                             return Column( |                             return Column( | ||||||
|                               children: [const Icon(Symbols.document_scanner)], |                               mainAxisAlignment: MainAxisAlignment.center, | ||||||
|  |                               children: [ | ||||||
|  |                                 Icon(fallbackIcon), | ||||||
|  |                                 const Gap(6), | ||||||
|  |                                 Text( | ||||||
|  |                                   formatFileSize(item.data.length), | ||||||
|  |                                 ).fontSize(11), | ||||||
|  |                               ], | ||||||
|                             ); |                             ); | ||||||
|                         } |                         } | ||||||
|                       } |                       } | ||||||
|   | |||||||
| @@ -1,12 +1,15 @@ | |||||||
| import 'dart:convert'; | import 'dart:convert'; | ||||||
|  | import 'dart:io'; | ||||||
| import 'dart:math' as math; | import 'dart:math' as math; | ||||||
| import 'dart:ui'; | import 'dart:ui'; | ||||||
|  |  | ||||||
| import 'package:dismissible_page/dismissible_page.dart'; | import 'package:dismissible_page/dismissible_page.dart'; | ||||||
| import 'package:easy_localization/easy_localization.dart'; | import 'package:easy_localization/easy_localization.dart'; | ||||||
|  | import 'package:flutter_blurhash/flutter_blurhash.dart'; | ||||||
|  | import 'package:file_saver/file_saver.dart'; | ||||||
|  | import 'package:flutter/foundation.dart'; | ||||||
| import 'package:flutter/material.dart'; | import 'package:flutter/material.dart'; | ||||||
| import 'package:flutter/services.dart'; | import 'package:flutter/services.dart'; | ||||||
| import 'package:flutter_blurhash/flutter_blurhash.dart'; |  | ||||||
| import 'package:flutter_hooks/flutter_hooks.dart'; | import 'package:flutter_hooks/flutter_hooks.dart'; | ||||||
| import 'package:gap/gap.dart'; | import 'package:gap/gap.dart'; | ||||||
| import 'package:gal/gal.dart'; | import 'package:gal/gal.dart'; | ||||||
| @@ -14,6 +17,7 @@ import 'package:hooks_riverpod/hooks_riverpod.dart'; | |||||||
| import 'package:island/models/file.dart'; | import 'package:island/models/file.dart'; | ||||||
| import 'package:island/pods/config.dart'; | import 'package:island/pods/config.dart'; | ||||||
| import 'package:island/pods/network.dart'; | import 'package:island/pods/network.dart'; | ||||||
|  | import 'package:island/utils/format.dart'; | ||||||
| import 'package:island/widgets/alert.dart'; | import 'package:island/widgets/alert.dart'; | ||||||
| import 'package:island/widgets/content/cloud_files.dart'; | import 'package:island/widgets/content/cloud_files.dart'; | ||||||
| import 'package:island/widgets/content/sensitive.dart'; | import 'package:island/widgets/content/sensitive.dart'; | ||||||
| @@ -321,7 +325,7 @@ class CloudFileZoomIn extends HookConsumerWidget { | |||||||
|     Future<void> saveToGallery() async { |     Future<void> saveToGallery() async { | ||||||
|       try { |       try { | ||||||
|         // Show loading indicator |         // Show loading indicator | ||||||
|         showSnackBar('Saving image to gallery...'); |         showSnackBar('Saving image...'); | ||||||
|  |  | ||||||
|         // Get the image URL |         // Get the image URL | ||||||
|         final client = ref.watch(apiClientProvider); |         final client = ref.watch(apiClientProvider); | ||||||
| @@ -339,25 +343,23 @@ class CloudFileZoomIn extends HookConsumerWidget { | |||||||
|           filePath, |           filePath, | ||||||
|           queryParameters: {'original': true}, |           queryParameters: {'original': true}, | ||||||
|         ); |         ); | ||||||
|         await Gal.putImage(filePath, album: 'Solar Network'); |         if (!kIsWeb && (Platform.isAndroid || Platform.isIOS)) { | ||||||
|  |           // Save to gallery | ||||||
|         // Show success message |           await Gal.putImage(filePath, album: 'Solar Network'); | ||||||
|         showSnackBar('Image saved to gallery'); |           // Show success message | ||||||
|  |           showSnackBar('Image saved to gallery'); | ||||||
|  |         } else { | ||||||
|  |           await FileSaver.instance.saveFile( | ||||||
|  |             name: item.name.isEmpty ? '${item.id}.$extName' : item.name, | ||||||
|  |             file: File(filePath), | ||||||
|  |           ); | ||||||
|  |           showSnackBar('Image saved to $filePath'); | ||||||
|  |         } | ||||||
|       } catch (e) { |       } catch (e) { | ||||||
|         showErrorAlert(e); |         showErrorAlert(e); | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     String formatFileSize(int bytes) { |  | ||||||
|       if (bytes <= 0) return '0 B'; |  | ||||||
|       if (bytes < 1024) return '$bytes B'; |  | ||||||
|       if (bytes < 1024 * 1024) return '${(bytes / 1024).toStringAsFixed(2)} KB'; |  | ||||||
|       if (bytes < 1024 * 1024 * 1024) { |  | ||||||
|         return '${(bytes / (1024 * 1024)).toStringAsFixed(2)} MB'; |  | ||||||
|       } |  | ||||||
|       return '${(bytes / (1024 * 1024 * 1024)).toStringAsFixed(2)} GB'; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     void showInfoSheet() { |     void showInfoSheet() { | ||||||
|       final theme = Theme.of(context); |       final theme = Theme.of(context); | ||||||
|       final exifData = item.fileMeta?['exif'] as Map<String, dynamic>? ?? {}; |       final exifData = item.fileMeta?['exif'] as Map<String, dynamic>? ?? {}; | ||||||
| @@ -437,7 +439,24 @@ class CloudFileZoomIn extends HookConsumerWidget { | |||||||
|                     ).padding(horizontal: 24, vertical: 16), |                     ).padding(horizontal: 24, vertical: 16), | ||||||
|                     const Divider(height: 1), |                     const Divider(height: 1), | ||||||
|                     ListTile( |                     ListTile( | ||||||
|                       leading: const Icon(Icons.file_present), |                       leading: const Icon(Symbols.tag), | ||||||
|  |                       title: Text('ID').tr(), | ||||||
|  |                       subtitle: Text( | ||||||
|  |                         item.id, | ||||||
|  |                         maxLines: 1, | ||||||
|  |                         overflow: TextOverflow.ellipsis, | ||||||
|  |                       ), | ||||||
|  |                       contentPadding: EdgeInsets.symmetric(horizontal: 24), | ||||||
|  |                       trailing: IconButton( | ||||||
|  |                         icon: const Icon(Icons.copy), | ||||||
|  |                         onPressed: () { | ||||||
|  |                           Clipboard.setData(ClipboardData(text: item.id)); | ||||||
|  |                           showSnackBar('File ID copied to clipboard'); | ||||||
|  |                         }, | ||||||
|  |                       ), | ||||||
|  |                     ), | ||||||
|  |                     ListTile( | ||||||
|  |                       leading: const Icon(Symbols.file_present), | ||||||
|                       title: Text('Name').tr(), |                       title: Text('Name').tr(), | ||||||
|                       subtitle: Text( |                       subtitle: Text( | ||||||
|                         item.name, |                         item.name, | ||||||
| @@ -623,6 +642,10 @@ class CloudFileZoomIn extends HookConsumerWidget { | |||||||
|       ); |       ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     final shadow = [ | ||||||
|  |       Shadow(color: Colors.black54, blurRadius: 5.0, offset: Offset(1.0, 1.0)), | ||||||
|  |     ]; | ||||||
|  |  | ||||||
|     return DismissiblePage( |     return DismissiblePage( | ||||||
|       isFullScreen: true, |       isFullScreen: true, | ||||||
|       backgroundColor: Colors.transparent, |       backgroundColor: Colors.transparent, | ||||||
| @@ -660,22 +683,17 @@ class CloudFileZoomIn extends HookConsumerWidget { | |||||||
|               children: [ |               children: [ | ||||||
|                 Row( |                 Row( | ||||||
|                   children: [ |                   children: [ | ||||||
|                     IconButton( |                     if (!kIsWeb) | ||||||
|                       icon: Icon( |                       IconButton( | ||||||
|                         Icons.save_alt, |                         icon: Icon( | ||||||
|                         color: Colors.white, |                           Icons.save_alt, | ||||||
|                         shadows: [ |                           color: Colors.white, | ||||||
|                           Shadow( |                           shadows: shadow, | ||||||
|                             color: Colors.black54, |                         ), | ||||||
|                             blurRadius: 5.0, |                         onPressed: () async { | ||||||
|                             offset: Offset(1.0, 1.0), |                           saveToGallery(); | ||||||
|                           ), |                         }, | ||||||
|                         ], |  | ||||||
|                       ), |                       ), | ||||||
|                       onPressed: () async { |  | ||||||
|                         saveToGallery(); |  | ||||||
|                       }, |  | ||||||
|                     ), |  | ||||||
|                     IconButton( |                     IconButton( | ||||||
|                       onPressed: () { |                       onPressed: () { | ||||||
|                         showOriginal.value = !showOriginal.value; |                         showOriginal.value = !showOriginal.value; | ||||||
| @@ -683,29 +701,13 @@ class CloudFileZoomIn extends HookConsumerWidget { | |||||||
|                       icon: Icon( |                       icon: Icon( | ||||||
|                         showOriginal.value ? Symbols.hd : Symbols.sd, |                         showOriginal.value ? Symbols.hd : Symbols.sd, | ||||||
|                         color: Colors.white, |                         color: Colors.white, | ||||||
|                         shadows: [ |                         shadows: shadow, | ||||||
|                           Shadow( |  | ||||||
|                             color: Colors.black54, |  | ||||||
|                             blurRadius: 5.0, |  | ||||||
|                             offset: Offset(1.0, 1.0), |  | ||||||
|                           ), |  | ||||||
|                         ], |  | ||||||
|                       ), |                       ), | ||||||
|                     ), |                     ), | ||||||
|                   ], |                   ], | ||||||
|                 ), |                 ), | ||||||
|                 IconButton( |                 IconButton( | ||||||
|                   icon: Icon( |                   icon: Icon(Icons.close, color: Colors.white, shadows: shadow), | ||||||
|                     Icons.close, |  | ||||||
|                     color: Colors.white, |  | ||||||
|                     shadows: [ |  | ||||||
|                       Shadow( |  | ||||||
|                         color: Colors.black54, |  | ||||||
|                         blurRadius: 5.0, |  | ||||||
|                         offset: Offset(1.0, 1.0), |  | ||||||
|                       ), |  | ||||||
|                     ], |  | ||||||
|                   ), |  | ||||||
|                   onPressed: () => Navigator.of(context).pop(), |                   onPressed: () => Navigator.of(context).pop(), | ||||||
|                 ), |                 ), | ||||||
|               ], |               ], | ||||||
| @@ -722,26 +724,24 @@ class CloudFileZoomIn extends HookConsumerWidget { | |||||||
|                   icon: Icon( |                   icon: Icon( | ||||||
|                     Icons.info_outline, |                     Icons.info_outline, | ||||||
|                     color: Colors.white, |                     color: Colors.white, | ||||||
|                     shadows: [ |                     shadows: shadow, | ||||||
|                       Shadow( |  | ||||||
|                         color: Colors.black54, |  | ||||||
|                         blurRadius: 5.0, |  | ||||||
|                         offset: Offset(1.0, 1.0), |  | ||||||
|                       ), |  | ||||||
|                     ], |  | ||||||
|                   ), |                   ), | ||||||
|                   onPressed: showInfoSheet, |                   onPressed: showInfoSheet, | ||||||
|                 ), |                 ), | ||||||
|                 Spacer(), |                 Spacer(), | ||||||
|                 IconButton( |                 IconButton( | ||||||
|                   icon: Icon(Icons.remove, color: Colors.white), |                   icon: Icon( | ||||||
|  |                     Icons.remove, | ||||||
|  |                     color: Colors.white, | ||||||
|  |                     shadows: shadow, | ||||||
|  |                   ), | ||||||
|                   onPressed: () { |                   onPressed: () { | ||||||
|                     photoViewController.scale = |                     photoViewController.scale = | ||||||
|                         (photoViewController.scale ?? 1) - 0.05; |                         (photoViewController.scale ?? 1) - 0.05; | ||||||
|                   }, |                   }, | ||||||
|                 ), |                 ), | ||||||
|                 IconButton( |                 IconButton( | ||||||
|                   icon: Icon(Icons.add, color: Colors.white), |                   icon: Icon(Icons.add, color: Colors.white, shadows: shadow), | ||||||
|                   onPressed: () { |                   onPressed: () { | ||||||
|                     photoViewController.scale = |                     photoViewController.scale = | ||||||
|                         (photoViewController.scale ?? 1) + 0.05; |                         (photoViewController.scale ?? 1) + 0.05; | ||||||
| @@ -752,13 +752,7 @@ class CloudFileZoomIn extends HookConsumerWidget { | |||||||
|                   icon: Icon( |                   icon: Icon( | ||||||
|                     Icons.rotate_left, |                     Icons.rotate_left, | ||||||
|                     color: Colors.white, |                     color: Colors.white, | ||||||
|                     shadows: [ |                     shadows: shadow, | ||||||
|                       Shadow( |  | ||||||
|                         color: Colors.black54, |  | ||||||
|                         blurRadius: 5.0, |  | ||||||
|                         offset: Offset(1.0, 1.0), |  | ||||||
|                       ), |  | ||||||
|                     ], |  | ||||||
|                   ), |                   ), | ||||||
|                   onPressed: () { |                   onPressed: () { | ||||||
|                     rotation.value = (rotation.value - 1) % 4; |                     rotation.value = (rotation.value - 1) % 4; | ||||||
| @@ -810,164 +804,213 @@ class _CloudFileListEntry extends HookConsumerWidget { | |||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   Widget build(BuildContext context, WidgetRef ref) { |   Widget build(BuildContext context, WidgetRef ref) { | ||||||
|  |     final dataSaving = ref.watch( | ||||||
|  |       appSettingsNotifierProvider.select((s) => s.dataSavingMode), | ||||||
|  |     ); | ||||||
|     final showMature = useState(false); |     final showMature = useState(false); | ||||||
|  |     final showDataSaving = useState(!dataSaving); | ||||||
|  |     final lockedByDS = dataSaving && !showDataSaving.value; | ||||||
|  |     final lockedByMature = file.sensitiveMarks.isNotEmpty && !showMature.value; | ||||||
|  |     final meta = file.fileMeta is Map ? file.fileMeta as Map : const {}; | ||||||
|  |     final hasRatio = | ||||||
|  |         meta.containsKey('ratio') && | ||||||
|  |         (meta['ratio'] is num && (meta['ratio'] as num) != 0); | ||||||
|  |     final ratio = | ||||||
|  |         (meta['ratio'] is num && (meta['ratio'] as num) != 0) | ||||||
|  |             ? (meta['ratio'] as num).toDouble() | ||||||
|  |             : 1.0; | ||||||
|  |  | ||||||
|     var content = Stack( |     final fit = hasRatio ? BoxFit.cover : BoxFit.contain; | ||||||
|       fit: StackFit.expand, |  | ||||||
|       children: [ |     Widget bg = const SizedBox.shrink(); | ||||||
|         if (isImage) |     if (isImage) { | ||||||
|           Positioned.fill( |       if (meta['blur'] is String) { | ||||||
|             child: |         bg = BlurHash(hash: meta['blur'] as String); | ||||||
|                 file.fileMeta?['blur'] is String |       } else if (!lockedByDS && !lockedByMature) { | ||||||
|                     ? BlurHash(hash: file.fileMeta?['blur']) |         bg = ImageFiltered( | ||||||
|                     : ImageFiltered( |           imageFilter: ImageFilter.blur(sigmaX: 10, sigmaY: 10), | ||||||
|                       imageFilter: ImageFilter.blur(sigmaX: 10, sigmaY: 10), |           child: CloudFileWidget( | ||||||
|                       child: CloudFileWidget(item: file, noBlurhash: true), |             fit: fit, | ||||||
|                     ), |  | ||||||
|           ), |  | ||||||
|         if (isImage) |  | ||||||
|           CloudFileWidget( |  | ||||||
|             item: file, |             item: file, | ||||||
|             heroTag: heroTag, |  | ||||||
|             noBlurhash: true, |             noBlurhash: true, | ||||||
|             fit: BoxFit.contain, |             useInternalGate: false, | ||||||
|           ) |           ), | ||||||
|         else |         ); | ||||||
|           CloudFileWidget(item: file, heroTag: heroTag, fit: BoxFit.contain), |       } else { | ||||||
|       ], |         bg = const ColoredBox(color: Colors.black26); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     final bool fullyUnlocked = !lockedByDS && !lockedByMature; | ||||||
|  |     Widget fg = | ||||||
|  |         fullyUnlocked | ||||||
|  |             ? (isImage | ||||||
|  |                 ? CloudFileWidget( | ||||||
|  |                   item: file, | ||||||
|  |                   heroTag: heroTag, | ||||||
|  |                   noBlurhash: true, | ||||||
|  |                   fit: fit, | ||||||
|  |                   useInternalGate: false, | ||||||
|  |                 ) | ||||||
|  |                 : CloudFileWidget( | ||||||
|  |                   item: file, | ||||||
|  |                   heroTag: heroTag, | ||||||
|  |                   fit: fit, | ||||||
|  |                   useInternalGate: false, | ||||||
|  |                 )) | ||||||
|  |             : AspectRatio(aspectRatio: ratio, child: const SizedBox.shrink()); | ||||||
|  |  | ||||||
|  |     Widget overlays; | ||||||
|  |     if (lockedByDS) { | ||||||
|  |       overlays = _DataSavingOverlay(); | ||||||
|  |     } else if (file.sensitiveMarks.isNotEmpty) { | ||||||
|  |       overlays = _SensitiveOverlay( | ||||||
|  |         file: file, | ||||||
|  |         isRevealed: showMature.value, | ||||||
|  |         onHide: () => showMature.value = false, | ||||||
|  |       ); | ||||||
|  |     } else { | ||||||
|  |       overlays = const SizedBox.shrink(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     final content = Stack( | ||||||
|  |       fit: StackFit.expand, | ||||||
|  |       children: [if (isImage) Positioned.fill(child: bg), fg, overlays], | ||||||
|     ); |     ); | ||||||
|  |  | ||||||
|     if (file.sensitiveMarks.isNotEmpty) { |     return InkWell( | ||||||
|       // Show a blurred overlay only when not revealed yet, with a smooth transition |       borderRadius: const BorderRadius.all(Radius.circular(16)), | ||||||
|       content = Stack( |       onTap: () { | ||||||
|         children: [ |         if (lockedByDS) { | ||||||
|           content, |           showDataSaving.value = true; | ||||||
|           // Toggle blur overlay with animation |         } else if (lockedByMature) { | ||||||
|           Positioned.fill( |           showMature.value = true; | ||||||
|             child: AnimatedSwitcher( |         } else { | ||||||
|               duration: const Duration(milliseconds: 250), |           onTap?.call(); | ||||||
|               switchInCurve: Curves.easeOut, |         } | ||||||
|               switchOutCurve: Curves.easeIn, |       }, | ||||||
|               layoutBuilder: |       child: content, | ||||||
|                   (currentChild, previousChildren) => Stack( |     ); | ||||||
|                     fit: StackFit.expand, |   } | ||||||
|                     children: [ | } | ||||||
|                       ...previousChildren, |  | ||||||
|                       if (currentChild != null) currentChild, | class _SensitiveOverlay extends StatelessWidget { | ||||||
|                     ], |   final SnCloudFile file; | ||||||
|                   ), |   final VoidCallback? onHide; | ||||||
|               child: |   final bool isRevealed; | ||||||
|                   showMature.value |  | ||||||
|                       ? const SizedBox.shrink(key: ValueKey('revealed')) |   const _SensitiveOverlay({ | ||||||
|                       : ColoredBox( |     required this.file, | ||||||
|                         key: const ValueKey('blurred'), |     this.onHide, | ||||||
|                         color: Colors.transparent, |     this.isRevealed = false, | ||||||
|                         child: BackdropFilter( |   }); | ||||||
|                           filter: ImageFilter.blur(sigmaX: 64, sigmaY: 64), |  | ||||||
|                           child: Stack( |   @override | ||||||
|                             fit: StackFit.expand, |   Widget build(BuildContext context) { | ||||||
|                             children: [ |     if (isRevealed) { | ||||||
|                               const ColoredBox(color: Colors.transparent), |       return Positioned( | ||||||
|                               Center( |         top: 3, | ||||||
|                                 child: Container( |         left: 4, | ||||||
|                                   margin: const EdgeInsets.all(12), |         child: IconButton( | ||||||
|                                   padding: const EdgeInsets.symmetric( |           iconSize: 16, | ||||||
|                                     horizontal: 12, |           constraints: const BoxConstraints(), | ||||||
|                                     vertical: 8, |           icon: const Icon( | ||||||
|                                   ), |             Icons.visibility_off, | ||||||
|                                   decoration: BoxDecoration( |             color: Colors.white, | ||||||
|                                     color: Colors.black54, |             shadows: [ | ||||||
|                                     borderRadius: BorderRadius.circular(12), |               Shadow( | ||||||
|                                   ), |                 color: Colors.black, | ||||||
|                                   child: ConstrainedBox( |                 blurRadius: 5.0, | ||||||
|                                     constraints: const BoxConstraints( |                 offset: Offset(1.0, 1.0), | ||||||
|                                       maxWidth: 280, |               ), | ||||||
|                                     ), |             ], | ||||||
|                                     child: Column( |           ), | ||||||
|                                       mainAxisSize: MainAxisSize.min, |           tooltip: 'Blur content', | ||||||
|                                       children: [ |           onPressed: onHide, | ||||||
|                                         const Icon( |         ), | ||||||
|                                           Icons.warning, |       ); | ||||||
|                                           color: Colors.white, |     } | ||||||
|                                           fill: 1, |  | ||||||
|                                           size: 24, |     return BackdropFilter( | ||||||
|                                         ), |       filter: ImageFilter.blur(sigmaX: 64, sigmaY: 64), | ||||||
|                                         const Gap(4), |       child: Container( | ||||||
|                                         Text( |         color: Colors.transparent, | ||||||
|                                           file.sensitiveMarks |         child: Center( | ||||||
|                                               .map( |           child: _OverlayCard( | ||||||
|                                                 (e) => |             icon: Icons.warning, | ||||||
|                                                     SensitiveCategory |             title: file.sensitiveMarks | ||||||
|                                                         .values[e] |                 .map((e) => SensitiveCategory.values[e].i18nKey.tr()) | ||||||
|                                                         .i18nKey |                 .join(' · '), | ||||||
|                                                         .tr(), |             subtitle: 'Sensitive Content', | ||||||
|                                               ) |             hint: 'Tap to Reveal', | ||||||
|                                               .join(' · '), |           ), | ||||||
|                                           style: const TextStyle( |         ), | ||||||
|                                             color: Colors.white, |       ), | ||||||
|                                             fontWeight: FontWeight.w600, |     ); | ||||||
|                                           ), |   } | ||||||
|                                           textAlign: TextAlign.center, | } | ||||||
|                                         ), |  | ||||||
|                                         Text( | class _DataSavingOverlay extends StatelessWidget { | ||||||
|                                           'Sensitive Content', |   @override | ||||||
|                                           style: TextStyle( |   Widget build(BuildContext context) { | ||||||
|                                             color: Colors.white, |     return ColoredBox( | ||||||
|                                             fontSize: 13, |       color: Colors.black38, | ||||||
|                                           ), |       child: Center( | ||||||
|                                         ), |         child: _OverlayCard( | ||||||
|                                         const Gap(4), |           icon: Symbols.image, | ||||||
|                                         Text( |           title: 'Data Saving Mode', | ||||||
|                                           'Tap to Reveal', |           subtitle: '', | ||||||
|                                           style: TextStyle( |           hint: 'Tap to Load', | ||||||
|                                             color: Colors.white, |         ), | ||||||
|                                             fontSize: 11, |       ), | ||||||
|                                           ), |     ); | ||||||
|                                         ), |   } | ||||||
|                                       ], | } | ||||||
|                                     ), |  | ||||||
|                                   ).padding(horizontal: 24, vertical: 16), | class _OverlayCard extends StatelessWidget { | ||||||
|                                 ), |   final IconData icon; | ||||||
|                               ), |   final String title; | ||||||
|                             ], |   final String subtitle; | ||||||
|                           ), |   final String hint; | ||||||
|                         ), |  | ||||||
|                       ), |   const _OverlayCard({ | ||||||
|             ), |     required this.icon, | ||||||
|           ), |     required this.title, | ||||||
|           // When revealed (no blur), show a small control at top-left to re-enable blur |     required this.subtitle, | ||||||
|           if (showMature.value) |     required this.hint, | ||||||
|             Positioned( |   }); | ||||||
|               top: 3, |  | ||||||
|               left: 4, |   @override | ||||||
|               child: IconButton( |   Widget build(BuildContext context) { | ||||||
|                 iconSize: 16, |     return Container( | ||||||
|                 constraints: const BoxConstraints(), |       margin: const EdgeInsets.all(12), | ||||||
|                 icon: const Icon(Icons.visibility_off, color: Colors.white), |       padding: const EdgeInsets.symmetric(horizontal: 12, vertical: 8), | ||||||
|                 tooltip: 'Blur content', |       decoration: BoxDecoration( | ||||||
|                 onPressed: () { |         color: Colors.black54, | ||||||
|                   showMature.value = false; |         borderRadius: BorderRadius.circular(12), | ||||||
|                 }, |       ), | ||||||
|               ), |       constraints: const BoxConstraints(maxWidth: 280), | ||||||
|             ), |       child: Column( | ||||||
|         ], |         mainAxisSize: MainAxisSize.min, | ||||||
|       ); |         children: [ | ||||||
|     } |           Icon(icon, color: Colors.white, size: 24), | ||||||
|  |           const Gap(4), | ||||||
|     if (onTap != null) { |           Text( | ||||||
|       return InkWell( |             title, | ||||||
|         borderRadius: const BorderRadius.all(Radius.circular(16)), |             style: const TextStyle( | ||||||
|         onTap: () { |               color: Colors.white, | ||||||
|           if (!showMature.value) { |               fontWeight: FontWeight.w600, | ||||||
|             showMature.value = true; |             ), | ||||||
|           } else { |             textAlign: TextAlign.center, | ||||||
|             onTap?.call(); |           ), | ||||||
|           } |           Text( | ||||||
|         }, |             subtitle, | ||||||
|         child: content, |             style: const TextStyle(color: Colors.white, fontSize: 13), | ||||||
|       ); |           ), | ||||||
|     } |           const Gap(4), | ||||||
|  |           Text(hint, style: const TextStyle(color: Colors.white, fontSize: 11)), | ||||||
|     return content; |         ], | ||||||
|  |       ), | ||||||
|  |     ); | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,15 +1,20 @@ | |||||||
| import 'dart:math' as math; | import 'dart:math' as math; | ||||||
|  |  | ||||||
| import 'package:cached_network_image/cached_network_image.dart'; | import 'package:cached_network_image/cached_network_image.dart'; | ||||||
|  | import 'package:easy_localization/easy_localization.dart'; | ||||||
| import 'package:flutter/material.dart'; | import 'package:flutter/material.dart'; | ||||||
| import 'package:flutter_hooks/flutter_hooks.dart'; | import 'package:flutter_hooks/flutter_hooks.dart'; | ||||||
|  | import 'package:gap/gap.dart'; | ||||||
| import 'package:hooks_riverpod/hooks_riverpod.dart'; | import 'package:hooks_riverpod/hooks_riverpod.dart'; | ||||||
| import 'package:island/models/file.dart'; | import 'package:island/models/file.dart'; | ||||||
| import 'package:island/pods/config.dart'; | import 'package:island/pods/config.dart'; | ||||||
| import 'package:island/services/time.dart'; | import 'package:island/services/time.dart'; | ||||||
|  | import 'package:island/utils/format.dart'; | ||||||
| import 'package:island/widgets/content/audio.dart'; | import 'package:island/widgets/content/audio.dart'; | ||||||
| import 'package:material_symbols_icons/symbols.dart'; | import 'package:material_symbols_icons/symbols.dart'; | ||||||
| import 'package:styled_widget/styled_widget.dart'; | import 'package:styled_widget/styled_widget.dart'; | ||||||
|  | import 'package:url_launcher/url_launcher_string.dart'; | ||||||
|  | import 'package:island/widgets/data_saving_gate.dart'; | ||||||
|  |  | ||||||
| import 'image.dart'; | import 'image.dart'; | ||||||
| import 'video.dart'; | import 'video.dart'; | ||||||
| @@ -19,48 +24,97 @@ class CloudFileWidget extends HookConsumerWidget { | |||||||
|   final BoxFit fit; |   final BoxFit fit; | ||||||
|   final String? heroTag; |   final String? heroTag; | ||||||
|   final bool noBlurhash; |   final bool noBlurhash; | ||||||
|  |   final bool useInternalGate; | ||||||
|   const CloudFileWidget({ |   const CloudFileWidget({ | ||||||
|     super.key, |     super.key, | ||||||
|     required this.item, |     required this.item, | ||||||
|     this.fit = BoxFit.cover, |     this.fit = BoxFit.cover, | ||||||
|     this.heroTag, |     this.heroTag, | ||||||
|     this.noBlurhash = false, |     this.noBlurhash = false, | ||||||
|  |     this.useInternalGate = true, | ||||||
|   }); |   }); | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   Widget build(BuildContext context, WidgetRef ref) { |   Widget build(BuildContext context, WidgetRef ref) { | ||||||
|  |     final dataSaving = ref.watch( | ||||||
|  |       appSettingsNotifierProvider.select((s) => s.dataSavingMode), | ||||||
|  |     ); | ||||||
|     final serverUrl = ref.watch(serverUrlProvider); |     final serverUrl = ref.watch(serverUrlProvider); | ||||||
|     final uri = '$serverUrl/drive/files/${item.id}'; |     final uri = '$serverUrl/drive/files/${item.id}'; | ||||||
|  |  | ||||||
|     var ratio = |     final unlocked = useState(false); | ||||||
|         item.fileMeta?['ratio'] is num |  | ||||||
|             ? item.fileMeta!['ratio'].toDouble() |     final meta = item.fileMeta is Map ? (item.fileMeta as Map) : const {}; | ||||||
|             : 1.0; |     final blurHash = noBlurhash ? null : (meta['blur'] as String?); | ||||||
|  |     var ratio = meta['ratio'] is num ? (meta['ratio'] as num).toDouble() : 1.0; | ||||||
|     if (ratio == 0) ratio = 1.0; |     if (ratio == 0) ratio = 1.0; | ||||||
|  |  | ||||||
|  |     Widget cloudImage() => UniversalImage(uri: uri, blurHash: blurHash, fit: fit); | ||||||
|  |     Widget cloudVideo() => CloudVideoWidget(item: item); | ||||||
|  |  | ||||||
|  |     Widget dataPlaceHolder(IconData icon) => _DataSavingPlaceholder( | ||||||
|  |           icon: icon, | ||||||
|  |           onTap: () { | ||||||
|  |             unlocked.value = true; | ||||||
|  |           }, | ||||||
|  |         ); | ||||||
|  |  | ||||||
|     var content = switch (item.mimeType?.split('/').firstOrNull) { |     var content = switch (item.mimeType?.split('/').firstOrNull) { | ||||||
|       "image" => AspectRatio( |       'image' => AspectRatio( | ||||||
|         aspectRatio: ratio, |           aspectRatio: ratio, | ||||||
|         child: UniversalImage( |           child: (useInternalGate && dataSaving && !unlocked.value) ? dataPlaceHolder(Symbols.image) : cloudImage(), | ||||||
|           uri: uri, |  | ||||||
|           blurHash: |  | ||||||
|               noBlurhash |  | ||||||
|                   ? null |  | ||||||
|                   : (item.fileMeta is String ? item.fileMeta!['blur'] : null), |  | ||||||
|         ), |         ), | ||||||
|       ), |       'video' => AspectRatio( | ||||||
|       "video" => AspectRatio( |           aspectRatio: ratio, | ||||||
|         aspectRatio: ratio, |           child: (useInternalGate && dataSaving && !unlocked.value) ? dataPlaceHolder(Symbols.play_arrow) : cloudVideo(), | ||||||
|         child: CloudVideoWidget(item: item), |         ), | ||||||
|       ), |       'audio' => Center( | ||||||
|       "audio" => Center( |           child: ConstrainedBox( | ||||||
|         child: ConstrainedBox( |             constraints: BoxConstraints( | ||||||
|           constraints: BoxConstraints( |               maxWidth: math.min(360, MediaQuery.of(context).size.width * 0.8), | ||||||
|             maxWidth: math.min(360, MediaQuery.of(context).size.width * 0.8), |             ), | ||||||
|  |             child: UniversalAudio(uri: uri, filename: item.name), | ||||||
|           ), |           ), | ||||||
|           child: UniversalAudio(uri: uri, filename: item.name), |  | ||||||
|         ), |         ), | ||||||
|       ), |       _ => Column( | ||||||
|       _ => Text('Unable render for ${item.mimeType}'), |           mainAxisSize: MainAxisSize.min, | ||||||
|  |           mainAxisAlignment: MainAxisAlignment.center, | ||||||
|  |           children: [ | ||||||
|  |             Icon( | ||||||
|  |               Symbols.insert_drive_file, | ||||||
|  |               size: 48, | ||||||
|  |               color: Theme.of(context).colorScheme.onSurfaceVariant, | ||||||
|  |             ), | ||||||
|  |             const Gap(8), | ||||||
|  |             Text( | ||||||
|  |               item.name, | ||||||
|  |               maxLines: 1, | ||||||
|  |               overflow: TextOverflow.ellipsis, | ||||||
|  |               style: TextStyle( | ||||||
|  |                 fontSize: 14, | ||||||
|  |                 color: Theme.of(context).colorScheme.onSurfaceVariant, | ||||||
|  |               ), | ||||||
|  |             ), | ||||||
|  |             Text( | ||||||
|  |               formatFileSize(item.size), | ||||||
|  |               style: TextStyle( | ||||||
|  |                 fontSize: 12, | ||||||
|  |                 color: Theme.of(context).colorScheme.onSurfaceVariant, | ||||||
|  |               ), | ||||||
|  |             ), | ||||||
|  |             const Gap(8), | ||||||
|  |             TextButton.icon( | ||||||
|  |               onPressed: () { | ||||||
|  |                 launchUrlString( | ||||||
|  |                   'https://fs.solian.app/files/${item.id}', | ||||||
|  |                   mode: LaunchMode.externalApplication, | ||||||
|  |                 ); | ||||||
|  |               }, | ||||||
|  |               icon: const Icon(Symbols.launch), | ||||||
|  |               label: Text('openInBrowser').tr(), | ||||||
|  |             ), | ||||||
|  |           ], | ||||||
|  |         ).padding(all: 8), | ||||||
|     }; |     }; | ||||||
|  |  | ||||||
|     if (heroTag != null) { |     if (heroTag != null) { | ||||||
| @@ -71,6 +125,35 @@ class CloudFileWidget extends HookConsumerWidget { | |||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | class _DataSavingPlaceholder extends StatelessWidget { | ||||||
|  |   final IconData icon; | ||||||
|  |   final VoidCallback onTap; | ||||||
|  |   const _DataSavingPlaceholder({required this.icon, required this.onTap}); | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Widget build(BuildContext context) { | ||||||
|  |     return GestureDetector( | ||||||
|  |       onTap: onTap, | ||||||
|  |       child: Container( | ||||||
|  |         color: Colors.black26, | ||||||
|  |         alignment: Alignment.center, | ||||||
|  |         child: Column( | ||||||
|  |           mainAxisSize: MainAxisSize.min, | ||||||
|  |           children: [ | ||||||
|  |             Icon(icon, size: 36, | ||||||
|  |               color: Theme.of(context).colorScheme.onSurfaceVariant), | ||||||
|  |             const Gap(8), | ||||||
|  |             Text( | ||||||
|  |               'dataSavingHint'.tr(), | ||||||
|  |               style: Theme.of(context).textTheme.bodySmall, | ||||||
|  |               textAlign: TextAlign.center, | ||||||
|  |             ), | ||||||
|  |           ], | ||||||
|  |         ), | ||||||
|  |       ), | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  | } | ||||||
| class CloudVideoWidget extends HookConsumerWidget { | class CloudVideoWidget extends HookConsumerWidget { | ||||||
|   final SnCloudFile item; |   final SnCloudFile item; | ||||||
|   const CloudVideoWidget({super.key, required this.item}); |   const CloudVideoWidget({super.key, required this.item}); | ||||||
| @@ -269,32 +352,35 @@ class ProfilePictureWidget extends ConsumerWidget { | |||||||
|     this.fallbackColor, |     this.fallbackColor, | ||||||
|   }); |   }); | ||||||
|  |  | ||||||
|   @override | @override | ||||||
|   Widget build(BuildContext context, WidgetRef ref) { |   Widget build(BuildContext context, WidgetRef ref) { | ||||||
|     final serverUrl = ref.watch(serverUrlProvider); |     final serverUrl = ref.watch(serverUrlProvider); | ||||||
|     final uri = '$serverUrl/drive/files/${file?.id ?? fileId}'; |     final String? id = file?.id ?? fileId; | ||||||
|  |  | ||||||
|  |     final fallback = Icon( | ||||||
|  |       fallbackIcon ?? Symbols.account_circle, | ||||||
|  |       size: radius, | ||||||
|  |       color: fallbackColor ?? Theme.of(context).colorScheme.onPrimaryContainer, | ||||||
|  |     ).center(); | ||||||
|  |  | ||||||
|     return ClipRRect( |     return ClipRRect( | ||||||
|       borderRadius: |       borderRadius: borderRadius == null | ||||||
|           borderRadius == null |           ? BorderRadius.all(Radius.circular(radius)) | ||||||
|               ? BorderRadius.all(Radius.circular(radius)) |           : BorderRadius.all(Radius.circular(borderRadius!)), | ||||||
|               : BorderRadius.all(Radius.circular(borderRadius!)), |  | ||||||
|       child: Container( |       child: Container( | ||||||
|         width: radius * 2, |         width: radius * 2, | ||||||
|         height: radius * 2, |         height: radius * 2, | ||||||
|         color: Theme.of(context).colorScheme.primaryContainer, |         color: Theme.of(context).colorScheme.primaryContainer, | ||||||
|         child: |         child: id == null | ||||||
|             file != null |             ? fallback | ||||||
|                 ? CloudFileWidget(item: file!, fit: BoxFit.cover) |             : DataSavingGate( | ||||||
|                 : fileId == null |                 bypass: true, | ||||||
|                 ? Icon( |                 placeholder: fallback, | ||||||
|                   fallbackIcon ?? Symbols.account_circle, |                 content: () => UniversalImage( | ||||||
|                   size: radius, |                   uri: '$serverUrl/drive/files/$id', | ||||||
|                   color: |                   fit: BoxFit.cover, | ||||||
|                       fallbackColor ?? |                 ), | ||||||
|                       Theme.of(context).colorScheme.onPrimaryContainer, |               ), | ||||||
|                 ).center() |  | ||||||
|                 : UniversalImage(uri: uri, fit: BoxFit.cover), |  | ||||||
|       ), |       ), | ||||||
|     ); |     ); | ||||||
|   } |   } | ||||||
|   | |||||||
| @@ -52,12 +52,10 @@ class UniversalImage extends StatelessWidget { | |||||||
|             }, |             }, | ||||||
|             errorWidget: (context, url, error) { |             errorWidget: (context, url, error) { | ||||||
|               return Image.asset( |               return Image.asset( | ||||||
|                 'assets/images/media-offline.png', |                 'assets/images/media-offline.jpg', | ||||||
|                 fit: BoxFit.cover, |                 fit: BoxFit.cover, | ||||||
|  |                 key: Key('image-broke-$uri'), | ||||||
|               ); |               ); | ||||||
|               // return const Center( |  | ||||||
|               //   child: Icon(Icons.broken_image, color: Colors.white, size: 16), |  | ||||||
|               // ); |  | ||||||
|             }, |             }, | ||||||
|           ), |           ), | ||||||
|         ], |         ], | ||||||
|   | |||||||
							
								
								
									
										27
									
								
								lib/widgets/data_saving_gate.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								lib/widgets/data_saving_gate.dart
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,27 @@ | |||||||
|  | import 'package:flutter/material.dart'; | ||||||
|  | import 'package:hooks_riverpod/hooks_riverpod.dart'; | ||||||
|  | import 'package:island/pods/config.dart'; | ||||||
|  |  | ||||||
|  |  | ||||||
|  | typedef WidgetBuilder0 = Widget Function(); | ||||||
|  |  | ||||||
|  | class DataSavingGate extends ConsumerWidget { | ||||||
|  |   final bool bypass; | ||||||
|  |   final WidgetBuilder0 content; | ||||||
|  |   final Widget placeholder; | ||||||
|  |  | ||||||
|  |   const DataSavingGate({ | ||||||
|  |     super.key, | ||||||
|  |     required this.bypass, | ||||||
|  |     required this.content, | ||||||
|  |     required this.placeholder, | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Widget build(BuildContext context, WidgetRef ref) { | ||||||
|  |     final dataSaving = | ||||||
|  |         ref.watch(appSettingsNotifierProvider.select((s) => s.dataSavingMode)); | ||||||
|  |     if (bypass || !dataSaving) return content(); | ||||||
|  |     return placeholder; | ||||||
|  |   } | ||||||
|  | } | ||||||
							
								
								
									
										67
									
								
								lib/widgets/extended_refresh_indicator.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										67
									
								
								lib/widgets/extended_refresh_indicator.dart
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,67 @@ | |||||||
|  | import 'package:flutter/material.dart'; | ||||||
|  | import 'package:flutter/services.dart'; | ||||||
|  |  | ||||||
|  | class RefreshIntent extends Intent { | ||||||
|  |   const RefreshIntent(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class ExtendedRefreshIndicator extends StatefulWidget { | ||||||
|  |   final Widget child; | ||||||
|  |   final RefreshCallback onRefresh; | ||||||
|  |  | ||||||
|  |   const ExtendedRefreshIndicator({ | ||||||
|  |     super.key, | ||||||
|  |     required this.child, | ||||||
|  |     required this.onRefresh, | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   State<ExtendedRefreshIndicator> createState() => | ||||||
|  |       _ExtendedRefreshIndicatorState(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class _ExtendedRefreshIndicatorState extends State<ExtendedRefreshIndicator> { | ||||||
|  |   late final FocusNode _focusNode; | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   void initState() { | ||||||
|  |     super.initState(); | ||||||
|  |     _focusNode = FocusNode(); | ||||||
|  |     WidgetsBinding.instance.addPostFrameCallback((_) { | ||||||
|  |       _focusNode.requestFocus(); | ||||||
|  |     }); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   void dispose() { | ||||||
|  |     _focusNode.dispose(); | ||||||
|  |     super.dispose(); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Widget build(BuildContext context) { | ||||||
|  |     return Shortcuts( | ||||||
|  |       shortcuts: <LogicalKeySet, Intent>{ | ||||||
|  |         LogicalKeySet(LogicalKeyboardKey.control, LogicalKeyboardKey.keyR): | ||||||
|  |             const RefreshIntent(), | ||||||
|  |         LogicalKeySet(LogicalKeyboardKey.meta, LogicalKeyboardKey.keyR): | ||||||
|  |             const RefreshIntent(), | ||||||
|  |         LogicalKeySet(LogicalKeyboardKey.f5): const RefreshIntent(), | ||||||
|  |       }, | ||||||
|  |       child: Actions( | ||||||
|  |         actions: <Type, Action<Intent>>{ | ||||||
|  |           RefreshIntent: CallbackAction<RefreshIntent>( | ||||||
|  |             onInvoke: (RefreshIntent intent) => widget.onRefresh(), | ||||||
|  |           ), | ||||||
|  |         }, | ||||||
|  |         child: Focus( | ||||||
|  |           focusNode: _focusNode, | ||||||
|  |           child: RefreshIndicator( | ||||||
|  |             onRefresh: widget.onRefresh, | ||||||
|  |             child: widget.child, | ||||||
|  |           ), | ||||||
|  |         ), | ||||||
|  |       ), | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  | } | ||||||
| @@ -1,6 +1,7 @@ | |||||||
| import 'package:collection/collection.dart'; | import 'package:collection/collection.dart'; | ||||||
| import 'package:dio/dio.dart'; | import 'package:dio/dio.dart'; | ||||||
| import 'package:easy_localization/easy_localization.dart'; | import 'package:easy_localization/easy_localization.dart'; | ||||||
|  | import 'package:file_picker/file_picker.dart'; | ||||||
| import 'package:flutter/material.dart'; | import 'package:flutter/material.dart'; | ||||||
| import 'package:flutter/services.dart'; | import 'package:flutter/services.dart'; | ||||||
| import 'package:hooks_riverpod/hooks_riverpod.dart'; | import 'package:hooks_riverpod/hooks_riverpod.dart'; | ||||||
| @@ -388,26 +389,32 @@ class ComposeLogic { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   static Future<void> pickPhotoMedia(WidgetRef ref, ComposeState state) async { |   static Future<void> pickPhotoMedia(WidgetRef ref, ComposeState state) async { | ||||||
|     final result = await ref |     final result = await FilePicker.platform.pickFiles( | ||||||
|         .watch(imagePickerProvider) |       type: FileType.image, | ||||||
|         .pickMultiImage(requestFullMetadata: true); |       allowMultiple: true, | ||||||
|     if (result.isEmpty) return; |       allowCompression: false, | ||||||
|  |     ); | ||||||
|  |     if (result == null || result.count == 0) return; | ||||||
|     state.attachments.value = [ |     state.attachments.value = [ | ||||||
|       ...state.attachments.value, |       ...state.attachments.value, | ||||||
|       ...result.map( |       ...result.files.map( | ||||||
|         (e) => UniversalFile(data: e, type: UniversalFileType.image), |         (e) => UniversalFile(data: e.xFile, type: UniversalFileType.image), | ||||||
|       ), |       ), | ||||||
|     ]; |     ]; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   static Future<void> pickVideoMedia(WidgetRef ref, ComposeState state) async { |   static Future<void> pickVideoMedia(WidgetRef ref, ComposeState state) async { | ||||||
|     final result = await ref |     final result = await FilePicker.platform.pickFiles( | ||||||
|         .watch(imagePickerProvider) |       type: FileType.video, | ||||||
|         .pickVideo(source: ImageSource.gallery); |       allowMultiple: true, | ||||||
|     if (result == null) return; |       allowCompression: false, | ||||||
|  |     ); | ||||||
|  |     if (result == null || result.count == 0) return; | ||||||
|     state.attachments.value = [ |     state.attachments.value = [ | ||||||
|       ...state.attachments.value, |       ...state.attachments.value, | ||||||
|       UniversalFile(data: result, type: UniversalFileType.video), |       ...result.files.map( | ||||||
|  |         (e) => UniversalFile(data: e.xFile, type: UniversalFileType.video), | ||||||
|  |       ), | ||||||
|     ]; |     ]; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,3 +1,4 @@ | |||||||
|  | import 'package:easy_localization/easy_localization.dart'; | ||||||
| import 'package:flutter/material.dart'; | import 'package:flutter/material.dart'; | ||||||
| import 'package:flutter_hooks/flutter_hooks.dart'; | import 'package:flutter_hooks/flutter_hooks.dart'; | ||||||
| import 'package:hooks_riverpod/hooks_riverpod.dart'; | import 'package:hooks_riverpod/hooks_riverpod.dart'; | ||||||
| @@ -47,7 +48,7 @@ class PostFeaturedList extends HookConsumerWidget { | |||||||
|         'PostFeaturedList: isCollapsed changed to ${isCollapsed.value}', |         'PostFeaturedList: isCollapsed changed to ${isCollapsed.value}', | ||||||
|       ); |       ); | ||||||
|       return null; |       return null; | ||||||
|     }, [isCollapsed.value]); |     }, [isCollapsed]); | ||||||
|  |  | ||||||
|     useEffect(() { |     useEffect(() { | ||||||
|       if (featuredPostsAsync.hasValue && featuredPostsAsync.value!.isNotEmpty) { |       if (featuredPostsAsync.hasValue && featuredPostsAsync.value!.isNotEmpty) { | ||||||
| @@ -92,7 +93,7 @@ class PostFeaturedList extends HookConsumerWidget { | |||||||
|         ); |         ); | ||||||
|       } |       } | ||||||
|       return null; |       return null; | ||||||
|     }, [featuredPostsAsync.value]); |     }, [featuredPostsAsync]); | ||||||
|  |  | ||||||
|     return ClipRRect( |     return ClipRRect( | ||||||
|       borderRadius: const BorderRadius.all(Radius.circular(8)), |       borderRadius: const BorderRadius.all(Radius.circular(8)), | ||||||
| @@ -105,7 +106,7 @@ class PostFeaturedList extends HookConsumerWidget { | |||||||
|               spacing: 8, |               spacing: 8, | ||||||
|               children: [ |               children: [ | ||||||
|                 const Icon(Symbols.highlight), |                 const Icon(Symbols.highlight), | ||||||
|                 Text('Highlight Posts'), |                 const Text('highlightPost').tr(), | ||||||
|                 Spacer(), |                 Spacer(), | ||||||
|                 IconButton( |                 IconButton( | ||||||
|                   padding: EdgeInsets.zero, |                   padding: EdgeInsets.zero, | ||||||
|   | |||||||
| @@ -18,6 +18,7 @@ import 'package:island/screens/posts/compose.dart'; | |||||||
| import 'package:island/widgets/alert.dart'; | import 'package:island/widgets/alert.dart'; | ||||||
| import 'package:island/widgets/content/markdown.dart'; | import 'package:island/widgets/content/markdown.dart'; | ||||||
| import 'package:island/widgets/post/post_item_screenshot.dart'; | import 'package:island/widgets/post/post_item_screenshot.dart'; | ||||||
|  | import 'package:island/widgets/post/post_pin_sheet.dart'; | ||||||
| import 'package:island/widgets/post/post_shared.dart'; | import 'package:island/widgets/post/post_shared.dart'; | ||||||
| import 'package:island/widgets/safety/abuse_report_helper.dart'; | import 'package:island/widgets/safety/abuse_report_helper.dart'; | ||||||
| import 'package:island/widgets/share/share_sheet.dart'; | import 'package:island/widgets/share/share_sheet.dart'; | ||||||
| @@ -35,6 +36,7 @@ class PostActionableItem extends HookConsumerWidget { | |||||||
|   final bool isShowReference; |   final bool isShowReference; | ||||||
|   final bool isEmbedReply; |   final bool isEmbedReply; | ||||||
|   final bool isEmbedOpenable; |   final bool isEmbedOpenable; | ||||||
|  |   final bool isCompact; | ||||||
|   final double? borderRadius; |   final double? borderRadius; | ||||||
|   final VoidCallback? onRefresh; |   final VoidCallback? onRefresh; | ||||||
|   final Function(SnPost)? onUpdate; |   final Function(SnPost)? onUpdate; | ||||||
| @@ -47,6 +49,7 @@ class PostActionableItem extends HookConsumerWidget { | |||||||
|     this.isShowReference = true, |     this.isShowReference = true, | ||||||
|     this.isEmbedReply = true, |     this.isEmbedReply = true, | ||||||
|     this.isEmbedOpenable = false, |     this.isEmbedOpenable = false, | ||||||
|  |     this.isCompact = false, | ||||||
|     this.borderRadius, |     this.borderRadius, | ||||||
|     this.onRefresh, |     this.onRefresh, | ||||||
|     this.onUpdate, |     this.onUpdate, | ||||||
| @@ -75,6 +78,7 @@ class PostActionableItem extends HookConsumerWidget { | |||||||
|         isEmbedReply: isEmbedReply, |         isEmbedReply: isEmbedReply, | ||||||
|         isEmbedOpenable: isEmbedOpenable, |         isEmbedOpenable: isEmbedOpenable, | ||||||
|         isTextSelectable: false, |         isTextSelectable: false, | ||||||
|  |         isCompact: isCompact, | ||||||
|         onRefresh: onRefresh, |         onRefresh: onRefresh, | ||||||
|         onUpdate: onUpdate, |         onUpdate: onUpdate, | ||||||
|         onOpen: onOpen, |         onOpen: onOpen, | ||||||
| @@ -202,6 +206,45 @@ class PostActionableItem extends HookConsumerWidget { | |||||||
|                 ); |                 ); | ||||||
|               }, |               }, | ||||||
|             ), |             ), | ||||||
|  |             if (isAuthor && item.pinMode == null) | ||||||
|  |               MenuAction( | ||||||
|  |                 title: 'pinPost'.tr(), | ||||||
|  |                 image: MenuImage.icon(Symbols.keep), | ||||||
|  |                 callback: () { | ||||||
|  |                   showModalBottomSheet( | ||||||
|  |                     context: context, | ||||||
|  |                     isScrollControlled: true, | ||||||
|  |                     builder: (context) => PostPinSheet(post: item), | ||||||
|  |                   ).then((value) { | ||||||
|  |                     if (value is int) { | ||||||
|  |                       onUpdate?.call(item.copyWith(pinMode: value)); | ||||||
|  |                     } | ||||||
|  |                   }); | ||||||
|  |                 }, | ||||||
|  |               ) | ||||||
|  |             else if (isAuthor && item.pinMode != null) | ||||||
|  |               MenuAction( | ||||||
|  |                 title: 'unpinPost'.tr(), | ||||||
|  |                 image: MenuImage.icon(Symbols.keep_off), | ||||||
|  |                 callback: () { | ||||||
|  |                   showConfirmAlert('unpinPostHint'.tr(), 'unpinPost'.tr()).then( | ||||||
|  |                     (confirm) async { | ||||||
|  |                       if (confirm) { | ||||||
|  |                         final client = ref.watch(apiClientProvider); | ||||||
|  |                         try { | ||||||
|  |                           if (context.mounted) showLoadingModal(context); | ||||||
|  |                           await client.delete('/sphere/posts/${item.id}/pin'); | ||||||
|  |                           onUpdate?.call(item.copyWith(pinMode: null)); | ||||||
|  |                         } catch (err) { | ||||||
|  |                           showErrorAlert(err); | ||||||
|  |                         } finally { | ||||||
|  |                           if (context.mounted) hideLoadingModal(context); | ||||||
|  |                         } | ||||||
|  |                       } | ||||||
|  |                     }, | ||||||
|  |                   ); | ||||||
|  |                 }, | ||||||
|  |               ), | ||||||
|             MenuSeparator(), |             MenuSeparator(), | ||||||
|             MenuAction( |             MenuAction( | ||||||
|               title: 'share'.tr(), |               title: 'share'.tr(), | ||||||
| @@ -258,6 +301,7 @@ class PostItem extends HookConsumerWidget { | |||||||
|   final bool isEmbedOpenable; |   final bool isEmbedOpenable; | ||||||
|   final bool isTextSelectable; |   final bool isTextSelectable; | ||||||
|   final bool isTranslatable; |   final bool isTranslatable; | ||||||
|  |   final bool isCompact; | ||||||
|   final VoidCallback? onRefresh; |   final VoidCallback? onRefresh; | ||||||
|   final Function(SnPost)? onUpdate; |   final Function(SnPost)? onUpdate; | ||||||
|   final VoidCallback? onOpen; |   final VoidCallback? onOpen; | ||||||
| @@ -271,6 +315,7 @@ class PostItem extends HookConsumerWidget { | |||||||
|     this.isEmbedOpenable = false, |     this.isEmbedOpenable = false, | ||||||
|     this.isTextSelectable = true, |     this.isTextSelectable = true, | ||||||
|     this.isTranslatable = true, |     this.isTranslatable = true, | ||||||
|  |     this.isCompact = false, | ||||||
|     this.onRefresh, |     this.onRefresh, | ||||||
|     this.onUpdate, |     this.onUpdate, | ||||||
|     this.onOpen, |     this.onOpen, | ||||||
| @@ -300,7 +345,14 @@ class PostItem extends HookConsumerWidget { | |||||||
|             final delta = isRemoving ? -1 : 1; |             final delta = isRemoving ? -1 : 1; | ||||||
|             final reactionsCount = Map<String, int>.from(item.reactionsCount); |             final reactionsCount = Map<String, int>.from(item.reactionsCount); | ||||||
|             reactionsCount[symbol] = (reactionsCount[symbol] ?? 0) + delta; |             reactionsCount[symbol] = (reactionsCount[symbol] ?? 0) + delta; | ||||||
|             onUpdate?.call(item.copyWith(reactionsCount: reactionsCount)); |             final reactionsMade = Map<String, bool>.from(item.reactionsMade); | ||||||
|  |             reactionsMade[symbol] = delta == 1 ? true : false; | ||||||
|  |             onUpdate?.call( | ||||||
|  |               item.copyWith( | ||||||
|  |                 reactionsCount: reactionsCount, | ||||||
|  |                 reactionsMade: reactionsMade, | ||||||
|  |               ), | ||||||
|  |             ); | ||||||
|             HapticFeedback.heavyImpact(); |             HapticFeedback.heavyImpact(); | ||||||
|           }); |           }); | ||||||
|       reacting.value = false; |       reacting.value = false; | ||||||
| @@ -418,54 +470,64 @@ class PostItem extends HookConsumerWidget { | |||||||
|         PostHeader( |         PostHeader( | ||||||
|           item: item, |           item: item, | ||||||
|           isFullPost: isFullPost, |           isFullPost: isFullPost, | ||||||
|  |           isCompact: isCompact, | ||||||
|           renderingPadding: renderingPadding, |           renderingPadding: renderingPadding, | ||||||
|           trailing: IconButton( |           trailing: | ||||||
|             icon: |               isCompact | ||||||
|                 mostReaction == null |                   ? null | ||||||
|                     ? const Icon(Symbols.add_reaction) |                   : IconButton( | ||||||
|                     : Badge( |                     icon: | ||||||
|                       label: Center( |                         mostReaction == null | ||||||
|                         child: Text( |                             ? const Icon(Symbols.add_reaction) | ||||||
|                           'x${item.reactionsCount[mostReaction]}', |                             : Badge( | ||||||
|                           style: const TextStyle(fontSize: 11), |                               label: Center( | ||||||
|                           textAlign: TextAlign.center, |                                 child: Text( | ||||||
|                         ), |                                   'x${item.reactionsCount[mostReaction]}', | ||||||
|                       ), |                                   style: const TextStyle(fontSize: 11), | ||||||
|                       offset: const Offset(4, 20), |                                   textAlign: TextAlign.center, | ||||||
|                       backgroundColor: Theme.of( |                                 ), | ||||||
|                         context, |                               ), | ||||||
|                       ).colorScheme.primary.withOpacity(0.75), |                               offset: const Offset(4, 20), | ||||||
|                       textColor: Theme.of(context).colorScheme.onPrimary, |                               backgroundColor: Theme.of( | ||||||
|                       child: Text( |                                 context, | ||||||
|                         kReactionTemplates[mostReaction]?.icon ?? '', |                               ).colorScheme.primary.withOpacity(0.75), | ||||||
|                         style: const TextStyle(fontSize: 20), |                               textColor: | ||||||
|  |                                   Theme.of(context).colorScheme.onPrimary, | ||||||
|  |                               child: Text( | ||||||
|  |                                 kReactionTemplates[mostReaction]?.icon ?? '', | ||||||
|  |                                 style: const TextStyle(fontSize: 20), | ||||||
|  |                               ), | ||||||
|  |                             ), | ||||||
|  |                     style: ButtonStyle( | ||||||
|  |                       backgroundColor: WidgetStatePropertyAll( | ||||||
|  |                         (item.reactionsMade[mostReaction] ?? false) | ||||||
|  |                             ? Theme.of( | ||||||
|  |                               context, | ||||||
|  |                             ).colorScheme.primary.withOpacity(0.5) | ||||||
|  |                             : null, | ||||||
|                       ), |                       ), | ||||||
|                     ), |                     ), | ||||||
|             style: ButtonStyle( |                     onPressed: () { | ||||||
|               backgroundColor: WidgetStatePropertyAll( |                       showModalBottomSheet( | ||||||
|                 (item.reactionsMade[mostReaction] ?? false) |                         context: context, | ||||||
|                     ? Theme.of(context).colorScheme.primary.withOpacity(0.5) |                         useRootNavigator: true, | ||||||
|                     : null, |                         builder: (BuildContext context) { | ||||||
|               ), |                           return _PostReactionSheet( | ||||||
|             ), |                             reactionsCount: item.reactionsCount, | ||||||
|             onPressed: () { |                             reactionsMade: item.reactionsMade, | ||||||
|               showModalBottomSheet( |                             onReact: (symbol, attitude) { | ||||||
|                 context: context, |                               reactPost(symbol, attitude); | ||||||
|                 useRootNavigator: true, |                             }, | ||||||
|                 builder: (BuildContext context) { |                           ); | ||||||
|                   return _PostReactionSheet( |                         }, | ||||||
|                     reactionsCount: item.reactionsCount, |                       ); | ||||||
|                     reactionsMade: item.reactionsMade, |  | ||||||
|                     onReact: (symbol, attitude) { |  | ||||||
|                       reactPost(symbol, attitude); |  | ||||||
|                     }, |                     }, | ||||||
|                   ); |                     padding: EdgeInsets.zero, | ||||||
|                 }, |                     visualDensity: const VisualDensity( | ||||||
|               ); |                       horizontal: -3, | ||||||
|             }, |                       vertical: -3, | ||||||
|             padding: EdgeInsets.zero, |                     ), | ||||||
|             visualDensity: const VisualDensity(horizontal: -3, vertical: -3), |                   ), | ||||||
|           ), |  | ||||||
|         ), |         ), | ||||||
|         PostBody( |         PostBody( | ||||||
|           item: item, |           item: item, | ||||||
|   | |||||||
| @@ -10,6 +10,7 @@ import 'package:island/widgets/alert.dart'; | |||||||
| import 'package:island/widgets/post/post_item.dart'; | import 'package:island/widgets/post/post_item.dart'; | ||||||
| import 'package:island/widgets/post/post_shared.dart'; | import 'package:island/widgets/post/post_shared.dart'; | ||||||
| import 'package:material_symbols_icons/symbols.dart'; | import 'package:material_symbols_icons/symbols.dart'; | ||||||
|  | import 'package:styled_widget/styled_widget.dart'; | ||||||
| import 'package:super_context_menu/super_context_menu.dart'; | import 'package:super_context_menu/super_context_menu.dart'; | ||||||
|  |  | ||||||
| class PostItemCreator extends HookConsumerWidget { | class PostItemCreator extends HookConsumerWidget { | ||||||
| @@ -33,7 +34,7 @@ class PostItemCreator extends HookConsumerWidget { | |||||||
|   @override |   @override | ||||||
|   Widget build(BuildContext context, WidgetRef ref) { |   Widget build(BuildContext context, WidgetRef ref) { | ||||||
|     final renderingPadding = |     final renderingPadding = | ||||||
|         padding ?? const EdgeInsets.symmetric(horizontal: 16, vertical: 16); |         padding ?? const EdgeInsets.symmetric(horizontal: 8, vertical: 8); | ||||||
|  |  | ||||||
|     return ContextMenuWidget( |     return ContextMenuWidget( | ||||||
|       menuProvider: (_) { |       menuProvider: (_) { | ||||||
| @@ -97,18 +98,22 @@ class PostItemCreator extends HookConsumerWidget { | |||||||
|               context.goNamed('postDetail', pathParameters: {'id': item.id}); |               context.goNamed('postDetail', pathParameters: {'id': item.id}); | ||||||
|             } |             } | ||||||
|           }, |           }, | ||||||
|           child: Padding( |           child: Column( | ||||||
|             padding: renderingPadding, |             crossAxisAlignment: CrossAxisAlignment.start, | ||||||
|             child: Column( |             children: [ | ||||||
|               crossAxisAlignment: CrossAxisAlignment.start, |               Gap(renderingPadding.vertical), | ||||||
|               children: [ |               PostHeader(item: item, renderingPadding: renderingPadding), | ||||||
|                 PostHeader(item: item), |               PostBody(item: item, renderingPadding: renderingPadding), | ||||||
|                 PostBody(item: item), |               ReferencedPostWidget( | ||||||
|                 ReferencedPostWidget(item: item), |                 item: item, | ||||||
|                 const Gap(16), |                 renderingPadding: renderingPadding, | ||||||
|                 _buildAnalyticsSection(context), |               ), | ||||||
|               ], |               const Gap(16), | ||||||
|             ), |               _buildAnalyticsSection( | ||||||
|  |                 context, | ||||||
|  |               ).padding(horizontal: renderingPadding.horizontal), | ||||||
|  |               Gap(renderingPadding.vertical), | ||||||
|  |             ], | ||||||
|           ), |           ), | ||||||
|         ), |         ), | ||||||
|       ), |       ), | ||||||
|   | |||||||
| @@ -21,6 +21,7 @@ class PostListNotifier extends _$PostListNotifier | |||||||
|     int? type, |     int? type, | ||||||
|     List<String>? categories, |     List<String>? categories, | ||||||
|     List<String>? tags, |     List<String>? tags, | ||||||
|  |     bool? pinned, | ||||||
|     bool shuffle = false, |     bool shuffle = false, | ||||||
|   }) { |   }) { | ||||||
|     return fetch(cursor: null); |     return fetch(cursor: null); | ||||||
| @@ -40,6 +41,7 @@ class PostListNotifier extends _$PostListNotifier | |||||||
|       if (tags != null) 'tags': tags, |       if (tags != null) 'tags': tags, | ||||||
|       if (categories != null) 'categories': categories, |       if (categories != null) 'categories': categories, | ||||||
|       if (shuffle) 'shuffle': true, |       if (shuffle) 'shuffle': true, | ||||||
|  |       if (pinned != null) 'pinned': pinned, | ||||||
|     }; |     }; | ||||||
|  |  | ||||||
|     final response = await client.get( |     final response = await client.get( | ||||||
| @@ -77,6 +79,7 @@ class SliverPostList extends HookConsumerWidget { | |||||||
|   final List<String>? categories; |   final List<String>? categories; | ||||||
|   final List<String>? tags; |   final List<String>? tags; | ||||||
|   final bool shuffle; |   final bool shuffle; | ||||||
|  |   final bool? pinned; | ||||||
|   final PostItemType itemType; |   final PostItemType itemType; | ||||||
|   final Color? backgroundColor; |   final Color? backgroundColor; | ||||||
|   final EdgeInsets? padding; |   final EdgeInsets? padding; | ||||||
| @@ -93,6 +96,7 @@ class SliverPostList extends HookConsumerWidget { | |||||||
|     this.categories, |     this.categories, | ||||||
|     this.tags, |     this.tags, | ||||||
|     this.shuffle = false, |     this.shuffle = false, | ||||||
|  |     this.pinned, | ||||||
|     this.itemType = PostItemType.regular, |     this.itemType = PostItemType.regular, | ||||||
|     this.backgroundColor, |     this.backgroundColor, | ||||||
|     this.padding, |     this.padding, | ||||||
| @@ -104,33 +108,19 @@ class SliverPostList extends HookConsumerWidget { | |||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   Widget build(BuildContext context, WidgetRef ref) { |   Widget build(BuildContext context, WidgetRef ref) { | ||||||
|  |     final provider = postListNotifierProvider( | ||||||
|  |       pubName: pubName, | ||||||
|  |       realm: realm, | ||||||
|  |       type: type, | ||||||
|  |       categories: categories, | ||||||
|  |       tags: tags, | ||||||
|  |       shuffle: shuffle, | ||||||
|  |       pinned: pinned, | ||||||
|  |     ); | ||||||
|     return PagingHelperSliverView( |     return PagingHelperSliverView( | ||||||
|       provider: postListNotifierProvider( |       provider: provider, | ||||||
|         pubName: pubName, |       futureRefreshable: provider.future, | ||||||
|         realm: realm, |       notifierRefreshable: provider.notifier, | ||||||
|         type: type, |  | ||||||
|         categories: categories, |  | ||||||
|         tags: tags, |  | ||||||
|         shuffle: shuffle, |  | ||||||
|       ), |  | ||||||
|       futureRefreshable: |  | ||||||
|           postListNotifierProvider( |  | ||||||
|             pubName: pubName, |  | ||||||
|             realm: realm, |  | ||||||
|             type: type, |  | ||||||
|             categories: categories, |  | ||||||
|             tags: tags, |  | ||||||
|             shuffle: shuffle, |  | ||||||
|           ).future, |  | ||||||
|       notifierRefreshable: |  | ||||||
|           postListNotifierProvider( |  | ||||||
|             pubName: pubName, |  | ||||||
|             realm: realm, |  | ||||||
|             type: type, |  | ||||||
|             categories: categories, |  | ||||||
|             tags: tags, |  | ||||||
|             shuffle: shuffle, |  | ||||||
|           ).notifier, |  | ||||||
|       contentBuilder: |       contentBuilder: | ||||||
|           (data, widgetCount, endItemView) => SliverList.builder( |           (data, widgetCount, endItemView) => SliverList.builder( | ||||||
|             itemCount: widgetCount, |             itemCount: widgetCount, | ||||||
|   | |||||||
| @@ -6,7 +6,7 @@ part of 'post_list.dart'; | |||||||
| // RiverpodGenerator | // RiverpodGenerator | ||||||
| // ************************************************************************** | // ************************************************************************** | ||||||
|  |  | ||||||
| String _$postListNotifierHash() => r'faa0b939fae56367ff120ce63d9deb17b1995c9c'; | String _$postListNotifierHash() => r'3c0a8154ded4bcd8f5456f7a4ea2e542f57efa85'; | ||||||
|  |  | ||||||
| /// Copied from Dart SDK | /// Copied from Dart SDK | ||||||
| class _SystemHash { | class _SystemHash { | ||||||
| @@ -36,6 +36,7 @@ abstract class _$PostListNotifier | |||||||
|   late final int? type; |   late final int? type; | ||||||
|   late final List<String>? categories; |   late final List<String>? categories; | ||||||
|   late final List<String>? tags; |   late final List<String>? tags; | ||||||
|  |   late final bool? pinned; | ||||||
|   late final bool shuffle; |   late final bool shuffle; | ||||||
|  |  | ||||||
|   FutureOr<CursorPagingData<SnPost>> build({ |   FutureOr<CursorPagingData<SnPost>> build({ | ||||||
| @@ -44,6 +45,7 @@ abstract class _$PostListNotifier | |||||||
|     int? type, |     int? type, | ||||||
|     List<String>? categories, |     List<String>? categories, | ||||||
|     List<String>? tags, |     List<String>? tags, | ||||||
|  |     bool? pinned, | ||||||
|     bool shuffle = false, |     bool shuffle = false, | ||||||
|   }); |   }); | ||||||
| } | } | ||||||
| @@ -65,6 +67,7 @@ class PostListNotifierFamily | |||||||
|     int? type, |     int? type, | ||||||
|     List<String>? categories, |     List<String>? categories, | ||||||
|     List<String>? tags, |     List<String>? tags, | ||||||
|  |     bool? pinned, | ||||||
|     bool shuffle = false, |     bool shuffle = false, | ||||||
|   }) { |   }) { | ||||||
|     return PostListNotifierProvider( |     return PostListNotifierProvider( | ||||||
| @@ -73,6 +76,7 @@ class PostListNotifierFamily | |||||||
|       type: type, |       type: type, | ||||||
|       categories: categories, |       categories: categories, | ||||||
|       tags: tags, |       tags: tags, | ||||||
|  |       pinned: pinned, | ||||||
|       shuffle: shuffle, |       shuffle: shuffle, | ||||||
|     ); |     ); | ||||||
|   } |   } | ||||||
| @@ -87,6 +91,7 @@ class PostListNotifierFamily | |||||||
|       type: provider.type, |       type: provider.type, | ||||||
|       categories: provider.categories, |       categories: provider.categories, | ||||||
|       tags: provider.tags, |       tags: provider.tags, | ||||||
|  |       pinned: provider.pinned, | ||||||
|       shuffle: provider.shuffle, |       shuffle: provider.shuffle, | ||||||
|     ); |     ); | ||||||
|   } |   } | ||||||
| @@ -120,6 +125,7 @@ class PostListNotifierProvider | |||||||
|     int? type, |     int? type, | ||||||
|     List<String>? categories, |     List<String>? categories, | ||||||
|     List<String>? tags, |     List<String>? tags, | ||||||
|  |     bool? pinned, | ||||||
|     bool shuffle = false, |     bool shuffle = false, | ||||||
|   }) : this._internal( |   }) : this._internal( | ||||||
|          () => |          () => | ||||||
| @@ -129,6 +135,7 @@ class PostListNotifierProvider | |||||||
|                ..type = type |                ..type = type | ||||||
|                ..categories = categories |                ..categories = categories | ||||||
|                ..tags = tags |                ..tags = tags | ||||||
|  |                ..pinned = pinned | ||||||
|                ..shuffle = shuffle, |                ..shuffle = shuffle, | ||||||
|          from: postListNotifierProvider, |          from: postListNotifierProvider, | ||||||
|          name: r'postListNotifierProvider', |          name: r'postListNotifierProvider', | ||||||
| @@ -144,6 +151,7 @@ class PostListNotifierProvider | |||||||
|          type: type, |          type: type, | ||||||
|          categories: categories, |          categories: categories, | ||||||
|          tags: tags, |          tags: tags, | ||||||
|  |          pinned: pinned, | ||||||
|          shuffle: shuffle, |          shuffle: shuffle, | ||||||
|        ); |        ); | ||||||
|  |  | ||||||
| @@ -159,6 +167,7 @@ class PostListNotifierProvider | |||||||
|     required this.type, |     required this.type, | ||||||
|     required this.categories, |     required this.categories, | ||||||
|     required this.tags, |     required this.tags, | ||||||
|  |     required this.pinned, | ||||||
|     required this.shuffle, |     required this.shuffle, | ||||||
|   }) : super.internal(); |   }) : super.internal(); | ||||||
|  |  | ||||||
| @@ -167,6 +176,7 @@ class PostListNotifierProvider | |||||||
|   final int? type; |   final int? type; | ||||||
|   final List<String>? categories; |   final List<String>? categories; | ||||||
|   final List<String>? tags; |   final List<String>? tags; | ||||||
|  |   final bool? pinned; | ||||||
|   final bool shuffle; |   final bool shuffle; | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
| @@ -179,6 +189,7 @@ class PostListNotifierProvider | |||||||
|       type: type, |       type: type, | ||||||
|       categories: categories, |       categories: categories, | ||||||
|       tags: tags, |       tags: tags, | ||||||
|  |       pinned: pinned, | ||||||
|       shuffle: shuffle, |       shuffle: shuffle, | ||||||
|     ); |     ); | ||||||
|   } |   } | ||||||
| @@ -195,6 +206,7 @@ class PostListNotifierProvider | |||||||
|               ..type = type |               ..type = type | ||||||
|               ..categories = categories |               ..categories = categories | ||||||
|               ..tags = tags |               ..tags = tags | ||||||
|  |               ..pinned = pinned | ||||||
|               ..shuffle = shuffle, |               ..shuffle = shuffle, | ||||||
|         from: from, |         from: from, | ||||||
|         name: null, |         name: null, | ||||||
| @@ -206,6 +218,7 @@ class PostListNotifierProvider | |||||||
|         type: type, |         type: type, | ||||||
|         categories: categories, |         categories: categories, | ||||||
|         tags: tags, |         tags: tags, | ||||||
|  |         pinned: pinned, | ||||||
|         shuffle: shuffle, |         shuffle: shuffle, | ||||||
|       ), |       ), | ||||||
|     ); |     ); | ||||||
| @@ -228,6 +241,7 @@ class PostListNotifierProvider | |||||||
|         other.type == type && |         other.type == type && | ||||||
|         other.categories == categories && |         other.categories == categories && | ||||||
|         other.tags == tags && |         other.tags == tags && | ||||||
|  |         other.pinned == pinned && | ||||||
|         other.shuffle == shuffle; |         other.shuffle == shuffle; | ||||||
|   } |   } | ||||||
|  |  | ||||||
| @@ -239,6 +253,7 @@ class PostListNotifierProvider | |||||||
|     hash = _SystemHash.combine(hash, type.hashCode); |     hash = _SystemHash.combine(hash, type.hashCode); | ||||||
|     hash = _SystemHash.combine(hash, categories.hashCode); |     hash = _SystemHash.combine(hash, categories.hashCode); | ||||||
|     hash = _SystemHash.combine(hash, tags.hashCode); |     hash = _SystemHash.combine(hash, tags.hashCode); | ||||||
|  |     hash = _SystemHash.combine(hash, pinned.hashCode); | ||||||
|     hash = _SystemHash.combine(hash, shuffle.hashCode); |     hash = _SystemHash.combine(hash, shuffle.hashCode); | ||||||
|  |  | ||||||
|     return _SystemHash.finish(hash); |     return _SystemHash.finish(hash); | ||||||
| @@ -264,6 +279,9 @@ mixin PostListNotifierRef | |||||||
|   /// The parameter `tags` of this provider. |   /// The parameter `tags` of this provider. | ||||||
|   List<String>? get tags; |   List<String>? get tags; | ||||||
|  |  | ||||||
|  |   /// The parameter `pinned` of this provider. | ||||||
|  |   bool? get pinned; | ||||||
|  |  | ||||||
|   /// The parameter `shuffle` of this provider. |   /// The parameter `shuffle` of this provider. | ||||||
|   bool get shuffle; |   bool get shuffle; | ||||||
| } | } | ||||||
| @@ -289,6 +307,8 @@ class _PostListNotifierProviderElement | |||||||
|   @override |   @override | ||||||
|   List<String>? get tags => (origin as PostListNotifierProvider).tags; |   List<String>? get tags => (origin as PostListNotifierProvider).tags; | ||||||
|   @override |   @override | ||||||
|  |   bool? get pinned => (origin as PostListNotifierProvider).pinned; | ||||||
|  |   @override | ||||||
|   bool get shuffle => (origin as PostListNotifierProvider).shuffle; |   bool get shuffle => (origin as PostListNotifierProvider).shuffle; | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										124
									
								
								lib/widgets/post/post_pin_sheet.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										124
									
								
								lib/widgets/post/post_pin_sheet.dart
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,124 @@ | |||||||
|  | import 'package:easy_localization/easy_localization.dart'; | ||||||
|  | import 'package:flutter/material.dart'; | ||||||
|  | import 'package:flutter_hooks/flutter_hooks.dart'; | ||||||
|  | import 'package:hooks_riverpod/hooks_riverpod.dart'; | ||||||
|  | import 'package:island/models/post.dart'; | ||||||
|  | import 'package:island/pods/network.dart'; | ||||||
|  | import 'package:island/widgets/alert.dart'; | ||||||
|  | import 'package:island/widgets/content/sheet.dart'; | ||||||
|  | import 'package:material_symbols_icons/symbols.dart'; | ||||||
|  | import 'package:styled_widget/styled_widget.dart'; | ||||||
|  |  | ||||||
|  | class PostPinSheet extends HookConsumerWidget { | ||||||
|  |   final SnPost post; | ||||||
|  |   const PostPinSheet({super.key, required this.post}); | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   Widget build(BuildContext context, WidgetRef ref) { | ||||||
|  |     final mode = useState(0); | ||||||
|  |  | ||||||
|  |     Future<void> pinPost() async { | ||||||
|  |       try { | ||||||
|  |         showLoadingModal(context); | ||||||
|  |         final client = ref.watch(apiClientProvider); | ||||||
|  |         await client.post( | ||||||
|  |           '/sphere/posts/${post.id}/pin', | ||||||
|  |           data: {'mode': mode.value}, | ||||||
|  |         ); | ||||||
|  |  | ||||||
|  |         if (context.mounted) Navigator.of(context).pop(mode.value); | ||||||
|  |       } catch (e) { | ||||||
|  |         showErrorAlert(e); | ||||||
|  |       } finally { | ||||||
|  |         if (context.mounted) hideLoadingModal(context); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return SheetScaffold( | ||||||
|  |       titleText: 'pinPost'.tr(), | ||||||
|  |       heightFactor: 0.6, | ||||||
|  |       child: Column( | ||||||
|  |         crossAxisAlignment: CrossAxisAlignment.stretch, | ||||||
|  |         children: [ | ||||||
|  |           // Publisher page pin option (always available) | ||||||
|  |           ListTile( | ||||||
|  |             leading: Radio<int>( | ||||||
|  |               value: 0, | ||||||
|  |               groupValue: mode.value, | ||||||
|  |               onChanged: (value) { | ||||||
|  |                 mode.value = value!; | ||||||
|  |               }, | ||||||
|  |             ), | ||||||
|  |             title: Text('publisherPage'.tr()), | ||||||
|  |             subtitle: Text('pinPostPublisherHint'.tr()), | ||||||
|  |             onTap: () { | ||||||
|  |               mode.value = 0; | ||||||
|  |             }, | ||||||
|  |           ), | ||||||
|  |  | ||||||
|  |           // Realm page pin option (show always, but disabled when not available) | ||||||
|  |           ListTile( | ||||||
|  |             leading: Radio<int>( | ||||||
|  |               value: 1, | ||||||
|  |               groupValue: mode.value, | ||||||
|  |               onChanged: | ||||||
|  |                   post.realmId != null && post.realmId!.isNotEmpty | ||||||
|  |                       ? (value) { | ||||||
|  |                         mode.value = value!; | ||||||
|  |                       } | ||||||
|  |                       : null, | ||||||
|  |             ), | ||||||
|  |             title: Text('realmPage'.tr()), | ||||||
|  |             subtitle: | ||||||
|  |                 post.realmId != null && post.realmId!.isNotEmpty | ||||||
|  |                     ? Text('pinPostRealmHint'.tr()) | ||||||
|  |                     : Text('pinPostRealmDisabledHint'.tr()), | ||||||
|  |             onTap: | ||||||
|  |                 post.realmId != null && post.realmId!.isNotEmpty | ||||||
|  |                     ? () { | ||||||
|  |                       mode.value = 1; | ||||||
|  |                     } | ||||||
|  |                     : null, | ||||||
|  |             enabled: post.realmId != null && post.realmId!.isNotEmpty, | ||||||
|  |           ), | ||||||
|  |  | ||||||
|  |           // Reply page pin option (show always, but disabled when not available) | ||||||
|  |           // Disabled for now because im being lazy | ||||||
|  |           // ListTile( | ||||||
|  |           //   leading: Radio<int>( | ||||||
|  |           //     value: 2, | ||||||
|  |           //     groupValue: mode.value, | ||||||
|  |           //     onChanged: | ||||||
|  |           //         post.repliedPostId != null && post.repliedPostId!.isNotEmpty | ||||||
|  |           //             ? (value) { | ||||||
|  |           //               mode.value = value!; | ||||||
|  |           //             } | ||||||
|  |           //             : null, | ||||||
|  |           //   ), | ||||||
|  |           //   title: Text('replyPage'.tr()), | ||||||
|  |           //   subtitle: | ||||||
|  |           //       post.repliedPostId != null && post.repliedPostId!.isNotEmpty | ||||||
|  |           //           ? Text('pinPostReplyHint'.tr()) | ||||||
|  |           //           : Text('pinPostReplyDisabledHint'.tr()), | ||||||
|  |           //   onTap: | ||||||
|  |           //       post.repliedPostId != null && post.repliedPostId!.isNotEmpty | ||||||
|  |           //           ? () { | ||||||
|  |           //             mode.value = 2; | ||||||
|  |           //           } | ||||||
|  |           //           : null, | ||||||
|  |           //   enabled: | ||||||
|  |           //       post.repliedPostId != null && post.repliedPostId!.isNotEmpty, | ||||||
|  |           // ), | ||||||
|  |           const SizedBox(height: 16), | ||||||
|  |  | ||||||
|  |           // Pin button | ||||||
|  |           FilledButton.icon( | ||||||
|  |             onPressed: pinPost, | ||||||
|  |             icon: const Icon(Symbols.keep), | ||||||
|  |             label: Text('pin'.tr()), | ||||||
|  |           ).padding(horizontal: 24), | ||||||
|  |         ], | ||||||
|  |       ), | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  | } | ||||||
| @@ -532,6 +532,7 @@ class PostHeader extends StatelessWidget { | |||||||
|   final bool isInteractive; |   final bool isInteractive; | ||||||
|   final EdgeInsets renderingPadding; |   final EdgeInsets renderingPadding; | ||||||
|   final bool isRelativeTime; |   final bool isRelativeTime; | ||||||
|  |   final bool isCompact; | ||||||
|  |  | ||||||
|   const PostHeader({ |   const PostHeader({ | ||||||
|     super.key, |     super.key, | ||||||
| @@ -541,111 +542,140 @@ class PostHeader extends StatelessWidget { | |||||||
|     this.isInteractive = true, |     this.isInteractive = true, | ||||||
|     this.renderingPadding = EdgeInsets.zero, |     this.renderingPadding = EdgeInsets.zero, | ||||||
|     this.isRelativeTime = true, |     this.isRelativeTime = true, | ||||||
|  |     this.isCompact = false, | ||||||
|   }); |   }); | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   Widget build(BuildContext context) { |   Widget build(BuildContext context) { | ||||||
|     return Row( |     return Column( | ||||||
|       crossAxisAlignment: CrossAxisAlignment.center, |  | ||||||
|       spacing: 12, |  | ||||||
|       children: [ |       children: [ | ||||||
|         GestureDetector( |         if (item.pinMode != null) | ||||||
|           onTap: |           Row( | ||||||
|               isInteractive |             spacing: 4, | ||||||
|                   ? () { |  | ||||||
|                     context.pushNamed( |  | ||||||
|                       'publisherProfile', |  | ||||||
|                       pathParameters: {'name': item.publisher.name}, |  | ||||||
|                     ); |  | ||||||
|                   } |  | ||||||
|                   : null, |  | ||||||
|           child: ProfilePictureWidget( |  | ||||||
|             file: item.publisher.picture, |  | ||||||
|             radius: 16, |  | ||||||
|             borderRadius: item.publisher.type == 0 ? null : 6, |  | ||||||
|           ), |  | ||||||
|         ), |  | ||||||
|         Expanded( |  | ||||||
|           child: Column( |  | ||||||
|             mainAxisSize: MainAxisSize.min, |  | ||||||
|             crossAxisAlignment: CrossAxisAlignment.start, |  | ||||||
|             children: [ |             children: [ | ||||||
|               Row( |               const Icon(Symbols.keep, size: 15, fill: 1), | ||||||
|                 crossAxisAlignment: CrossAxisAlignment.center, |               Text('pinnedPost').tr().fontSize(13), | ||||||
|                 spacing: 4, |             ], | ||||||
|  |           ).opacity(0.8).padding(horizontal: 8, bottom: 4), | ||||||
|  |         Row( | ||||||
|  |           crossAxisAlignment: CrossAxisAlignment.center, | ||||||
|  |           spacing: 12, | ||||||
|  |           children: [ | ||||||
|  |             GestureDetector( | ||||||
|  |               onTap: | ||||||
|  |                   isInteractive | ||||||
|  |                       ? () { | ||||||
|  |                         context.pushNamed( | ||||||
|  |                           'publisherProfile', | ||||||
|  |                           pathParameters: {'name': item.publisher.name}, | ||||||
|  |                         ); | ||||||
|  |                       } | ||||||
|  |                       : null, | ||||||
|  |               child: ProfilePictureWidget( | ||||||
|  |                 file: item.publisher.picture, | ||||||
|  |                 radius: 16, | ||||||
|  |                 borderRadius: item.publisher.type == 0 ? null : 6, | ||||||
|  |               ), | ||||||
|  |             ), | ||||||
|  |             Expanded( | ||||||
|  |               child: Column( | ||||||
|  |                 mainAxisSize: MainAxisSize.min, | ||||||
|  |                 crossAxisAlignment: CrossAxisAlignment.start, | ||||||
|                 children: [ |                 children: [ | ||||||
|                   Text(item.publisher.nick).bold(), |                   Row( | ||||||
|                   if (item.publisher.verification != null) |                     crossAxisAlignment: CrossAxisAlignment.center, | ||||||
|                     VerificationMark(mark: item.publisher.verification!), |                     spacing: 4, | ||||||
|                   if (item.realm == null) |                     children: [ | ||||||
|                     Text('@${item.publisher.name}').fontSize(11) |  | ||||||
|                   else |  | ||||||
|                     ...([ |  | ||||||
|                       const Icon(Symbols.arrow_right, size: 14), |  | ||||||
|                       Flexible( |                       Flexible( | ||||||
|                         child: InkWell( |                         child: | ||||||
|                           child: Row( |                             Text( | ||||||
|                             mainAxisSize: MainAxisSize.min, |                               item.publisher.nick, | ||||||
|                             spacing: 5, |                               maxLines: 1, | ||||||
|                             children: [ |                               overflow: TextOverflow.ellipsis, | ||||||
|                               Flexible( |                             ).bold(), | ||||||
|                                 child: Text( |                       ), | ||||||
|                                   item.realm!.name, |                       if (item.publisher.verification != null) | ||||||
|                                   maxLines: 1, |                         VerificationMark(mark: item.publisher.verification!), | ||||||
|                                   overflow: TextOverflow.ellipsis, |                       if (item.realm == null) | ||||||
|                                 ), |                         Flexible( | ||||||
|                               ), |                           child: | ||||||
|                               ProfilePictureWidget( |                               isCompact | ||||||
|                                 file: item.realm!.picture, |                                   ? const SizedBox.shrink() | ||||||
|                                 fallbackIcon: Symbols.group, |                                   : Text( | ||||||
|                                 radius: 9, |                                     '@${item.publisher.name}', | ||||||
|  |                                     maxLines: 1, | ||||||
|  |                                     overflow: TextOverflow.ellipsis, | ||||||
|  |                                   ).fontSize(11), | ||||||
|  |                         ) | ||||||
|  |                       else | ||||||
|  |                         ...([ | ||||||
|  |                           const Icon(Symbols.arrow_right, size: 14), | ||||||
|  |                           Flexible( | ||||||
|  |                             child: InkWell( | ||||||
|  |                               child: Row( | ||||||
|  |                                 mainAxisSize: MainAxisSize.min, | ||||||
|  |                                 spacing: 5, | ||||||
|  |                                 children: [ | ||||||
|  |                                   Flexible( | ||||||
|  |                                     child: Text( | ||||||
|  |                                       item.realm!.name, | ||||||
|  |                                       maxLines: 1, | ||||||
|  |                                       overflow: TextOverflow.ellipsis, | ||||||
|  |                                     ), | ||||||
|  |                                   ), | ||||||
|  |                                   ProfilePictureWidget( | ||||||
|  |                                     file: item.realm!.picture, | ||||||
|  |                                     fallbackIcon: Symbols.group, | ||||||
|  |                                     radius: 9, | ||||||
|  |                                   ), | ||||||
|  |                                 ], | ||||||
|                               ), |                               ), | ||||||
|  |                               onTap: () { | ||||||
|  |                                 GoRouter.of(context).pushNamed( | ||||||
|  |                                   'realmDetail', | ||||||
|  |                                   pathParameters: {'slug': item.realm!.slug}, | ||||||
|  |                                 ); | ||||||
|  |                               }, | ||||||
|  |                             ), | ||||||
|  |                           ), | ||||||
|  |                         ]), | ||||||
|  |                     ], | ||||||
|  |                   ), | ||||||
|  |                   Row( | ||||||
|  |                     spacing: 6, | ||||||
|  |                     crossAxisAlignment: CrossAxisAlignment.end, | ||||||
|  |                     children: [ | ||||||
|  |                       Text( | ||||||
|  |                         !isFullPost && isRelativeTime | ||||||
|  |                             ? (item.publishedAt ?? item.createdAt)! | ||||||
|  |                                 .formatRelative(context) | ||||||
|  |                             : (item.publishedAt ?? item.createdAt)! | ||||||
|  |                                 .formatSystem(), | ||||||
|  |                       ).fontSize(10), | ||||||
|  |                       if (item.editedAt != null) | ||||||
|  |                         Text( | ||||||
|  |                           'editedAt'.tr( | ||||||
|  |                             args: [ | ||||||
|  |                               !isFullPost && isRelativeTime | ||||||
|  |                                   ? item.editedAt!.formatRelative(context) | ||||||
|  |                                   : item.editedAt!.formatSystem(), | ||||||
|                             ], |                             ], | ||||||
|                           ), |                           ), | ||||||
|                           onTap: () { |                         ).fontSize(10), | ||||||
|                             GoRouter.of(context).pushNamed( |                       if (item.visibility != 0) | ||||||
|                               'realmDetail', |                         Text( | ||||||
|                               pathParameters: {'slug': item.realm!.slug}, |                           PostVisibilityHelpers.getVisibilityText( | ||||||
|                             ); |                             item.visibility, | ||||||
|                           }, |                           ).tr(), | ||||||
|                         ), |                         ).fontSize(10), | ||||||
|                       ), |                     ], | ||||||
|                     ]), |                   ), | ||||||
|                 ], |                 ], | ||||||
|               ), |               ), | ||||||
|               Row( |             ), | ||||||
|                 spacing: 6, |             if (trailing != null) trailing!, | ||||||
|                 crossAxisAlignment: CrossAxisAlignment.end, |           ], | ||||||
|                 children: [ |  | ||||||
|                   Text( |  | ||||||
|                     !isFullPost && isRelativeTime |  | ||||||
|                         ? (item.publishedAt ?? item.createdAt)!.formatRelative( |  | ||||||
|                           context, |  | ||||||
|                         ) |  | ||||||
|                         : (item.publishedAt ?? item.createdAt)!.formatSystem(), |  | ||||||
|                   ).fontSize(10), |  | ||||||
|                   if (item.editedAt != null) |  | ||||||
|                     Text( |  | ||||||
|                       'editedAt'.tr( |  | ||||||
|                         args: [ |  | ||||||
|                           !isFullPost && isRelativeTime |  | ||||||
|                               ? item.editedAt!.formatRelative(context) |  | ||||||
|                               : item.editedAt!.formatSystem(), |  | ||||||
|                         ], |  | ||||||
|                       ), |  | ||||||
|                     ).fontSize(10), |  | ||||||
|                   if (item.visibility != 0) |  | ||||||
|                     Text( |  | ||||||
|                       PostVisibilityHelpers.getVisibilityText( |  | ||||||
|                         item.visibility, |  | ||||||
|                       ).tr(), |  | ||||||
|                     ).fontSize(10), |  | ||||||
|                 ], |  | ||||||
|               ), |  | ||||||
|             ], |  | ||||||
|           ), |  | ||||||
|         ), |         ), | ||||||
|         if (trailing != null) trailing!, |  | ||||||
|       ], |       ], | ||||||
|     ).padding(horizontal: renderingPadding.horizontal, bottom: 4); |     ).padding(horizontal: renderingPadding.horizontal, bottom: 4); | ||||||
|   } |   } | ||||||
| @@ -750,7 +780,9 @@ class PostBody extends ConsumerWidget { | |||||||
|                   ).padding(bottom: 4), |                   ).padding(bottom: 4), | ||||||
|                 MarkdownTextContent( |                 MarkdownTextContent( | ||||||
|                   content: |                   content: | ||||||
|                       item.isTruncated ? '${item.content!}...' : item.content!, |                       item.isTruncated | ||||||
|  |                           ? '${item.content!}...' | ||||||
|  |                           : item.content ?? '', | ||||||
|                   isSelectable: isTextSelectable, |                   isSelectable: isTextSelectable, | ||||||
|                 ), |                 ), | ||||||
|                 if (translationSection != null) translationSection!, |                 if (translationSection != null) translationSection!, | ||||||
|   | |||||||
| @@ -25,7 +25,7 @@ class PostShuffleScreen extends HookConsumerWidget { | |||||||
|       return cardSwiperController.dispose; |       return cardSwiperController.dispose; | ||||||
|     }, []); |     }, []); | ||||||
|  |  | ||||||
|     const kBottomControlHeight = 64.0; |     const kBottomControlHeight = 80.0; | ||||||
|  |  | ||||||
|     return AppScaffold( |     return AppScaffold( | ||||||
|       appBar: AppBar(title: const Text('postShuffle').tr()), |       appBar: AppBar(title: const Text('postShuffle').tr()), | ||||||
| @@ -36,45 +36,50 @@ class PostShuffleScreen extends HookConsumerWidget { | |||||||
|               bottom: |               bottom: | ||||||
|                   kBottomControlHeight + MediaQuery.of(context).padding.bottom, |                   kBottomControlHeight + MediaQuery.of(context).padding.bottom, | ||||||
|             ), |             ), | ||||||
|             child: |             child: Builder( | ||||||
|                 (postListState.value?.items.length ?? 0) > 0 |               key: ValueKey(postListState.value?.items.length ?? 0), | ||||||
|                     ? CardSwiper( |               builder: (context) { | ||||||
|                       controller: cardSwiperController, |                 if ((postListState.value?.items.length ?? 0) > 0) { | ||||||
|                       cardsCount: postListState.value!.items.length, |                   return CardSwiper( | ||||||
|                       cardBuilder: ( |                     controller: cardSwiperController, | ||||||
|                         context, |                     cardsCount: postListState.value!.items.length, | ||||||
|                         index, |                     isLoop: false, | ||||||
|                         horizontalOffsetPercentage, |                     cardBuilder: ( | ||||||
|                         verticalOffsetPercentage, |                       context, | ||||||
|                       ) { |                       index, | ||||||
|                         return Center( |                       horizontalOffsetPercentage, | ||||||
|                           child: ConstrainedBox( |                       verticalOffsetPercentage, | ||||||
|                             constraints: BoxConstraints(maxWidth: 540), |                     ) { | ||||||
|                             child: SingleChildScrollView( |                       return Center( | ||||||
|                               child: Card( |                         child: ConstrainedBox( | ||||||
|                                 margin: EdgeInsets.zero, |                           constraints: BoxConstraints(maxWidth: 540), | ||||||
|                                 child: ClipRRect( |                           child: SingleChildScrollView( | ||||||
|                                   borderRadius: const BorderRadius.all( |                             child: Card( | ||||||
|                                     Radius.circular(8), |                               margin: EdgeInsets.zero, | ||||||
|                                   ), |                               child: ClipRRect( | ||||||
|                                   child: PostActionableItem( |                                 borderRadius: const BorderRadius.all( | ||||||
|                                     item: postListState.value!.items[index], |                                   Radius.circular(8), | ||||||
|                                   ), |                                 ), | ||||||
|  |                                 child: PostActionableItem( | ||||||
|  |                                   item: postListState.value!.items[index], | ||||||
|                                 ), |                                 ), | ||||||
|                               ), |                               ), | ||||||
|                             ), |                             ), | ||||||
|                           ), |                           ), | ||||||
|                         ); |                         ), | ||||||
|                       }, |                       ); | ||||||
|                       onEnd: () { |                     }, | ||||||
|                         if (postListState.value?.hasMore ?? true) { |                     onEnd: () async { | ||||||
|                           postListNotifier.fetch( |                       if (postListState.value?.hasMore ?? true) { | ||||||
|                             cursor: postListState.value?.nextCursor, |                         postListNotifier.forceRefresh(); | ||||||
|                           ); |                       } | ||||||
|                         } |                     }, | ||||||
|                       }, |                   ); | ||||||
|                     ) |                 } else { | ||||||
|                     : Center(child: CircularProgressIndicator()), |                   return Center(child: CircularProgressIndicator()); | ||||||
|  |                 } | ||||||
|  |               }, | ||||||
|  |             ), | ||||||
|           ), |           ), | ||||||
|           Positioned( |           Positioned( | ||||||
|             left: 0, |             left: 0, | ||||||
|   | |||||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user