Files
iiEsaywebUIapp/lib/features/navigation/views/chats_list_page.dart

1884 lines
67 KiB
Dart
Raw Normal View History

2025-08-10 01:20:45 +05:30
import 'package:flutter/material.dart';
import '../../../core/widgets/error_boundary.dart';
import '../../../core/services/focus_management_service.dart';
import '../../../shared/theme/theme_extensions.dart';
import 'package:flutter/cupertino.dart';
import 'package:flutter_riverpod/flutter_riverpod.dart';
import 'package:flutter_animate/flutter_animate.dart';
import '../../../shared/widgets/loading_states.dart';
import 'dart:async';
import 'dart:io' show Platform;
import '../../../core/providers/app_providers.dart';
import '../../../shared/widgets/themed_dialogs.dart';
import '../../../shared/widgets/conduit_components.dart';
import '../../chat/providers/chat_providers.dart';
2025-08-17 00:05:30 +05:30
import '../../chat/views/chat_page_helpers.dart';
2025-08-10 01:20:45 +05:30
/// Optimized conversation list page with Conduit design aesthetics
class ChatsListPage extends ConsumerStatefulWidget {
final bool isOverlay;
const ChatsListPage({super.key, this.isOverlay = false});
@override
ConsumerState<ChatsListPage> createState() => _ChatsListPageState();
}
class _ChatsListPageState extends ConsumerState<ChatsListPage>
with AutomaticKeepAliveClientMixin {
final TextEditingController _searchController = TextEditingController();
late final FocusNode _searchFocusNode;
final ScrollController _scrollController = ScrollController();
// Debounce search to improve performance
String _searchQuery = '';
Timer? _debounceTimer;
bool _isLoadingConversation = false;
bool _hasAddedFocusListener = false;
// Provider for archived section visibility
static final _showArchivedProvider = StateProvider<bool>((ref) => false);
2025-08-17 00:05:30 +05:30
// Provider for folder expansion state (Map<folderId, isExpanded>)
2025-08-17 16:11:19 +05:30
// Start with folders expanded by default for better discoverability
2025-08-17 00:05:30 +05:30
static final _expandedFoldersProvider = StateProvider<Map<String, bool>>((ref) => {});
2025-08-10 01:20:45 +05:30
@override
bool get wantKeepAlive => true; // Keep state alive for better performance
@override
void initState() {
super.initState();
_searchFocusNode = FocusManagementService.registerFocusNode(
'chats_list_search',
debugLabel: 'Chats List Search',
);
_searchController.addListener(_onSearchChanged);
}
@override
void dispose() {
_searchController.removeListener(_onSearchChanged);
_searchController.dispose();
_scrollController.dispose();
_debounceTimer?.cancel();
FocusManagementService.disposeFocusNode('chats_list_search');
super.dispose();
}
void _onSearchChanged() {
// Cancel previous timer
_debounceTimer?.cancel();
// Set new timer for debounced search
_debounceTimer = Timer(const Duration(milliseconds: 300), () {
if (_searchQuery != _searchController.text) {
setState(() {
_searchQuery = _searchController.text;
});
ref.read(searchQueryProvider.notifier).state = _searchQuery;
}
});
}
@override
Widget build(BuildContext context) {
super.build(context); // Required for AutomaticKeepAliveClientMixin
return ErrorBoundary(
child: Scaffold(
backgroundColor: context.conduitTheme.surfaceBackground,
appBar: _buildAppBar(),
body: Column(
children: [
_buildSearchBar(),
Expanded(child: _wrapWithRefresh(_buildConversationsList())),
_buildBottomActions(),
],
),
floatingActionButton: FloatingActionButton(
2025-08-17 00:05:30 +05:30
onPressed: _showCreateMenu,
2025-08-10 01:20:45 +05:30
backgroundColor: context.conduitTheme.buttonPrimary,
foregroundColor: context.conduitTheme.buttonPrimaryText,
elevation: Elevation.medium,
shape: RoundedRectangleBorder(
borderRadius: BorderRadius.circular(AppBorderRadius.floatingButton),
),
child: Icon(
Platform.isIOS ? CupertinoIcons.plus : Icons.add_rounded,
size: IconSize.large,
),
),
),
);
}
PreferredSizeWidget _buildAppBar() {
return AppBar(
backgroundColor: context.conduitTheme.surfaceBackground,
elevation: Elevation.none,
scrolledUnderElevation: Elevation.none,
leading: widget.isOverlay
? ConduitIconButton(
icon: Platform.isIOS ? CupertinoIcons.xmark : Icons.close_rounded,
onPressed: () => Navigator.pop(context),
)
: ConduitIconButton(
icon: Platform.isIOS
? CupertinoIcons.back
: Icons.arrow_back_rounded,
onPressed: () => Navigator.pop(context),
),
title: Text(
'Chats',
style: AppTypography.headlineMediumStyle.copyWith(
color: context.conduitTheme.textPrimary,
fontWeight: FontWeight.w600,
),
),
actions: [
ConduitIconButton(
icon: Platform.isIOS
? CupertinoIcons.ellipsis
: Icons.more_vert_rounded,
onPressed: _showOptions,
),
],
);
}
Widget _buildSearchBar() {
// Listen to focus changes and update UI
final isFocused = _searchFocusNode.hasFocus;
// Attach listener only once
if (!_hasAddedFocusListener) {
_searchFocusNode.addListener(() {
setState(() {});
});
_hasAddedFocusListener = true;
}
return GestureDetector(
onTap: () {
// Focus the search field when the container is tapped
_searchFocusNode.requestFocus();
},
child: Container(
2025-08-17 00:05:30 +05:30
margin: const EdgeInsets.all(Spacing.md),
2025-08-10 01:20:45 +05:30
decoration: BoxDecoration(
2025-08-17 00:05:30 +05:30
gradient: LinearGradient(
colors: [
context.conduitTheme.inputBackground.withValues(alpha: 0.6),
context.conduitTheme.inputBackground.withValues(alpha: 0.3),
],
begin: Alignment.topLeft,
end: Alignment.bottomRight,
),
borderRadius: BorderRadius.circular(AppBorderRadius.lg),
2025-08-10 01:20:45 +05:30
border: Border.all(
color: isFocused
2025-08-17 00:05:30 +05:30
? context.conduitTheme.buttonPrimary.withValues(alpha: 0.8)
: context.conduitTheme.inputBorder.withValues(alpha: 0.3),
width: BorderWidth.thin,
2025-08-10 01:20:45 +05:30
),
),
2025-08-17 00:05:30 +05:30
child: TextField(
controller: _searchController,
focusNode: _searchFocusNode,
style: TextStyle(
color: context.conduitTheme.inputText,
fontSize: AppTypography.bodyMedium,
),
decoration: InputDecoration(
hintText: 'Search conversations...',
hintStyle: TextStyle(
color: context.conduitTheme.inputPlaceholder.withValues(alpha: 0.8),
fontSize: AppTypography.bodyMedium,
),
prefixIcon: Icon(
Platform.isIOS ? CupertinoIcons.search : Icons.search,
2025-08-10 01:20:45 +05:30
color: context.conduitTheme.iconSecondary,
2025-08-17 00:05:30 +05:30
size: IconSize.md,
2025-08-10 01:20:45 +05:30
),
2025-08-17 00:05:30 +05:30
suffixIcon: _searchQuery.isNotEmpty
? IconButton(
icon: Icon(
Platform.isIOS
? CupertinoIcons.clear_circled_solid
: Icons.clear,
color: context.conduitTheme.iconSecondary,
size: IconSize.md,
),
onPressed: () {
_searchController.clear();
_searchQuery = '';
ref.read(searchQueryProvider.notifier).state = '';
_searchFocusNode.unfocus();
},
)
: null,
border: InputBorder.none,
contentPadding: const EdgeInsets.symmetric(
horizontal: 16,
vertical: 14,
2025-08-10 01:20:45 +05:30
),
2025-08-17 00:05:30 +05:30
),
2025-08-10 01:20:45 +05:30
),
),
).animate().fadeIn(
duration: AnimationDuration.microInteraction,
curve: AnimationCurves.microInteraction,
);
}
Widget _buildConversationsList() {
return Consumer(
builder: (context, ref, child) {
2025-08-17 16:11:19 +05:30
// Use ref.watch to properly react to changes
2025-08-10 01:20:45 +05:30
final conversationsAsync = ref.watch(conversationsProvider);
return conversationsAsync.when(
data: (conversations) {
if (conversations.isEmpty) {
return _buildEmptyState();
}
final filteredConversations = _filterConversations(conversations);
if (filteredConversations.isEmpty) {
return _buildNoResultsState();
}
2025-08-17 16:11:19 +05:30
// Deduplicate by ID as a safety measure in case provider has duplicates
final deduplicatedConversations = <String, dynamic>{};
for (final conv in filteredConversations) {
deduplicatedConversations[conv.id] = conv;
}
final uniqueConversations = deduplicatedConversations.values.toList();
2025-08-17 00:05:30 +05:30
// Separate conversations by status and folder
2025-08-17 16:11:19 +05:30
final pinnedConversations = uniqueConversations
2025-08-10 01:20:45 +05:30
.where((c) => c.pinned == true)
.toList();
2025-08-17 16:11:19 +05:30
final regularConversations = uniqueConversations
2025-08-17 00:05:30 +05:30
.where((c) => c.pinned != true && c.archived != true && (c.folderId == null || c.folderId!.isEmpty))
.toList();
2025-08-17 16:11:19 +05:30
final folderConversations = uniqueConversations
2025-08-17 00:05:30 +05:30
.where((c) => c.pinned != true && c.archived != true && c.folderId != null && c.folderId!.isNotEmpty)
2025-08-10 01:20:45 +05:30
.toList();
2025-08-17 16:11:19 +05:30
final archivedConversations = uniqueConversations
2025-08-10 01:20:45 +05:30
.where((c) => c.archived == true)
.toList();
2025-08-17 00:05:30 +05:30
// Debug logging
2025-08-17 16:11:19 +05:30
print('🔍 DEBUG: Total conversations: ${uniqueConversations.length} (filtered: ${filteredConversations.length}, original: ${conversations.length})');
2025-08-17 00:05:30 +05:30
print('🔍 DEBUG: Pinned: ${pinnedConversations.length}');
print('🔍 DEBUG: Regular: ${regularConversations.length}');
print('🔍 DEBUG: Folder: ${folderConversations.length}');
print('🔍 DEBUG: Archived: ${archivedConversations.length}');
// Check first few conversations for folder IDs
2025-08-17 16:11:19 +05:30
for (int i = 0; i < uniqueConversations.take(5).length; i++) {
final conv = uniqueConversations[i];
2025-08-17 00:05:30 +05:30
print('🔍 DEBUG: Conv ${i}: id=${conv.id.substring(0, 8)}, folderId=${conv.folderId}, pinned=${conv.pinned}, archived=${conv.archived}');
}
2025-08-10 01:20:45 +05:30
return ListView(
controller: _scrollController,
2025-08-17 00:05:30 +05:30
padding: const EdgeInsets.all(Spacing.md),
2025-08-10 01:20:45 +05:30
children: [
// Pinned conversations section
if (pinnedConversations.isNotEmpty) ...[
_buildSectionHeader('Pinned', pinnedConversations.length),
...pinnedConversations.asMap().entries.map((entry) {
return _buildConversationTile(
entry.value,
entry.key,
isPinned: true,
);
}),
2025-08-17 00:05:30 +05:30
const SizedBox(height: Spacing.md),
],
// Folder conversations sections (after pinned, before recent)
if (folderConversations.isNotEmpty) ...[
...ref.watch(foldersProvider).when(
data: (folders) {
// Group conversations by folder
final groupedByFolder = <String, List<dynamic>>{};
for (final conv in folderConversations) {
if (conv.folderId != null) {
groupedByFolder.putIfAbsent(conv.folderId!, () => []).add(conv);
}
}
// Build folder sections
return folders.where((folder) => groupedByFolder.containsKey(folder.id)).map((folder) {
final conversations = groupedByFolder[folder.id]!;
final expandedFolders = ref.watch(_expandedFoldersProvider);
final isExpanded = expandedFolders[folder.id] ?? false;
return Column(
children: [
_buildFolderHeader(folder.id, folder.name, conversations.length),
// Only show conversations if folder is expanded
if (isExpanded) ...[
...conversations.asMap().entries.map((entry) {
return _buildConversationTile(entry.value, entry.key, inFolder: true);
}),
],
const SizedBox(height: Spacing.md),
],
);
}).toList();
},
loading: () => [const SizedBox.shrink()],
error: (_, __) => [const SizedBox.shrink()],
),
2025-08-10 01:20:45 +05:30
],
// Regular conversations section
if (regularConversations.isNotEmpty) ...[
_buildSectionHeader('Recent', regularConversations.length),
...regularConversations.asMap().entries.map((entry) {
return _buildConversationTile(entry.value, entry.key);
}),
],
// Archived conversations section (collapsed by default)
if (archivedConversations.isNotEmpty) ...[
2025-08-17 00:05:30 +05:30
const SizedBox(height: Spacing.md),
2025-08-10 01:20:45 +05:30
_buildArchivedSection(archivedConversations),
],
],
);
},
loading: () => _buildLoadingState(),
error: (error, stackTrace) => _buildErrorState(error),
);
},
);
}
Widget _wrapWithRefresh(Widget child) {
return ConduitRefreshIndicator(
onRefresh: () async {
2025-08-17 16:11:19 +05:30
// Invalidate to force a fresh fetch
2025-08-10 01:20:45 +05:30
ref.invalidate(conversationsProvider);
2025-08-17 16:11:19 +05:30
// Wait for the provider to complete
await ref.read(conversationsProvider.future);
2025-08-10 01:20:45 +05:30
},
child: child,
);
}
Widget _buildConversationTile(
dynamic conversation,
int index, {
bool isPinned = false,
bool isArchived = false,
2025-08-17 00:05:30 +05:30
bool inFolder = false,
2025-08-10 01:20:45 +05:30
}) {
final isSelected =
ref.watch(activeConversationProvider)?.id == conversation.id;
final isLoading = _isLoadingConversation && isSelected;
2025-08-17 00:05:30 +05:30
return PressableScale(
onTap: isLoading ? null : () => _selectConversation(conversation),
borderRadius: BorderRadius.circular(AppBorderRadius.md),
child: Container(
margin: EdgeInsets.only(
bottom: Spacing.md,
left: inFolder ? Spacing.md : 0.0,
),
decoration: BoxDecoration(
gradient: isSelected
? LinearGradient(
colors: [
context.conduitTheme.buttonPrimary.withValues(alpha: 0.2),
context.conduitTheme.buttonPrimary.withValues(alpha: 0.1),
],
)
: null,
2025-08-10 01:20:45 +05:30
color: isSelected
2025-08-17 00:05:30 +05:30
? null
2025-08-10 01:20:45 +05:30
: isArchived
2025-08-17 00:05:30 +05:30
? context.conduitTheme.surfaceBackground.withValues(alpha: 0.3)
: context.conduitTheme.surfaceBackground.withValues(alpha: 0.05),
borderRadius: BorderRadius.circular(AppBorderRadius.md),
border: Border.all(
color: isSelected
? context.conduitTheme.buttonPrimary.withValues(alpha: 0.5)
: context.conduitTheme.dividerColor,
width: BorderWidth.regular,
),
boxShadow: isSelected ? ConduitShadows.card : null,
2025-08-10 01:20:45 +05:30
),
2025-08-17 00:05:30 +05:30
child: Padding(
padding: const EdgeInsets.symmetric(
horizontal: Spacing.md,
vertical: Spacing.sm,
),
child: Row(
children: [
// Conversation icon (32x32 like model selector)
Container(
width: 32,
height: 32,
decoration: BoxDecoration(
color: context.conduitTheme.buttonPrimary.withValues(
alpha: 0.15,
2025-08-10 01:20:45 +05:30
),
2025-08-17 00:05:30 +05:30
borderRadius: BorderRadius.circular(AppBorderRadius.md),
2025-08-10 01:20:45 +05:30
),
2025-08-17 00:05:30 +05:30
child: Icon(
Platform.isIOS
? CupertinoIcons.chat_bubble
: Icons.chat_rounded,
color: context.conduitTheme.buttonPrimary,
size: 16,
2025-08-10 01:20:45 +05:30
),
2025-08-17 00:05:30 +05:30
),
const SizedBox(width: Spacing.md),
2025-08-10 01:20:45 +05:30
2025-08-17 00:05:30 +05:30
// Conversation details
Expanded(
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
2025-08-10 01:20:45 +05:30
children: [
2025-08-17 00:05:30 +05:30
Text(
conversation.title ?? 'New Chat',
style: TextStyle(
color: isArchived
? context.conduitTheme.textSecondary
: context.conduitTheme.textPrimary,
fontWeight: FontWeight.w600,
fontSize: AppTypography.bodyMedium,
2025-08-10 01:20:45 +05:30
),
2025-08-17 00:05:30 +05:30
maxLines: 1,
overflow: TextOverflow.ellipsis,
),
const SizedBox(height: Spacing.xs),
Row(
children: [
if (isPinned)
_buildStatusChip(
icon: Platform.isIOS
? CupertinoIcons.pin_fill
: Icons.push_pin,
label: 'Pinned',
color: context.conduitTheme.warning,
2025-08-10 01:20:45 +05:30
),
2025-08-17 00:05:30 +05:30
if (isArchived)
_buildStatusChip(
icon: Platform.isIOS
? CupertinoIcons.archivebox_fill
: Icons.archive,
label: 'Archived',
color: context.conduitTheme.textSecondary,
2025-08-10 01:20:45 +05:30
),
2025-08-17 00:05:30 +05:30
if (!isPinned && !isArchived)
Text(
_formatConversationDate(conversation.updatedAt),
style: TextStyle(
color: context.conduitTheme.textTertiary,
fontSize: AppTypography.labelSmall,
2025-08-10 01:20:45 +05:30
),
),
2025-08-17 00:05:30 +05:30
],
),
2025-08-10 01:20:45 +05:30
],
),
2025-08-17 00:05:30 +05:30
),
// Action indicator (like model selector check)
GestureDetector(
onTap: () => _showConversationOptions(conversation),
child: AnimatedOpacity(
opacity: isSelected ? 1 : 0.6,
duration: AnimationDuration.fast,
child: Container(
padding: const EdgeInsets.all(Spacing.xxs),
decoration: BoxDecoration(
color: isSelected
? context.conduitTheme.buttonPrimary
: context.conduitTheme.surfaceBackground,
borderRadius: BorderRadius.circular(AppBorderRadius.md),
border: Border.all(
color: isSelected
? context.conduitTheme.buttonPrimary.withValues(
alpha: 0.6,
)
: context.conduitTheme.dividerColor,
),
),
child: isLoading
? SizedBox(
width: 14,
height: 14,
child: CircularProgressIndicator(
strokeWidth: 2,
valueColor: AlwaysStoppedAnimation<Color>(
isSelected
? context.conduitTheme.textInverse
: context.conduitTheme.buttonPrimary,
),
),
)
: Icon(
isSelected
? (Platform.isIOS
? CupertinoIcons.check_mark
: Icons.check)
: (Platform.isIOS
? CupertinoIcons.ellipsis
: Icons.more_vert),
color: isSelected
? context.conduitTheme.textInverse
: context.conduitTheme.iconSecondary,
size: 14,
),
),
),
),
],
2025-08-10 01:20:45 +05:30
),
),
),
).animate().fadeIn(
duration: AnimationDuration.messageAppear,
delay: Duration(
milliseconds: index * AnimationDelay.staggeredDelay.inMilliseconds,
),
curve: AnimationCurves.messageSlide,
);
}
2025-08-17 00:05:30 +05:30
Widget _buildStatusChip({
required IconData icon,
required String label,
required Color color,
}) {
return Container(
margin: const EdgeInsets.only(right: Spacing.xs),
padding: const EdgeInsets.symmetric(horizontal: Spacing.xs, vertical: 2),
decoration: BoxDecoration(
color: color.withValues(alpha: 0.08),
borderRadius: BorderRadius.circular(AppBorderRadius.chip),
border: Border.all(
color: color.withValues(alpha: 0.3),
width: BorderWidth.thin,
),
),
child: Row(
mainAxisSize: MainAxisSize.min,
children: [
Icon(icon, size: 12, color: color),
const SizedBox(width: 4),
Text(
label,
style: TextStyle(
fontSize: AppTypography.labelSmall,
color: color,
fontWeight: FontWeight.w500,
),
),
],
),
);
}
2025-08-10 01:20:45 +05:30
Widget _buildEmptyState() {
return Center(
child: Column(
mainAxisAlignment: MainAxisAlignment.center,
children: [
Icon(
Platform.isIOS ? CupertinoIcons.chat_bubble : Icons.chat_rounded,
size: IconSize.xxl,
color: context.conduitTheme.iconSecondary,
),
const SizedBox(height: Spacing.lg),
Text(
'No conversations yet',
style: AppTypography.headlineSmallStyle.copyWith(
color: context.conduitTheme.textPrimary,
fontWeight: FontWeight.w600,
),
),
const SizedBox(height: Spacing.sm),
Text(
'Start a new chat to begin your conversation',
style: AppTypography.bodyMediumStyle.copyWith(
color: context.conduitTheme.textSecondary,
),
textAlign: TextAlign.center,
),
const SizedBox(height: Spacing.xl),
SizedBox(
width: double.infinity,
child: ElevatedButton(
onPressed: _startNewChat,
style: ElevatedButton.styleFrom(
backgroundColor: context.conduitTheme.buttonPrimary,
foregroundColor: context.conduitTheme.buttonPrimaryText,
padding: const EdgeInsets.symmetric(
horizontal: Spacing.buttonPadding,
vertical: Spacing.md,
),
shape: RoundedRectangleBorder(
borderRadius: BorderRadius.circular(AppBorderRadius.button),
),
elevation: Elevation.none,
),
child: Text(
'Start New Chat',
style: AppTypography.labelStyle.copyWith(
color: context.conduitTheme.buttonPrimaryText,
fontWeight: FontWeight.w600,
),
),
),
),
],
),
).animate().fadeIn(
duration: AnimationDuration.pageTransition,
curve: AnimationCurves.pageTransition,
);
}
Widget _buildNoResultsState() {
return Center(
child: Column(
mainAxisAlignment: MainAxisAlignment.center,
children: [
Icon(
Platform.isIOS ? CupertinoIcons.search : Icons.search_rounded,
size: IconSize.xxl,
color: context.conduitTheme.iconSecondary,
),
const SizedBox(height: Spacing.lg),
Text(
'No conversations found',
style: AppTypography.headlineSmallStyle.copyWith(
color: context.conduitTheme.textPrimary,
fontWeight: FontWeight.w600,
),
),
const SizedBox(height: Spacing.sm),
Text(
'Try adjusting your search terms',
style: AppTypography.bodyMediumStyle.copyWith(
color: context.conduitTheme.textSecondary,
),
textAlign: TextAlign.center,
),
],
),
).animate().fadeIn(
duration: AnimationDuration.pageTransition,
curve: AnimationCurves.pageTransition,
);
}
Widget _buildLoadingState() {
return ListView.builder(
padding: const EdgeInsets.all(Spacing.pagePadding),
itemCount: 6,
itemBuilder: (context, index) {
return Container(
margin: const EdgeInsets.only(bottom: Spacing.listGap),
padding: const EdgeInsets.all(Spacing.listItemPadding),
decoration: BoxDecoration(
color: context.conduitTheme.cardBackground,
borderRadius: BorderRadius.circular(AppBorderRadius.card),
border: Border.all(
color: context.conduitTheme.cardBorder,
width: BorderWidth.regular,
),
boxShadow: ConduitShadows.low,
),
child: Row(
children: [
Container(
width: IconSize.avatar,
height: IconSize.avatar,
decoration: BoxDecoration(
color: context.conduitTheme.shimmerBase,
borderRadius: BorderRadius.circular(AppBorderRadius.avatar),
),
).animate().shimmer(duration: AnimationDuration.slow),
const SizedBox(width: Spacing.md),
Expanded(
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Container(
height: AppTypography.bodyLarge,
decoration: BoxDecoration(
color: context.conduitTheme.shimmerBase,
borderRadius: BorderRadius.circular(AppBorderRadius.xs),
),
).animate().shimmer(duration: AnimationDuration.slow),
const SizedBox(height: Spacing.xs),
Container(
height: AppTypography.bodySmall,
width: double.infinity,
decoration: BoxDecoration(
color: context.conduitTheme.shimmerBase,
borderRadius: BorderRadius.circular(AppBorderRadius.xs),
),
).animate().shimmer(duration: AnimationDuration.slow),
],
),
),
],
),
);
},
);
}
Widget _buildErrorState(Object error) {
return Center(
child: Column(
mainAxisAlignment: MainAxisAlignment.center,
children: [
Icon(
Platform.isIOS
? CupertinoIcons.exclamationmark_triangle
: Icons.error_rounded,
size: IconSize.xxl,
color: context.conduitTheme.error,
),
const SizedBox(height: Spacing.lg),
Text(
'Failed to load conversations',
style: AppTypography.headlineSmallStyle.copyWith(
color: context.conduitTheme.textPrimary,
fontWeight: FontWeight.w600,
),
),
const SizedBox(height: Spacing.sm),
Text(
'Please try again later',
style: AppTypography.bodyMediumStyle.copyWith(
color: context.conduitTheme.textSecondary,
),
textAlign: TextAlign.center,
),
const SizedBox(height: Spacing.xl),
ElevatedButton(
onPressed: () => ref.invalidate(conversationsProvider),
style: ElevatedButton.styleFrom(
backgroundColor: context.conduitTheme.buttonPrimary,
foregroundColor: context.conduitTheme.buttonPrimaryText,
padding: const EdgeInsets.symmetric(
horizontal: Spacing.buttonPadding,
vertical: Spacing.md,
),
shape: RoundedRectangleBorder(
borderRadius: BorderRadius.circular(AppBorderRadius.button),
),
elevation: Elevation.none,
),
child: Text(
'Retry',
style: AppTypography.labelStyle.copyWith(
color: context.conduitTheme.buttonPrimaryText,
fontWeight: FontWeight.w600,
),
),
),
],
),
);
}
Widget _buildBottomActions() {
return const SizedBox.shrink(); // Remove bottom actions since we'll use FAB
}
// Helper methods
List<dynamic> _filterConversations(List<dynamic> conversations) {
if (_searchQuery.isEmpty) return conversations;
return conversations.where((conversation) {
final title = conversation.title?.toLowerCase() ?? '';
final query = _searchQuery.toLowerCase();
2025-08-17 00:05:30 +05:30
return title.contains(query);
2025-08-10 01:20:45 +05:30
}).toList();
}
2025-08-17 00:05:30 +05:30
2025-08-10 01:20:45 +05:30
String _formatConversationDate(DateTime? date) {
if (date == null) return '';
final now = DateTime.now();
final difference = now.difference(date);
if (difference.inDays == 0) {
// Same day - show time
final hour = date.hour;
final minute = date.minute;
final period = hour >= 12 ? 'PM' : 'AM';
final displayHour = hour > 12 ? hour - 12 : (hour == 0 ? 12 : hour);
return '$displayHour:${minute.toString().padLeft(2, '0')} $period';
} else if (difference.inDays == 1) {
return 'Yesterday';
} else if (difference.inDays < 7) {
// Show day name for this week
final days = ['Mon', 'Tue', 'Wed', 'Thu', 'Fri', 'Sat', 'Sun'];
return days[date.weekday - 1];
} else if (difference.inDays < 365) {
// Show month and day for this year
final months = [
'Jan',
'Feb',
'Mar',
'Apr',
'May',
'Jun',
'Jul',
'Aug',
'Sep',
'Oct',
'Nov',
'Dec',
];
return '${months[date.month - 1]} ${date.day}';
} else {
// Show full date for older conversations
return '${date.month}/${date.day}/${date.year}';
}
}
// TODO: Implement search toggle functionality when needed
// void _toggleSearch() {
// // Focus the search field when search is toggled
// FocusScope.of(context).requestFocus(FocusNode());
// _searchController.clear();
// setState(() {
// _searchQuery = '';
// });
// ref.read(searchQueryProvider.notifier).state = '';
// }
void _showOptions() {
showModalBottomSheet(
context: context,
backgroundColor: Colors.transparent,
builder: (context) => Container(
decoration: BoxDecoration(
color: context.conduitTheme.surfaceBackground,
borderRadius: const BorderRadius.vertical(
top: Radius.circular(AppBorderRadius.bottomSheet),
),
border: Border.all(
color: context.conduitTheme.dividerColor,
width: BorderWidth.regular,
),
boxShadow: ConduitShadows.modal,
),
child: SafeArea(
top: false,
bottom: true,
child: Padding(
padding: const EdgeInsets.all(Spacing.bottomSheetPadding),
child: Column(
mainAxisSize: MainAxisSize.min,
children: [
// Handle bar
Container(
width: 40,
height: 4,
margin: const EdgeInsets.only(bottom: Spacing.md),
decoration: BoxDecoration(
color: context.conduitTheme.dividerColor,
borderRadius: BorderRadius.circular(AppBorderRadius.xs),
),
),
// Options
ListTile(
leading: Icon(
Platform.isIOS
? CupertinoIcons.archivebox
: Icons.archive_rounded,
color: context.conduitTheme.iconPrimary,
),
title: Text(
'Archived Chats',
style: AppTypography.bodyMediumStyle.copyWith(
color: context.conduitTheme.textPrimary,
),
),
onTap: () {
Navigator.pop(context);
_showArchivedSection();
},
),
],
),
),
),
),
);
}
void _selectConversation(dynamic conversation) async {
if (_isLoadingConversation) return; // Prevent multiple loads
setState(() {
_isLoadingConversation = true;
});
try {
// Mark global conversation loading state to show skeletons in chat
ref.read(isLoadingConversationProvider.notifier).state = true;
// Load the full conversation with messages
final api = ref.read(apiServiceProvider);
if (api != null) {
debugPrint('DEBUG: Loading full conversation: ${conversation.id}');
final fullConversation = await api.getConversation(conversation.id);
debugPrint(
'DEBUG: Loaded conversation with ${fullConversation.messages.length} messages',
);
// Set the full conversation as active
ref.read(activeConversationProvider.notifier).state = fullConversation;
// Clear global loading before navigating so chat doesn't stick on skeletons
ref.read(isLoadingConversationProvider.notifier).state = false;
} else {
// Fallback to the conversation from the list
ref.read(activeConversationProvider.notifier).state = conversation;
// Clear global loading before navigating
ref.read(isLoadingConversationProvider.notifier).state = false;
}
// Do not navigate synchronously after async awaits; schedule for next frame
WidgetsBinding.instance.addPostFrameCallback((_) {
if (!mounted) return;
if (widget.isOverlay) {
Navigator.of(context).pop();
} else {
Navigator.of(context).pop();
}
});
} catch (e) {
debugPrint('DEBUG: Error loading conversation: $e');
// Fallback to the conversation from the list
ref.read(activeConversationProvider.notifier).state = conversation;
// Ensure global loading is cleared even on error
ref.read(isLoadingConversationProvider.notifier).state = false;
WidgetsBinding.instance.addPostFrameCallback((_) {
if (!mounted) return;
if (widget.isOverlay) {
Navigator.of(context).pop();
} else {
Navigator.of(context).pop();
}
});
} finally {
if (mounted) {
setState(() {
_isLoadingConversation = false;
});
}
}
}
void _showConversationOptions(dynamic conversation) {
showModalBottomSheet(
context: context,
backgroundColor: Colors.transparent,
builder: (context) => Container(
decoration: BoxDecoration(
color: context.conduitTheme.surfaceBackground,
borderRadius: const BorderRadius.vertical(
top: Radius.circular(AppBorderRadius.bottomSheet),
),
border: Border.all(
color: context.conduitTheme.dividerColor,
width: BorderWidth.regular,
),
boxShadow: ConduitShadows.modal,
),
child: SafeArea(
top: false,
bottom: true,
child: Padding(
padding: const EdgeInsets.all(Spacing.bottomSheetPadding),
child: Column(
mainAxisSize: MainAxisSize.min,
children: [
// Handle bar
Container(
width: 40,
height: 4,
margin: const EdgeInsets.only(bottom: Spacing.md),
decoration: BoxDecoration(
color: context.conduitTheme.dividerColor,
borderRadius: BorderRadius.circular(AppBorderRadius.xs),
),
),
// Conversation title
Padding(
padding: const EdgeInsets.only(bottom: Spacing.sm),
child: Text(
conversation.title ?? 'New Chat',
style: AppTypography.headlineSmallStyle.copyWith(
color: context.conduitTheme.textPrimary,
fontWeight: FontWeight.w600,
),
textAlign: TextAlign.center,
),
),
// Options
ListTile(
leading: Icon(
Platform.isIOS ? CupertinoIcons.pin : Icons.push_pin,
color: conversation.pinned == true
? context.conduitTheme.warning
: context.conduitTheme.iconPrimary,
),
title: Text(
conversation.pinned == true ? 'Unpin Chat' : 'Pin Chat',
style: AppTypography.bodyMediumStyle.copyWith(
color: context.conduitTheme.textPrimary,
),
),
onTap: () {
Navigator.pop(context);
_togglePinConversation(conversation);
},
),
2025-08-17 00:05:30 +05:30
2025-08-10 01:20:45 +05:30
ListTile(
leading: Icon(
Platform.isIOS
? CupertinoIcons.archivebox
: Icons.archive_rounded,
color: context.conduitTheme.iconPrimary,
),
title: Text(
'Archive Chat',
style: AppTypography.bodyMediumStyle.copyWith(
color: context.conduitTheme.textPrimary,
),
),
onTap: () {
Navigator.pop(context);
_archiveConversation(conversation);
},
),
ListTile(
leading: Icon(
Platform.isIOS
? CupertinoIcons.delete
: Icons.delete_rounded,
color: context.conduitTheme.error,
),
title: Text(
'Delete Chat',
style: AppTypography.bodyMediumStyle.copyWith(
color: context.conduitTheme.error,
),
),
onTap: () {
Navigator.pop(context);
_deleteConversation(conversation);
},
),
],
),
),
),
),
);
}
void _startNewChat() {
startNewChat(ref);
if (widget.isOverlay) {
Navigator.of(context).pop(); // Close the overlay
} else {
Navigator.of(context).pop(); // Go back to main navigation
}
}
2025-08-17 00:05:30 +05:30
void _showCreateMenu() {
showModalBottomSheet(
context: context,
backgroundColor: Colors.transparent,
builder: (context) => Container(
decoration: BoxDecoration(
color: context.conduitTheme.surfaceBackground,
borderRadius: const BorderRadius.vertical(
top: Radius.circular(AppBorderRadius.bottomSheet),
),
border: Border.all(
color: context.conduitTheme.dividerColor,
width: BorderWidth.regular,
),
boxShadow: ConduitShadows.modal,
),
child: SafeArea(
top: false,
bottom: true,
child: Padding(
padding: const EdgeInsets.all(Spacing.bottomSheetPadding),
child: Column(
mainAxisSize: MainAxisSize.min,
children: [
// Handle bar
Container(
width: 40,
height: 4,
margin: const EdgeInsets.only(bottom: Spacing.lg),
decoration: BoxDecoration(
color: context.conduitTheme.dividerColor,
borderRadius: BorderRadius.circular(AppBorderRadius.xs),
),
),
// Header
Padding(
padding: const EdgeInsets.only(bottom: Spacing.md),
child: Text(
'Create New',
style: AppTypography.headlineSmallStyle.copyWith(
color: context.conduitTheme.textPrimary,
fontWeight: FontWeight.w600,
),
),
),
// Options
ListTile(
leading: Container(
width: 40,
height: 40,
decoration: BoxDecoration(
color: context.conduitTheme.buttonPrimary.withValues(alpha: 0.1),
borderRadius: BorderRadius.circular(AppBorderRadius.lg),
),
child: Icon(
Platform.isIOS ? CupertinoIcons.chat_bubble : Icons.chat_rounded,
color: context.conduitTheme.buttonPrimary,
size: IconSize.medium,
),
),
title: Text(
'New Chat',
style: AppTypography.bodyLargeStyle.copyWith(
color: context.conduitTheme.textPrimary,
fontWeight: FontWeight.w600,
),
),
subtitle: Text(
'Start a new conversation',
style: AppTypography.bodyMediumStyle.copyWith(
color: context.conduitTheme.textSecondary,
),
),
onTap: () {
Navigator.pop(context);
_startNewChat();
},
),
const SizedBox(height: Spacing.sm),
ListTile(
leading: Container(
width: 40,
height: 40,
decoration: BoxDecoration(
color: context.conduitTheme.info.withValues(alpha: 0.1),
borderRadius: BorderRadius.circular(AppBorderRadius.lg),
),
child: Icon(
Platform.isIOS ? CupertinoIcons.folder_badge_plus : Icons.create_new_folder_rounded,
color: context.conduitTheme.info,
size: IconSize.medium,
),
),
title: Text(
'New Folder',
style: AppTypography.bodyLargeStyle.copyWith(
color: context.conduitTheme.textPrimary,
fontWeight: FontWeight.w600,
),
),
subtitle: Text(
'Create a folder to organize chats',
style: AppTypography.bodyMediumStyle.copyWith(
color: context.conduitTheme.textSecondary,
),
),
onTap: () {
Navigator.pop(context);
_showCreateFolderDialog();
},
),
],
),
),
),
),
);
}
void _showCreateFolderDialog() {
final nameController = TextEditingController();
2025-08-10 01:20:45 +05:30
showDialog(
context: context,
2025-08-17 00:05:30 +05:30
builder: (dialogContext) => _CreateFolderDialog(
nameController: nameController,
onCreateFolder: (name) => _createFolderFromDialog(name, dialogContext),
),
2025-08-10 01:20:45 +05:30
);
}
2025-08-17 00:05:30 +05:30
Future<void> _createFolderFromDialog(String name, BuildContext dialogContext) async {
try {
final api = ref.read(apiServiceProvider);
if (api == null) throw Exception('No API service available');
await api.createFolder(name: name);
ref.invalidate(foldersProvider);
if (mounted) {
Navigator.pop(dialogContext);
ScaffoldMessenger.of(context).showSnackBar(
SnackBar(
content: Text(
'Folder "$name" created',
style: AppTypography.bodyMediumStyle.copyWith(
color: context.conduitTheme.textInverse,
),
),
backgroundColor: context.conduitTheme.success,
),
);
}
} catch (e) {
if (mounted) {
ScaffoldMessenger.of(context).showSnackBar(
SnackBar(
content: Text(
'Failed to create folder: $e',
style: AppTypography.bodyMediumStyle.copyWith(
color: context.conduitTheme.textInverse,
),
),
backgroundColor: context.conduitTheme.error,
),
);
}
}
}
2025-08-10 01:20:45 +05:30
void _togglePinConversation(dynamic conversation) async {
try {
final api = ref.read(apiServiceProvider);
if (api != null) {
final newPinnedState = !(conversation.pinned ?? false);
await api.pinConversation(conversation.id, newPinnedState);
// Refresh conversations list
ref.invalidate(conversationsProvider);
if (mounted) {
ScaffoldMessenger.of(context).showSnackBar(
SnackBar(
content: Text(
newPinnedState ? 'Chat pinned' : 'Chat unpinned',
style: AppTypography.bodyMediumStyle.copyWith(
color: context.conduitTheme.textInverse,
),
),
backgroundColor: context.conduitTheme.success,
),
);
}
}
} catch (e) {
debugPrint('DEBUG: Error toggling pin: $e');
if (mounted) {
ScaffoldMessenger.of(context).showSnackBar(
SnackBar(
content: Text(
'Failed to ${conversation.pinned == true ? 'unpin' : 'pin'} chat',
style: AppTypography.bodyMediumStyle.copyWith(
color: context.conduitTheme.textInverse,
),
),
backgroundColor: context.conduitTheme.error,
),
);
}
}
}
2025-08-17 00:05:30 +05:30
2025-08-10 01:20:45 +05:30
void _archiveConversation(dynamic conversation) async {
try {
final api = ref.read(apiServiceProvider);
if (api != null) {
await api.archiveConversation(conversation.id, true);
// Refresh conversations list
ref.invalidate(conversationsProvider);
if (mounted) {
ScaffoldMessenger.of(context).showSnackBar(
SnackBar(
content: Text(
'Chat archived',
style: AppTypography.bodyMediumStyle.copyWith(
color: context.conduitTheme.textInverse,
),
),
backgroundColor: context.conduitTheme.success,
),
);
}
}
} catch (e) {
debugPrint('DEBUG: Error archiving conversation: $e');
if (mounted) {
ScaffoldMessenger.of(context).showSnackBar(
SnackBar(
content: Text(
'Failed to archive chat',
style: AppTypography.bodyMediumStyle.copyWith(
color: context.conduitTheme.textInverse,
),
),
backgroundColor: context.conduitTheme.error,
),
);
}
}
}
void _deleteConversation(dynamic conversation) async {
// Show confirmation dialog
final confirmed = await ThemedDialogs.confirm(
context,
title: 'Delete Chat',
message:
'Are you sure you want to delete "${conversation.title ?? 'New Chat'}"? This action cannot be undone.',
confirmText: 'Delete',
isDestructive: true,
barrierDismissible: true,
);
if (confirmed == true) {
try {
final api = ref.read(apiServiceProvider);
if (api != null) {
await api.deleteConversation(conversation.id);
// Refresh conversations list
ref.invalidate(conversationsProvider);
if (mounted) {
ScaffoldMessenger.of(context).showSnackBar(
SnackBar(
content: Text(
'Chat deleted',
style: AppTypography.bodyMediumStyle.copyWith(
color: context.conduitTheme.textInverse,
),
),
backgroundColor: context.conduitTheme.success,
),
);
}
}
} catch (e) {
debugPrint('DEBUG: Error deleting conversation: $e');
if (mounted) {
ScaffoldMessenger.of(context).showSnackBar(
SnackBar(
content: Text(
'Failed to delete chat',
style: AppTypography.bodyMediumStyle.copyWith(
color: context.conduitTheme.textInverse,
),
),
backgroundColor: context.conduitTheme.error,
),
);
}
}
}
}
void _showArchivedSection() {
// Set the archived section to be visible
ref.read(_showArchivedProvider.notifier).state = true;
// Scroll to the archived section
WidgetsBinding.instance.addPostFrameCallback((_) {
if (_scrollController.hasClients) {
_scrollController.animateTo(
_scrollController.position.maxScrollExtent,
duration: const Duration(milliseconds: 500),
curve: Curves.easeInOut,
);
}
});
}
Widget _buildSectionHeader(String title, int count) {
return Padding(
padding: const EdgeInsets.symmetric(
2025-08-17 00:05:30 +05:30
horizontal: Spacing.md,
vertical: Spacing.sm,
2025-08-10 01:20:45 +05:30
),
child: Row(
children: [
Text(
title,
style: AppTypography.labelStyle.copyWith(
color: context.conduitTheme.textSecondary,
fontWeight: FontWeight.w600,
letterSpacing: 0.5,
),
),
2025-08-17 00:05:30 +05:30
const SizedBox(width: Spacing.sm),
2025-08-10 01:20:45 +05:30
Container(
padding: const EdgeInsets.symmetric(
horizontal: Spacing.xs,
vertical: Spacing.xxs,
),
decoration: BoxDecoration(
color: context.conduitTheme.surfaceContainer,
borderRadius: BorderRadius.circular(AppBorderRadius.badge),
),
child: Text(
count.toString(),
style: AppTypography.captionStyle.copyWith(
color: context.conduitTheme.textTertiary,
fontWeight: FontWeight.w600,
),
),
),
],
),
);
}
2025-08-17 00:05:30 +05:30
Widget _buildFolderHeader(String folderId, String folderName, int count) {
final expandedFolders = ref.watch(_expandedFoldersProvider);
final isExpanded = expandedFolders[folderId] ?? false;
2025-08-17 16:11:19 +05:30
return InkWell(
2025-08-17 00:05:30 +05:30
onTap: () {
final currentState = ref.read(_expandedFoldersProvider);
2025-08-17 16:11:19 +05:30
final newState = Map<String, bool>.from(currentState);
newState[folderId] = !isExpanded;
ref.read(_expandedFoldersProvider.notifier).state = newState;
2025-08-17 00:05:30 +05:30
},
2025-08-17 16:11:19 +05:30
borderRadius: BorderRadius.circular(AppBorderRadius.md),
2025-08-17 00:05:30 +05:30
child: Padding(
padding: const EdgeInsets.symmetric(
horizontal: Spacing.md,
vertical: Spacing.sm,
),
child: Row(
children: [
Icon(
isExpanded
? (Platform.isIOS ? CupertinoIcons.chevron_down : Icons.expand_more_rounded)
: (Platform.isIOS ? CupertinoIcons.chevron_right : Icons.chevron_right_rounded),
size: IconSize.small,
color: context.conduitTheme.textSecondary,
),
const SizedBox(width: Spacing.sm),
Icon(
Platform.isIOS ? CupertinoIcons.folder_fill : Icons.folder_rounded,
size: IconSize.small,
color: context.conduitTheme.textSecondary,
),
const SizedBox(width: Spacing.sm),
Text(
folderName,
style: AppTypography.labelStyle.copyWith(
color: context.conduitTheme.textSecondary,
fontWeight: FontWeight.w600,
letterSpacing: 0.5,
),
),
const SizedBox(width: Spacing.sm),
Container(
padding: const EdgeInsets.symmetric(
horizontal: Spacing.xs,
vertical: Spacing.xxs,
),
decoration: BoxDecoration(
color: context.conduitTheme.surfaceContainer,
borderRadius: BorderRadius.circular(AppBorderRadius.badge),
),
child: Text(
count.toString(),
style: AppTypography.captionStyle.copyWith(
color: context.conduitTheme.textTertiary,
fontWeight: FontWeight.w600,
),
),
),
],
),
),
);
}
2025-08-10 01:20:45 +05:30
Widget _buildArchivedSection(List<dynamic> archivedConversations) {
return Consumer(
builder: (context, ref, child) {
final showArchived = ref.watch(_showArchivedProvider);
return Column(
children: [
// Collapsible header
InkWell(
onTap: () {
ref.read(_showArchivedProvider.notifier).state = !showArchived;
},
borderRadius: BorderRadius.circular(AppBorderRadius.card),
child: Padding(
2025-08-17 00:05:30 +05:30
padding: const EdgeInsets.symmetric(
horizontal: Spacing.md,
vertical: Spacing.sm,
),
2025-08-10 01:20:45 +05:30
child: Row(
children: [
Icon(
Platform.isIOS
? CupertinoIcons.archivebox
: Icons.archive_rounded,
size: IconSize.small,
color: context.conduitTheme.textSecondary,
),
const SizedBox(width: Spacing.sm),
Text(
'Archived',
style: AppTypography.labelStyle.copyWith(
color: context.conduitTheme.textSecondary,
fontWeight: FontWeight.w600,
),
),
const SizedBox(width: Spacing.xs),
Container(
padding: const EdgeInsets.symmetric(
horizontal: Spacing.xs,
vertical: Spacing.xxs,
),
decoration: BoxDecoration(
color: context.conduitTheme.surfaceContainer,
borderRadius: BorderRadius.circular(
AppBorderRadius.badge,
),
),
child: Text(
archivedConversations.length.toString(),
style: AppTypography.captionStyle.copyWith(
color: context.conduitTheme.textTertiary,
fontWeight: FontWeight.w600,
),
),
),
const Spacer(),
Icon(
showArchived
? (Platform.isIOS
? CupertinoIcons.chevron_up
: Icons.keyboard_arrow_up)
: (Platform.isIOS
? CupertinoIcons.chevron_down
: Icons.keyboard_arrow_down),
size: IconSize.small,
color: context.conduitTheme.textSecondary,
),
],
),
),
),
// Archived conversations (collapsible)
if (showArchived) ...[
const SizedBox(height: Spacing.sm),
...archivedConversations.asMap().entries.map((entry) {
return _buildConversationTile(
entry.value,
entry.key,
isArchived: true,
);
}),
],
],
);
},
);
}
}
2025-08-17 00:05:30 +05:30
class _CreateFolderDialog extends StatefulWidget {
final TextEditingController nameController;
final Future<void> Function(String name) onCreateFolder;
const _CreateFolderDialog({
required this.nameController,
required this.onCreateFolder,
});
@override
State<_CreateFolderDialog> createState() => _CreateFolderDialogState();
}
class _CreateFolderDialogState extends State<_CreateFolderDialog> {
bool isCreating = false;
@override
Widget build(BuildContext context) {
return Directionality(
textDirection: TextDirection.ltr,
child: Dialog(
backgroundColor: Colors.transparent,
child: Container(
width: 400,
decoration: BoxDecoration(
color: context.conduitTheme.surfaceBackground,
borderRadius: BorderRadius.circular(AppBorderRadius.modal),
border: Border.all(
color: context.conduitTheme.cardBorder.withValues(alpha: 0.2),
width: BorderWidth.regular,
),
boxShadow: ConduitShadows.modal,
),
child: Column(
mainAxisSize: MainAxisSize.min,
crossAxisAlignment: CrossAxisAlignment.stretch,
children: [
// Header
Container(
padding: const EdgeInsets.all(Spacing.xl),
decoration: BoxDecoration(
color: context.conduitTheme.cardBackground,
borderRadius: const BorderRadius.vertical(
top: Radius.circular(AppBorderRadius.modal),
),
border: Border(
bottom: BorderSide(
color: context.conduitTheme.dividerColor.withValues(alpha: 0.1),
width: BorderWidth.regular,
),
),
),
child: Row(
children: [
Container(
width: 40,
height: 40,
decoration: BoxDecoration(
color: context.conduitTheme.info.withValues(alpha: 0.1),
borderRadius: BorderRadius.circular(AppBorderRadius.lg),
),
child: Icon(
Platform.isIOS ? CupertinoIcons.folder_badge_plus : Icons.create_new_folder_rounded,
color: context.conduitTheme.info,
size: IconSize.medium,
),
),
const SizedBox(width: Spacing.md),
Expanded(
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Text(
'Create New Folder',
style: AppTypography.headlineSmallStyle.copyWith(
color: context.conduitTheme.textPrimary,
fontWeight: FontWeight.w600,
),
),
const SizedBox(height: Spacing.xs),
Text(
'Enter a name for your folder',
style: AppTypography.bodyMediumStyle.copyWith(
color: context.conduitTheme.textSecondary,
),
),
],
),
),
ConduitIconButton(
icon: Platform.isIOS ? CupertinoIcons.xmark : Icons.close_rounded,
onPressed: isCreating ? null : () => Navigator.pop(context),
),
],
),
),
// Content
Padding(
padding: const EdgeInsets.all(Spacing.xl),
child: TextField(
controller: widget.nameController,
autofocus: true,
enabled: !isCreating,
decoration: InputDecoration(
labelText: 'Folder Name',
hintText: 'Enter folder name',
prefixIcon: Icon(
Platform.isIOS ? CupertinoIcons.folder : Icons.folder_outlined,
color: context.conduitTheme.iconSecondary,
),
border: OutlineInputBorder(
borderRadius: BorderRadius.circular(AppBorderRadius.input),
borderSide: BorderSide(
color: context.conduitTheme.inputBorder,
width: BorderWidth.regular,
),
),
focusedBorder: OutlineInputBorder(
borderRadius: BorderRadius.circular(AppBorderRadius.input),
borderSide: BorderSide(
color: context.conduitTheme.buttonPrimary,
width: BorderWidth.regular,
),
),
labelStyle: AppTypography.bodyMediumStyle.copyWith(
color: context.conduitTheme.textSecondary,
),
hintStyle: AppTypography.bodyMediumStyle.copyWith(
color: context.conduitTheme.inputPlaceholder,
),
),
style: AppTypography.bodyMediumStyle.copyWith(
color: context.conduitTheme.textPrimary,
),
onSubmitted: (value) {
if (value.trim().isNotEmpty && !isCreating) {
_createFolder();
}
},
),
),
// Actions
Container(
padding: const EdgeInsets.all(Spacing.xl),
decoration: BoxDecoration(
color: context.conduitTheme.cardBackground,
borderRadius: const BorderRadius.vertical(
bottom: Radius.circular(AppBorderRadius.modal),
),
border: Border(
top: BorderSide(
color: context.conduitTheme.dividerColor.withValues(alpha: 0.1),
width: BorderWidth.regular,
),
),
),
child: Row(
children: [
Expanded(
child: TextButton(
onPressed: isCreating ? null : () => Navigator.pop(context),
style: TextButton.styleFrom(
padding: const EdgeInsets.symmetric(vertical: Spacing.md),
shape: RoundedRectangleBorder(
borderRadius: BorderRadius.circular(AppBorderRadius.button),
),
),
child: Text(
'Cancel',
style: AppTypography.labelStyle.copyWith(
color: context.conduitTheme.textSecondary,
fontWeight: FontWeight.w600,
),
),
),
),
const SizedBox(width: Spacing.md),
Expanded(
child: ElevatedButton(
onPressed: isCreating ? null : () {
final name = widget.nameController.text.trim();
if (name.isNotEmpty) {
_createFolder();
}
},
style: ElevatedButton.styleFrom(
backgroundColor: context.conduitTheme.buttonPrimary,
foregroundColor: context.conduitTheme.buttonPrimaryText,
padding: const EdgeInsets.symmetric(vertical: Spacing.md),
shape: RoundedRectangleBorder(
borderRadius: BorderRadius.circular(AppBorderRadius.button),
),
elevation: Elevation.none,
),
child: isCreating
? SizedBox(
width: IconSize.medium,
height: IconSize.medium,
child: CircularProgressIndicator(
strokeWidth: 2,
valueColor: AlwaysStoppedAnimation<Color>(
context.conduitTheme.buttonPrimaryText,
),
),
)
: Text(
'Create',
style: AppTypography.labelStyle.copyWith(
color: context.conduitTheme.buttonPrimaryText,
fontWeight: FontWeight.w600,
),
),
),
),
],
),
),
],
),
).animate().slideY(
begin: 0.1,
duration: AnimationDuration.modalPresentation,
curve: AnimationCurves.modalPresentation,
).fadeIn(
duration: AnimationDuration.modalPresentation,
curve: AnimationCurves.easeOut,
),
),
);
}
Future<void> _createFolder() async {
final name = widget.nameController.text.trim();
if (name.isEmpty) return;
setState(() => isCreating = true);
try {
await widget.onCreateFolder(name);
} finally {
if (mounted) {
setState(() => isCreating = false);
}
}
}
}