✨ Sticker page & add sticker
This commit is contained in:
parent
c8c455bb57
commit
899d5f3e5e
@ -689,5 +689,13 @@
|
|||||||
"databaseDeleteDescription": "Remove the database on your local disk, the content will be fetched from server again.",
|
"databaseDeleteDescription": "Remove the database on your local disk, the content will be fetched from server again.",
|
||||||
"databaseDeleted": "The local database has been deleted.",
|
"databaseDeleted": "The local database has been deleted.",
|
||||||
"settingsEnablePushNotifications": "Enable Push Notifications",
|
"settingsEnablePushNotifications": "Enable Push Notifications",
|
||||||
"settingsEnablePushNotificationsDescription": "Re-enable and request permission to receive push notifications. Just in case it didn't run automatically."
|
"settingsEnablePushNotificationsDescription": "Re-enable and request permission to receive push notifications. Just in case it didn't run automatically.",
|
||||||
|
"screenStickers": "Stickers",
|
||||||
|
"stickersDiscovery": "Discovery",
|
||||||
|
"stickersOwned": "Owned",
|
||||||
|
"stickersCreated": "Created",
|
||||||
|
"stickersAdd": "Add Sticker Pack",
|
||||||
|
"stickersAdded": "Sticker pack has been added.",
|
||||||
|
"add": "Add",
|
||||||
|
"stickersRemoved": "Sticker pack has been removed, you can add it again anytime."
|
||||||
}
|
}
|
||||||
|
@ -687,5 +687,13 @@
|
|||||||
"databaseDeleteDescription": "删除本地数据库,内容将从服务器重新获取。",
|
"databaseDeleteDescription": "删除本地数据库,内容将从服务器重新获取。",
|
||||||
"databaseDeleted": "本地数据库已被删除。",
|
"databaseDeleted": "本地数据库已被删除。",
|
||||||
"settingsEnablePushNotifications": "启用推送数据",
|
"settingsEnablePushNotifications": "启用推送数据",
|
||||||
"settingsEnablePushNotificationsDescription": "重新启用并请求推送权限,以防自动激活失败。"
|
"settingsEnablePushNotificationsDescription": "重新启用并请求推送权限,以防自动激活失败。",
|
||||||
|
"screenStickers": "贴图",
|
||||||
|
"stickersDiscovery": "发现",
|
||||||
|
"stickersOwned": "由我拥有",
|
||||||
|
"stickersCreated": "由我发布",
|
||||||
|
"stickersAdd": "添加贴图包",
|
||||||
|
"stickersAdded": "贴图包已添加。",
|
||||||
|
"add": "添加",
|
||||||
|
"stickersRemoved": "贴图包已被移除,你可以随时再次添加回来。"
|
||||||
}
|
}
|
||||||
|
@ -685,5 +685,15 @@
|
|||||||
"databaseSize": "數據庫大小",
|
"databaseSize": "數據庫大小",
|
||||||
"databaseDelete": "刪除數據庫",
|
"databaseDelete": "刪除數據庫",
|
||||||
"databaseDeleteDescription": "刪除本地數據庫,內容將從服務器重新獲取。",
|
"databaseDeleteDescription": "刪除本地數據庫,內容將從服務器重新獲取。",
|
||||||
"databaseDeleted": "本地數據庫已被刪除。"
|
"databaseDeleted": "本地數據庫已被刪除。",
|
||||||
|
"settingsEnablePushNotifications": "啓用推送數據",
|
||||||
|
"settingsEnablePushNotificationsDescription": "重新啓用並請求推送權限,以防自動激活失敗。",
|
||||||
|
"screenStickers": "貼圖",
|
||||||
|
"stickersDiscovery": "發現",
|
||||||
|
"stickersOwned": "由我擁有",
|
||||||
|
"stickersCreated": "由我發佈",
|
||||||
|
"stickersAdd": "添加貼圖包",
|
||||||
|
"stickersAdded": "貼圖包已添加。",
|
||||||
|
"add": "添加",
|
||||||
|
"stickersRemoved": "貼圖包已被移除,你可以隨時再次添加回來。"
|
||||||
}
|
}
|
||||||
|
@ -685,5 +685,15 @@
|
|||||||
"databaseSize": "數據庫大小",
|
"databaseSize": "數據庫大小",
|
||||||
"databaseDelete": "刪除數據庫",
|
"databaseDelete": "刪除數據庫",
|
||||||
"databaseDeleteDescription": "刪除本地數據庫,內容將從服務器重新獲取。",
|
"databaseDeleteDescription": "刪除本地數據庫,內容將從服務器重新獲取。",
|
||||||
"databaseDeleted": "本地數據庫已被刪除。"
|
"databaseDeleted": "本地數據庫已被刪除。",
|
||||||
|
"settingsEnablePushNotifications": "啟用推送數據",
|
||||||
|
"settingsEnablePushNotificationsDescription": "重新啟用並請求推送權限,以防自動激活失敗。",
|
||||||
|
"screenStickers": "貼圖",
|
||||||
|
"stickersDiscovery": "發現",
|
||||||
|
"stickersOwned": "由我擁有",
|
||||||
|
"stickersCreated": "由我發佈",
|
||||||
|
"stickersAdd": "添加貼圖包",
|
||||||
|
"stickersAdded": "貼圖包已添加。",
|
||||||
|
"add": "添加",
|
||||||
|
"stickersRemoved": "貼圖包已被移除,你可以隨時再次添加回來。"
|
||||||
}
|
}
|
||||||
|
@ -63,6 +63,11 @@ class NavigationProvider extends ChangeNotifier {
|
|||||||
screen: 'news',
|
screen: 'news',
|
||||||
label: 'screenNews',
|
label: 'screenNews',
|
||||||
),
|
),
|
||||||
|
AppNavDestination(
|
||||||
|
icon: Icon(Symbols.emoji_emotions, weight: 400, opticalSize: 20),
|
||||||
|
screen: 'stickers',
|
||||||
|
label: 'screenStickers',
|
||||||
|
),
|
||||||
AppNavDestination(
|
AppNavDestination(
|
||||||
icon: Icon(Symbols.photo_library, weight: 400, opticalSize: 20),
|
icon: Icon(Symbols.photo_library, weight: 400, opticalSize: 20),
|
||||||
screen: 'album',
|
screen: 'album',
|
||||||
@ -88,7 +93,8 @@ class NavigationProvider extends ChangeNotifier {
|
|||||||
|
|
||||||
List<AppNavDestination> destinations = [];
|
List<AppNavDestination> destinations = [];
|
||||||
|
|
||||||
int get pinnedDestinationCount => destinations.where((ele) => ele.isPinned).length;
|
int get pinnedDestinationCount =>
|
||||||
|
destinations.where((ele) => ele.isPinned).length;
|
||||||
|
|
||||||
NavigationProvider() {
|
NavigationProvider() {
|
||||||
buildDestinations(kDefaultPinnedDestination);
|
buildDestinations(kDefaultPinnedDestination);
|
||||||
@ -117,13 +123,17 @@ class NavigationProvider extends ChangeNotifier {
|
|||||||
}
|
}
|
||||||
|
|
||||||
bool isIndexInRange(int min, int max) {
|
bool isIndexInRange(int min, int max) {
|
||||||
return _currentIndex != null && _currentIndex! >= min && _currentIndex! < max;
|
return _currentIndex != null &&
|
||||||
|
_currentIndex! >= min &&
|
||||||
|
_currentIndex! < max;
|
||||||
}
|
}
|
||||||
|
|
||||||
void autoDetectIndex(GoRouter? state) {
|
void autoDetectIndex(GoRouter? state) {
|
||||||
if (state == null) return;
|
if (state == null) return;
|
||||||
final idx = destinations.indexWhere(
|
final idx = destinations.indexWhere(
|
||||||
(ele) => ele.screen == state.routerDelegate.currentConfiguration.last.route.name,
|
(ele) =>
|
||||||
|
ele.screen ==
|
||||||
|
state.routerDelegate.currentConfiguration.last.route.name,
|
||||||
);
|
);
|
||||||
_currentIndex = idx == -1 ? null : idx;
|
_currentIndex = idx == -1 ? null : idx;
|
||||||
notifyListeners();
|
notifyListeners();
|
||||||
|
@ -11,7 +11,8 @@ class SnStickerProvider {
|
|||||||
|
|
||||||
final Map<int, List<SnSticker>> stickersByPack = {};
|
final Map<int, List<SnSticker>> stickersByPack = {};
|
||||||
|
|
||||||
List<SnSticker> get stickers => _cache.values.where((ele) => ele != null).cast<SnSticker>().toList();
|
List<SnSticker> get stickers =>
|
||||||
|
_cache.values.where((ele) => ele != null).cast<SnSticker>().toList();
|
||||||
|
|
||||||
SnStickerProvider(BuildContext context) {
|
SnStickerProvider(BuildContext context) {
|
||||||
_sn = context.read<SnNetworkProvider>();
|
_sn = context.read<SnNetworkProvider>();
|
||||||
@ -23,8 +24,18 @@ class SnStickerProvider {
|
|||||||
|
|
||||||
void _cacheSticker(SnSticker sticker) {
|
void _cacheSticker(SnSticker sticker) {
|
||||||
_cache['${sticker.pack.prefix}:${sticker.alias}'] = sticker;
|
_cache['${sticker.pack.prefix}:${sticker.alias}'] = sticker;
|
||||||
if (stickersByPack[sticker.pack.id] == null) stickersByPack[sticker.pack.id] = List.empty(growable: true);
|
if (stickersByPack[sticker.pack.id] == null) {
|
||||||
if (!stickersByPack[sticker.pack.id]!.contains(sticker)) stickersByPack[sticker.pack.id]!.add(sticker);
|
stickersByPack[sticker.pack.id] = List.empty(growable: true);
|
||||||
|
}
|
||||||
|
if (!stickersByPack[sticker.pack.id]!.contains(sticker)) {
|
||||||
|
stickersByPack[sticker.pack.id]!.add(sticker);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void putSticker(Iterable<SnSticker> sticker) {
|
||||||
|
for (final ele in sticker) {
|
||||||
|
_cacheSticker(ele);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<SnSticker?> lookupSticker(String alias) async {
|
Future<SnSticker?> lookupSticker(String alias) async {
|
||||||
@ -61,7 +72,8 @@ class SnStickerProvider {
|
|||||||
'offset': page * 10,
|
'offset': page * 10,
|
||||||
});
|
});
|
||||||
final data = resp.data;
|
final data = resp.data;
|
||||||
final stickers = List.from(data['data']).map((ele) => SnSticker.fromJson(ele));
|
final stickers =
|
||||||
|
List.from(data['data']).map((ele) => SnSticker.fromJson(ele));
|
||||||
for (final sticker in stickers) {
|
for (final sticker in stickers) {
|
||||||
_cacheSticker(sticker);
|
_cacheSticker(sticker);
|
||||||
}
|
}
|
||||||
|
136
lib/router.dart
136
lib/router.dart
@ -34,13 +34,14 @@ import 'package:surface/screens/realm/realm_detail.dart';
|
|||||||
import 'package:surface/screens/realm/realm_discovery.dart';
|
import 'package:surface/screens/realm/realm_discovery.dart';
|
||||||
import 'package:surface/screens/settings.dart';
|
import 'package:surface/screens/settings.dart';
|
||||||
import 'package:surface/screens/sharing.dart';
|
import 'package:surface/screens/sharing.dart';
|
||||||
|
import 'package:surface/screens/stickers.dart';
|
||||||
import 'package:surface/screens/wallet.dart';
|
import 'package:surface/screens/wallet.dart';
|
||||||
import 'package:surface/types/post.dart';
|
import 'package:surface/types/post.dart';
|
||||||
import 'package:surface/widgets/about.dart';
|
import 'package:surface/widgets/about.dart';
|
||||||
import 'package:surface/widgets/navigation/app_scaffold.dart';
|
import 'package:surface/widgets/navigation/app_scaffold.dart';
|
||||||
|
|
||||||
Widget _fadeThroughTransition(
|
Widget _fadeThroughTransition(BuildContext context, Animation<double> animation,
|
||||||
BuildContext context, Animation<double> animation, Animation<double> secondaryAnimation, Widget child) {
|
Animation<double> secondaryAnimation, Widget child) {
|
||||||
return FadeThroughTransition(
|
return FadeThroughTransition(
|
||||||
animation: animation,
|
animation: animation,
|
||||||
secondaryAnimation: secondaryAnimation,
|
secondaryAnimation: secondaryAnimation,
|
||||||
@ -82,13 +83,15 @@ final _appRoutes = [
|
|||||||
name: 'postSearch',
|
name: 'postSearch',
|
||||||
builder: (context, state) => PostSearchScreen(
|
builder: (context, state) => PostSearchScreen(
|
||||||
initialTags: state.uri.queryParameters['tags']?.split(','),
|
initialTags: state.uri.queryParameters['tags']?.split(','),
|
||||||
initialCategories: state.uri.queryParameters['categories']?.split(','),
|
initialCategories:
|
||||||
|
state.uri.queryParameters['categories']?.split(','),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
GoRoute(
|
GoRoute(
|
||||||
path: '/publishers/:name',
|
path: '/publishers/:name',
|
||||||
name: 'postPublisher',
|
name: 'postPublisher',
|
||||||
builder: (context, state) => PostPublisherScreen(name: state.pathParameters['name']!),
|
builder: (context, state) =>
|
||||||
|
PostPublisherScreen(name: state.pathParameters['name']!),
|
||||||
),
|
),
|
||||||
GoRoute(
|
GoRoute(
|
||||||
path: '/:slug',
|
path: '/:slug',
|
||||||
@ -100,52 +103,56 @@ final _appRoutes = [
|
|||||||
),
|
),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
GoRoute(path: '/account', name: 'account', builder: (context, state) => const AccountScreen(), routes: [
|
GoRoute(
|
||||||
GoRoute(
|
path: '/account',
|
||||||
path: '/wallet',
|
name: 'account',
|
||||||
name: 'accountWallet',
|
builder: (context, state) => const AccountScreen(),
|
||||||
builder: (context, state) => const WalletScreen(),
|
routes: [
|
||||||
),
|
GoRoute(
|
||||||
GoRoute(
|
path: '/wallet',
|
||||||
path: '/settings',
|
name: 'accountWallet',
|
||||||
name: 'accountSettings',
|
builder: (context, state) => const WalletScreen(),
|
||||||
builder: (context, state) => AccountSettingsScreen(),
|
),
|
||||||
),
|
GoRoute(
|
||||||
GoRoute(
|
path: '/settings',
|
||||||
path: '/settings/factors',
|
name: 'accountSettings',
|
||||||
name: 'factorSettings',
|
builder: (context, state) => AccountSettingsScreen(),
|
||||||
builder: (context, state) => FactorSettingsScreen(),
|
),
|
||||||
),
|
GoRoute(
|
||||||
GoRoute(
|
path: '/settings/factors',
|
||||||
path: '/profile/edit',
|
name: 'factorSettings',
|
||||||
name: 'accountProfileEdit',
|
builder: (context, state) => FactorSettingsScreen(),
|
||||||
builder: (context, state) => ProfileEditScreen(),
|
),
|
||||||
),
|
GoRoute(
|
||||||
GoRoute(
|
path: '/profile/edit',
|
||||||
path: '/publishers',
|
name: 'accountProfileEdit',
|
||||||
name: 'accountPublishers',
|
builder: (context, state) => ProfileEditScreen(),
|
||||||
builder: (context, state) => PublisherScreen(),
|
),
|
||||||
),
|
GoRoute(
|
||||||
GoRoute(
|
path: '/publishers',
|
||||||
path: '/publishers/new',
|
name: 'accountPublishers',
|
||||||
name: 'accountPublisherNew',
|
builder: (context, state) => PublisherScreen(),
|
||||||
builder: (context, state) => AccountPublisherNewScreen(),
|
),
|
||||||
),
|
GoRoute(
|
||||||
GoRoute(
|
path: '/publishers/new',
|
||||||
path: '/publishers/edit/:name',
|
name: 'accountPublisherNew',
|
||||||
name: 'accountPublisherEdit',
|
builder: (context, state) => AccountPublisherNewScreen(),
|
||||||
builder: (context, state) => AccountPublisherEditScreen(
|
),
|
||||||
name: state.pathParameters['name']!,
|
GoRoute(
|
||||||
),
|
path: '/publishers/edit/:name',
|
||||||
),
|
name: 'accountPublisherEdit',
|
||||||
GoRoute(
|
builder: (context, state) => AccountPublisherEditScreen(
|
||||||
path: '/:name',
|
name: state.pathParameters['name']!,
|
||||||
name: 'accountProfilePage',
|
),
|
||||||
pageBuilder: (context, state) => NoTransitionPage(
|
),
|
||||||
child: UserScreen(name: state.pathParameters['name']!),
|
GoRoute(
|
||||||
),
|
path: '/:name',
|
||||||
),
|
name: 'accountProfilePage',
|
||||||
]),
|
pageBuilder: (context, state) => NoTransitionPage(
|
||||||
|
child: UserScreen(name: state.pathParameters['name']!),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
]),
|
||||||
GoRoute(
|
GoRoute(
|
||||||
path: '/chat',
|
path: '/chat',
|
||||||
name: 'chat',
|
name: 'chat',
|
||||||
@ -208,19 +215,30 @@ final _appRoutes = [
|
|||||||
GoRoute(
|
GoRoute(
|
||||||
path: '/:alias',
|
path: '/:alias',
|
||||||
name: 'realmDetail',
|
name: 'realmDetail',
|
||||||
builder: (context, state) => RealmDetailScreen(alias: state.pathParameters['alias']!),
|
builder: (context, state) =>
|
||||||
|
RealmDetailScreen(alias: state.pathParameters['alias']!),
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
GoRoute(path: '/news', name: 'news', builder: (context, state) => const NewsScreen(), routes: [
|
GoRoute(
|
||||||
GoRoute(
|
path: '/news',
|
||||||
path: '/:hash',
|
name: 'news',
|
||||||
name: 'newsDetail',
|
builder: (context, state) => const NewsScreen(),
|
||||||
builder: (context, state) => NewsDetailScreen(
|
routes: [
|
||||||
hash: state.pathParameters['hash']!,
|
GoRoute(
|
||||||
|
path: '/:hash',
|
||||||
|
name: 'newsDetail',
|
||||||
|
builder: (context, state) => NewsDetailScreen(
|
||||||
|
hash: state.pathParameters['hash']!,
|
||||||
|
),
|
||||||
),
|
),
|
||||||
),
|
],
|
||||||
]),
|
),
|
||||||
|
GoRoute(
|
||||||
|
path: '/stickers',
|
||||||
|
name: 'stickers',
|
||||||
|
builder: (context, state) => const StickerScreen(),
|
||||||
|
),
|
||||||
GoRoute(
|
GoRoute(
|
||||||
path: '/album',
|
path: '/album',
|
||||||
name: 'album',
|
name: 'album',
|
||||||
|
@ -572,7 +572,12 @@ class _SettingsScreenState extends State<SettingsScreen> {
|
|||||||
trailing: const Icon(Symbols.chevron_right),
|
trailing: const Icon(Symbols.chevron_right),
|
||||||
onTap: () {
|
onTap: () {
|
||||||
final nty = context.read<NotificationProvider>();
|
final nty = context.read<NotificationProvider>();
|
||||||
nty.registerPushNotifications();
|
try {
|
||||||
|
nty.registerPushNotifications();
|
||||||
|
} catch (err) {
|
||||||
|
if (!mounted) return;
|
||||||
|
context.showErrorDialog(err);
|
||||||
|
}
|
||||||
},
|
},
|
||||||
),
|
),
|
||||||
ListTile(
|
ListTile(
|
||||||
|
304
lib/screens/stickers.dart
Normal file
304
lib/screens/stickers.dart
Normal file
@ -0,0 +1,304 @@
|
|||||||
|
import 'package:easy_localization/easy_localization.dart';
|
||||||
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:gap/gap.dart';
|
||||||
|
import 'package:material_symbols_icons/symbols.dart';
|
||||||
|
import 'package:provider/provider.dart';
|
||||||
|
import 'package:styled_widget/styled_widget.dart';
|
||||||
|
import 'package:surface/providers/sn_network.dart';
|
||||||
|
import 'package:surface/providers/sn_sticker.dart';
|
||||||
|
import 'package:surface/providers/userinfo.dart';
|
||||||
|
import 'package:surface/types/attachment.dart';
|
||||||
|
import 'package:surface/widgets/attachment/attachment_item.dart';
|
||||||
|
import 'package:surface/widgets/dialog.dart';
|
||||||
|
import 'package:surface/widgets/loading_indicator.dart';
|
||||||
|
import 'package:surface/widgets/navigation/app_scaffold.dart';
|
||||||
|
import 'package:very_good_infinite_list/very_good_infinite_list.dart';
|
||||||
|
|
||||||
|
class StickerScreen extends StatefulWidget {
|
||||||
|
const StickerScreen({super.key});
|
||||||
|
|
||||||
|
@override
|
||||||
|
State<StickerScreen> createState() => _StickerScreenState();
|
||||||
|
}
|
||||||
|
|
||||||
|
class _StickerScreenState extends State<StickerScreen>
|
||||||
|
with SingleTickerProviderStateMixin {
|
||||||
|
late final TabController _tabController =
|
||||||
|
TabController(length: 3, vsync: this);
|
||||||
|
|
||||||
|
bool _isBusy = false;
|
||||||
|
int? _totalCount;
|
||||||
|
final List<SnStickerPack> _packs = List.empty(growable: true);
|
||||||
|
|
||||||
|
Future<void> _fetchPacks() async {
|
||||||
|
try {
|
||||||
|
setState(() => _isBusy = true);
|
||||||
|
final sn = context.read<SnNetworkProvider>();
|
||||||
|
final ua = context.read<UserProvider>();
|
||||||
|
final resp = await sn.client.get(
|
||||||
|
_tabController.index == 1
|
||||||
|
? '/cgi/uc/stickers/packs/own'
|
||||||
|
: '/cgi/uc/stickers/packs',
|
||||||
|
queryParameters: {
|
||||||
|
'take': 10,
|
||||||
|
'offset': _packs.length,
|
||||||
|
if (_tabController.index == 2) 'author': ua.user?.id,
|
||||||
|
},
|
||||||
|
);
|
||||||
|
if (resp.data is Map<String, dynamic>) {
|
||||||
|
_totalCount = resp.data['count'] as int?;
|
||||||
|
final out = List<SnStickerPack>.from(
|
||||||
|
resp.data['data'].map((ele) => SnStickerPack.fromJson(ele)),
|
||||||
|
);
|
||||||
|
_packs.addAll(out);
|
||||||
|
} else {
|
||||||
|
_totalCount = 0;
|
||||||
|
final out = List<SnStickerPack>.from(
|
||||||
|
resp.data.map((ele) => SnStickerPack.fromJson(ele)),
|
||||||
|
);
|
||||||
|
_packs.addAll(out);
|
||||||
|
}
|
||||||
|
} catch (err) {
|
||||||
|
if (!mounted) return;
|
||||||
|
context.showErrorDialog(err);
|
||||||
|
} finally {
|
||||||
|
setState(() => _isBusy = false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Future<void> _removePack(SnStickerPack pack) async {
|
||||||
|
try {
|
||||||
|
setState(() => _isBusy = true);
|
||||||
|
final sn = context.read<SnNetworkProvider>();
|
||||||
|
await sn.client.delete('/cgi/uc/stickers/packs/${pack.id}/own');
|
||||||
|
if (!mounted) return;
|
||||||
|
context.showSnackbar('stickersRemoved'.tr());
|
||||||
|
_refreshPacks();
|
||||||
|
} catch (err) {
|
||||||
|
if (!mounted) return;
|
||||||
|
context.showErrorDialog(err);
|
||||||
|
} finally {
|
||||||
|
setState(() => _isBusy = false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Future<void> _refreshPacks() async {
|
||||||
|
_packs.clear();
|
||||||
|
_totalCount = null;
|
||||||
|
await _fetchPacks();
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
void initState() {
|
||||||
|
super.initState();
|
||||||
|
_fetchPacks();
|
||||||
|
_tabController.addListener(() {
|
||||||
|
if (_tabController.indexIsChanging) {
|
||||||
|
_refreshPacks();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
void dispose() {
|
||||||
|
_tabController.dispose();
|
||||||
|
super.dispose();
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
return AppScaffold(
|
||||||
|
appBar: AppBar(
|
||||||
|
title: Text('screenStickers').tr(),
|
||||||
|
bottom: TabBar(
|
||||||
|
controller: _tabController,
|
||||||
|
tabs: [
|
||||||
|
Tab(
|
||||||
|
child: Text('stickersDiscovery'.tr()).textColor(
|
||||||
|
Theme.of(context).appBarTheme.foregroundColor,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
Tab(
|
||||||
|
child: Text('stickersOwned'.tr()).textColor(
|
||||||
|
Theme.of(context).appBarTheme.foregroundColor,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
Tab(
|
||||||
|
child: Text('stickersCreated'.tr()).textColor(
|
||||||
|
Theme.of(context).appBarTheme.foregroundColor,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
body: MediaQuery.removePadding(
|
||||||
|
context: context,
|
||||||
|
removeTop: true,
|
||||||
|
child: RefreshIndicator(
|
||||||
|
onRefresh: _refreshPacks,
|
||||||
|
child: InfiniteList(
|
||||||
|
itemCount: _packs.length,
|
||||||
|
onFetchData: _fetchPacks,
|
||||||
|
hasReachedMax: _totalCount != null && _packs.length >= _totalCount!,
|
||||||
|
isLoading: _isBusy,
|
||||||
|
itemBuilder: (context, idx) {
|
||||||
|
final pack = _packs[idx];
|
||||||
|
return ListTile(
|
||||||
|
title: Text(pack.name),
|
||||||
|
subtitle: Text(
|
||||||
|
pack.description,
|
||||||
|
maxLines: 1,
|
||||||
|
overflow: TextOverflow.ellipsis,
|
||||||
|
),
|
||||||
|
contentPadding: const EdgeInsets.symmetric(horizontal: 16),
|
||||||
|
trailing: _tabController.index == 1
|
||||||
|
? IconButton(
|
||||||
|
onPressed: () {
|
||||||
|
_removePack(pack);
|
||||||
|
},
|
||||||
|
icon: const Icon(Symbols.remove),
|
||||||
|
)
|
||||||
|
: null,
|
||||||
|
onTap: () {
|
||||||
|
showModalBottomSheet(
|
||||||
|
context: context,
|
||||||
|
builder: (context) => _StickerPackAddPopup(pack: pack),
|
||||||
|
).then((value) {
|
||||||
|
if (value == true && _tabController.index == 1) {
|
||||||
|
_refreshPacks();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
},
|
||||||
|
);
|
||||||
|
},
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class _StickerPackAddPopup extends StatefulWidget {
|
||||||
|
final SnStickerPack pack;
|
||||||
|
const _StickerPackAddPopup({required this.pack});
|
||||||
|
|
||||||
|
@override
|
||||||
|
State<_StickerPackAddPopup> createState() => _StickerPackAddPopupState();
|
||||||
|
}
|
||||||
|
|
||||||
|
class _StickerPackAddPopupState extends State<_StickerPackAddPopup> {
|
||||||
|
SnStickerPack? _pack;
|
||||||
|
|
||||||
|
bool _isBusy = false;
|
||||||
|
|
||||||
|
Future<void> _fetchPack() async {
|
||||||
|
try {
|
||||||
|
setState(() => _isBusy = true);
|
||||||
|
final sn = context.read<SnNetworkProvider>();
|
||||||
|
final resp =
|
||||||
|
await sn.client.get('/cgi/uc/stickers/packs/${widget.pack.id}');
|
||||||
|
_pack = SnStickerPack.fromJson(resp.data);
|
||||||
|
} catch (err) {
|
||||||
|
if (!mounted) return;
|
||||||
|
context.showErrorDialog(err);
|
||||||
|
} finally {
|
||||||
|
setState(() => _isBusy = false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
void initState() {
|
||||||
|
super.initState();
|
||||||
|
_fetchPack();
|
||||||
|
}
|
||||||
|
|
||||||
|
bool _isAdding = false;
|
||||||
|
|
||||||
|
Future<void> _addPack() async {
|
||||||
|
if (_pack == null) return;
|
||||||
|
|
||||||
|
try {
|
||||||
|
setState(() => _isAdding = true);
|
||||||
|
final sn = context.read<SnNetworkProvider>();
|
||||||
|
final stickers = context.read<SnStickerProvider>();
|
||||||
|
await sn.client.post(
|
||||||
|
'/cgi/uc/stickers/packs/${widget.pack.id}/own',
|
||||||
|
);
|
||||||
|
if (!mounted) return;
|
||||||
|
context.showSnackbar('stickersAdded'.tr());
|
||||||
|
if (_pack?.stickers != null) stickers.putSticker(_pack!.stickers!);
|
||||||
|
Navigator.pop(context, true);
|
||||||
|
} catch (err) {
|
||||||
|
if (!mounted) return;
|
||||||
|
context.showErrorDialog(err);
|
||||||
|
} finally {
|
||||||
|
setState(() => _isAdding = false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
return Column(
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
|
children: [
|
||||||
|
Row(
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.center,
|
||||||
|
children: [
|
||||||
|
const Icon(Symbols.add, size: 24),
|
||||||
|
const Gap(16),
|
||||||
|
Text('stickersAdd', style: Theme.of(context).textTheme.titleLarge)
|
||||||
|
.tr(),
|
||||||
|
],
|
||||||
|
).padding(horizontal: 20, top: 16, bottom: 12),
|
||||||
|
Row(
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.end,
|
||||||
|
children: [
|
||||||
|
Expanded(
|
||||||
|
child: Column(
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
|
children: [
|
||||||
|
Text(widget.pack.name).bold(),
|
||||||
|
Text(
|
||||||
|
widget.pack.description,
|
||||||
|
maxLines: 2,
|
||||||
|
overflow: TextOverflow.ellipsis,
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
ElevatedButton(
|
||||||
|
onPressed: _isAdding ? null : _addPack,
|
||||||
|
child: Text('add').tr(),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
).padding(horizontal: 24),
|
||||||
|
LoadingIndicator(isActive: _isBusy),
|
||||||
|
if (_pack?.stickers != null)
|
||||||
|
Expanded(
|
||||||
|
child: GridView.extent(
|
||||||
|
padding: EdgeInsets.only(left: 20, right: 20, top: 8),
|
||||||
|
maxCrossAxisExtent: 48,
|
||||||
|
mainAxisSpacing: 8,
|
||||||
|
crossAxisSpacing: 8,
|
||||||
|
children: _pack!.stickers!
|
||||||
|
.map(
|
||||||
|
(ele) => ClipRRect(
|
||||||
|
borderRadius: BorderRadius.circular(8),
|
||||||
|
child: Container(
|
||||||
|
color:
|
||||||
|
Theme.of(context).colorScheme.surfaceContainerHigh,
|
||||||
|
child: AttachmentItem(
|
||||||
|
data: ele.attachment,
|
||||||
|
heroTag: 'sticker-pack-${ele.attachment.rid}',
|
||||||
|
fit: BoxFit.contain,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
)
|
||||||
|
.toList(),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user