170 lines
		
	
	
		
			5.8 KiB
		
	
	
	
		
			Dart
		
	
	
	
	
	
			
		
		
	
	
			170 lines
		
	
	
		
			5.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/webfeed.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 'feed_marketplace.g.dart';
 | |
| 
 | |
| @riverpod
 | |
| class MarketplaceWebFeedsNotifier extends _$MarketplaceWebFeedsNotifier
 | |
|     with CursorPagingNotifierMixin<SnWebFeed> {
 | |
|   String? _query;
 | |
| 
 | |
|   @override
 | |
|   Future<CursorPagingData<SnWebFeed>> build({required String? query}) {
 | |
|     _query = query;
 | |
|     return fetch(cursor: null);
 | |
|   }
 | |
| 
 | |
|   @override
 | |
|   Future<CursorPagingData<SnWebFeed>> fetch({required String? cursor}) async {
 | |
|     final client = ref.read(apiClientProvider);
 | |
|     final offset = cursor == null ? 0 : int.parse(cursor);
 | |
| 
 | |
|     final response = await client.get(
 | |
|       '/sphere/feeds/explore',
 | |
|       queryParameters: {
 | |
|         'offset': offset,
 | |
|         'take': 20,
 | |
|         if (_query != null && _query!.isNotEmpty) 'query': _query,
 | |
|       },
 | |
|     );
 | |
| 
 | |
|     final total = int.parse(response.headers.value('X-Total') ?? '0');
 | |
|     final List<dynamic> data = response.data;
 | |
|     final feeds = data.map((e) => SnWebFeed.fromJson(e)).toList();
 | |
| 
 | |
|     final hasMore = offset + feeds.length < total;
 | |
|     final nextCursor = hasMore ? (offset + feeds.length).toString() : null;
 | |
| 
 | |
|     return CursorPagingData(
 | |
|       items: feeds,
 | |
|       hasMore: hasMore,
 | |
|       nextCursor: nextCursor,
 | |
|     );
 | |
|   }
 | |
| }
 | |
| 
 | |
| /// Marketplace screen for browsing web feeds.
 | |
| class MarketplaceWebFeedsScreen extends HookConsumerWidget {
 | |
|   const MarketplaceWebFeedsScreen({super.key});
 | |
| 
 | |
|   @override
 | |
|   Widget build(BuildContext context, WidgetRef ref) {
 | |
|     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('webFeeds').tr(),
 | |
|         actions: const [Gap(8)],
 | |
|       ),
 | |
|       body: PagingHelperView(
 | |
|         provider: marketplaceWebFeedsNotifierProvider(query: query.value),
 | |
|         futureRefreshable:
 | |
|             marketplaceWebFeedsNotifierProvider(query: query.value).future,
 | |
|         notifierRefreshable:
 | |
|             marketplaceWebFeedsNotifierProvider(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 feed = data.items[index];
 | |
|                       return ListTile(
 | |
|                         title: Text(feed.title),
 | |
|                         subtitle: Text(feed.description ?? ''),
 | |
|                         trailing: const Icon(Symbols.chevron_right),
 | |
|                         onTap: () {
 | |
|                           // Navigate to web feed detail page
 | |
|                           context.pushNamed(
 | |
|                             'webFeedDetail',
 | |
|                             pathParameters: {'feedId': feed.id},
 | |
|                           );
 | |
|                         },
 | |
|                       );
 | |
|                     },
 | |
|                   ),
 | |
|                 ),
 | |
|               ],
 | |
|             ),
 | |
|       ),
 | |
|     );
 | |
|   }
 | |
| }
 |