200 lines
		
	
	
		
			6.8 KiB
		
	
	
	
		
			Dart
		
	
	
	
	
	
			
		
		
	
	
			200 lines
		
	
	
		
			6.8 KiB
		
	
	
	
		
			Dart
		
	
	
	
	
	
import 'package:easy_localization/easy_localization.dart';
 | 
						|
import 'package:flutter/material.dart';
 | 
						|
import 'package:flutter_hooks/flutter_hooks.dart';
 | 
						|
import 'package:go_router/go_router.dart';
 | 
						|
import 'package:gap/gap.dart';
 | 
						|
import 'package:hooks_riverpod/hooks_riverpod.dart';
 | 
						|
import 'package:island/models/sticker.dart';
 | 
						|
import 'package:island/pods/network.dart';
 | 
						|
import 'package:island/widgets/app_scaffold.dart';
 | 
						|
import 'package:material_symbols_icons/symbols.dart';
 | 
						|
import 'dart:async';
 | 
						|
 | 
						|
import 'package:riverpod_annotation/riverpod_annotation.dart';
 | 
						|
import 'package:riverpod_paging_utils/riverpod_paging_utils.dart';
 | 
						|
 | 
						|
part 'sticker_marketplace.g.dart';
 | 
						|
 | 
						|
@riverpod
 | 
						|
class MarketplaceStickerPacksNotifier extends _$MarketplaceStickerPacksNotifier
 | 
						|
    with CursorPagingNotifierMixin<SnStickerPack> {
 | 
						|
  @override
 | 
						|
  Future<CursorPagingData<SnStickerPack>> build({
 | 
						|
    required String? query,
 | 
						|
    required bool byUsage,
 | 
						|
  }) {
 | 
						|
    return fetch(cursor: null);
 | 
						|
  }
 | 
						|
 | 
						|
  @override
 | 
						|
  Future<CursorPagingData<SnStickerPack>> fetch({
 | 
						|
    required String? cursor,
 | 
						|
  }) async {
 | 
						|
    final client = ref.read(apiClientProvider);
 | 
						|
    final offset = cursor == null ? 0 : int.parse(cursor);
 | 
						|
 | 
						|
    final response = await client.get(
 | 
						|
      '/sphere/stickers',
 | 
						|
      queryParameters: {
 | 
						|
        'offset': offset,
 | 
						|
        'take': 20,
 | 
						|
        'order': byUsage ? 'usage' : 'date',
 | 
						|
        if (query != null && query!.isNotEmpty) 'query': query,
 | 
						|
      },
 | 
						|
    );
 | 
						|
 | 
						|
    final total = int.parse(response.headers.value('X-Total') ?? '0');
 | 
						|
    final List<dynamic> data = response.data;
 | 
						|
    final stickers = data.map((e) => SnStickerPack.fromJson(e)).toList();
 | 
						|
 | 
						|
    final hasMore = offset + stickers.length < total;
 | 
						|
    final nextCursor = hasMore ? (offset + stickers.length).toString() : null;
 | 
						|
 | 
						|
    return CursorPagingData(
 | 
						|
      items: stickers,
 | 
						|
      hasMore: hasMore,
 | 
						|
      nextCursor: nextCursor,
 | 
						|
    );
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
/// User-facing marketplace screen for browsing sticker packs.
 | 
						|
/// This version does NOT rely on publisher name (no pubName).
 | 
						|
class MarketplaceStickersScreen extends HookConsumerWidget {
 | 
						|
  const MarketplaceStickersScreen({super.key});
 | 
						|
 | 
						|
  @override
 | 
						|
  Widget build(BuildContext context, WidgetRef ref) {
 | 
						|
    final byUsage = useState(true);
 | 
						|
    final query = useState<String?>(null);
 | 
						|
    final searchController = useTextEditingController();
 | 
						|
    final focusNode = useFocusNode();
 | 
						|
    final debounceTimer = useState<Timer?>(null);
 | 
						|
 | 
						|
    // Clear search when query is cleared
 | 
						|
    useEffect(() {
 | 
						|
      if (query.value == null || query.value!.isEmpty) {
 | 
						|
        searchController.clear();
 | 
						|
      }
 | 
						|
      return null;
 | 
						|
    }, [query]);
 | 
						|
 | 
						|
    // Clean up timer on dispose
 | 
						|
    useEffect(() {
 | 
						|
      return () {
 | 
						|
        debounceTimer.value?.cancel();
 | 
						|
      };
 | 
						|
    }, []);
 | 
						|
 | 
						|
    return AppScaffold(
 | 
						|
      appBar: AppBar(
 | 
						|
        title: const Text('stickers').tr(),
 | 
						|
        actions: [
 | 
						|
          IconButton(
 | 
						|
            onPressed: () {
 | 
						|
              byUsage.value = !byUsage.value;
 | 
						|
            },
 | 
						|
            icon:
 | 
						|
                byUsage.value
 | 
						|
                    ? const Icon(Symbols.local_fire_department)
 | 
						|
                    : const Icon(Symbols.access_time),
 | 
						|
            tooltip:
 | 
						|
                byUsage.value
 | 
						|
                    ? 'orderByPopularity'.tr()
 | 
						|
                    : 'orderByReleaseDate'.tr(),
 | 
						|
          ),
 | 
						|
          const Gap(8),
 | 
						|
        ],
 | 
						|
      ),
 | 
						|
      body: PagingHelperView(
 | 
						|
        provider: marketplaceStickerPacksNotifierProvider(
 | 
						|
          byUsage: byUsage.value,
 | 
						|
          query: query.value,
 | 
						|
        ),
 | 
						|
        futureRefreshable:
 | 
						|
            marketplaceStickerPacksNotifierProvider(
 | 
						|
              byUsage: byUsage.value,
 | 
						|
              query: query.value,
 | 
						|
            ).future,
 | 
						|
        notifierRefreshable:
 | 
						|
            marketplaceStickerPacksNotifierProvider(
 | 
						|
              byUsage: byUsage.value,
 | 
						|
              query: query.value,
 | 
						|
            ).notifier,
 | 
						|
        contentBuilder:
 | 
						|
            (data, widgetCount, endItemView) => Column(
 | 
						|
              children: [
 | 
						|
                // Search bar above the list
 | 
						|
                Padding(
 | 
						|
                  padding: const EdgeInsets.all(16),
 | 
						|
                  child: SearchBar(
 | 
						|
                    elevation: WidgetStateProperty.all(4),
 | 
						|
                    controller: searchController,
 | 
						|
                    focusNode: focusNode,
 | 
						|
                    hintText: 'search'.tr(),
 | 
						|
                    leading: const Icon(Symbols.search),
 | 
						|
                    padding: WidgetStateProperty.all(
 | 
						|
                      const EdgeInsets.symmetric(horizontal: 24),
 | 
						|
                    ),
 | 
						|
                    onTapOutside:
 | 
						|
                        (_) => FocusManager.instance.primaryFocus?.unfocus(),
 | 
						|
                    trailing: [
 | 
						|
                      if (query.value != null && query.value!.isNotEmpty)
 | 
						|
                        IconButton(
 | 
						|
                          icon: const Icon(Symbols.close),
 | 
						|
                          onPressed: () {
 | 
						|
                            query.value = null;
 | 
						|
                            searchController.clear();
 | 
						|
                            focusNode.unfocus();
 | 
						|
                          },
 | 
						|
                        ),
 | 
						|
                    ],
 | 
						|
                    onChanged: (value) {
 | 
						|
                      // Debounce search to avoid excessive API calls
 | 
						|
                      debounceTimer.value?.cancel();
 | 
						|
                      debounceTimer.value = Timer(
 | 
						|
                        const Duration(milliseconds: 500),
 | 
						|
                        () {
 | 
						|
                          query.value = value.isEmpty ? null : value;
 | 
						|
                        },
 | 
						|
                      );
 | 
						|
                    },
 | 
						|
                    onSubmitted: (value) {
 | 
						|
                      query.value = value.isEmpty ? null : value;
 | 
						|
                      focusNode.unfocus();
 | 
						|
                    },
 | 
						|
                  ),
 | 
						|
                ),
 | 
						|
                Expanded(
 | 
						|
                  child: ListView.builder(
 | 
						|
                    padding: EdgeInsets.zero,
 | 
						|
                    itemCount: widgetCount,
 | 
						|
                    itemBuilder: (context, index) {
 | 
						|
                      if (index == widgetCount - 1) {
 | 
						|
                        return endItemView;
 | 
						|
                      }
 | 
						|
 | 
						|
                      final pack = data.items[index];
 | 
						|
                      return ListTile(
 | 
						|
                        title: Text(pack.name),
 | 
						|
                        subtitle: Text(pack.description),
 | 
						|
                        trailing: const Icon(Symbols.chevron_right),
 | 
						|
                        onTap: () {
 | 
						|
                          // Navigate to user-facing sticker pack detail page.
 | 
						|
                          // Adjust the route name/parameters if your app uses different ones.
 | 
						|
                          context.pushNamed(
 | 
						|
                            'stickerPackDetail',
 | 
						|
                            pathParameters: {'packId': pack.id},
 | 
						|
                          );
 | 
						|
                        },
 | 
						|
                      );
 | 
						|
                    },
 | 
						|
                  ),
 | 
						|
                ),
 | 
						|
              ],
 | 
						|
            ),
 | 
						|
      ),
 | 
						|
    );
 | 
						|
  }
 | 
						|
}
 |