fix(navigation): Adjust bottom padding for safe area in chats drawer

This commit is contained in:
cogwheel0
2025-12-15 20:17:56 +05:30
parent 5396fb8eec
commit 7619040e27
7 changed files with 495 additions and 771 deletions

View File

@@ -956,57 +956,9 @@ class _ChatPageState extends ConsumerState<ChatPage> {
required Widget child, required Widget child,
bool isCircular = false, bool isCircular = false,
}) { }) {
final theme = Theme.of(context); return FloatingAppBarPill(
final isDark = theme.brightness == Brightness.dark; isCircular: isCircular,
child: child,
// Use same high-contrast colors as the floating chat input
final backgroundColor = isDark
? Color.lerp(context.conduitTheme.cardBackground, Colors.white, 0.08)!
: Color.lerp(context.conduitTheme.inputBackground, Colors.black, 0.06)!;
final borderColor = context.conduitTheme.cardBorder.withValues(
alpha: isDark ? 0.65 : 0.55,
);
final borderRadius = isCircular
? BorderRadius.circular(100)
: BorderRadius.circular(AppBorderRadius.pill);
// For circular buttons, ensure the entire widget is constrained to a square
if (isCircular) {
return SizedBox(
width: 44,
height: 44,
child: ClipRRect(
borderRadius: borderRadius,
child: BackdropFilter(
filter: ImageFilter.blur(sigmaX: 16, sigmaY: 16),
child: Container(
decoration: BoxDecoration(
color: backgroundColor.withValues(alpha: 0.85),
borderRadius: borderRadius,
border: Border.all(color: borderColor, width: BorderWidth.thin),
),
child: Center(child: child),
),
),
),
);
}
return ClipRRect(
borderRadius: borderRadius,
child: BackdropFilter(
filter: ImageFilter.blur(sigmaX: 16, sigmaY: 16),
child: Container(
decoration: BoxDecoration(
color: backgroundColor.withValues(alpha: 0.85),
borderRadius: borderRadius,
border: Border.all(color: borderColor, width: BorderWidth.thin),
),
child: child,
),
),
); );
} }

View File

@@ -1,6 +1,5 @@
import 'dart:async'; import 'dart:async';
import 'dart:io' show Platform; import 'dart:io' show Platform;
import 'dart:ui' show ImageFilter;
import 'package:flutter/cupertino.dart'; import 'package:flutter/cupertino.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
@@ -23,6 +22,7 @@ import '../../../core/utils/user_avatar_utils.dart';
import '../../../shared/utils/conversation_context_menu.dart'; import '../../../shared/utils/conversation_context_menu.dart';
import '../../../shared/widgets/user_avatar.dart'; import '../../../shared/widgets/user_avatar.dart';
import '../../../shared/widgets/model_avatar.dart'; import '../../../shared/widgets/model_avatar.dart';
import '../../../shared/widgets/conduit_components.dart';
import '../../../shared/widgets/responsive_drawer_layout.dart'; import '../../../shared/widgets/responsive_drawer_layout.dart';
import '../../../core/models/model.dart'; import '../../../core/models/model.dart';
import '../../../core/models/conversation.dart'; import '../../../core/models/conversation.dart';
@@ -268,81 +268,60 @@ class _ChatsDrawerState extends ConsumerState<ChatsDrawer> {
} }
Widget _buildFloatingSearchField(BuildContext context) { Widget _buildFloatingSearchField(BuildContext context) {
final theme = Theme.of(context);
final conduitTheme = context.conduitTheme; final conduitTheme = context.conduitTheme;
final isDark = theme.brightness == Brightness.dark;
final backgroundColor = isDark return FloatingAppBarPill(
? Color.lerp(conduitTheme.cardBackground, Colors.white, 0.08)! child: Material(
: Color.lerp(conduitTheme.inputBackground, Colors.black, 0.06)!; color: Colors.transparent,
child: TextField(
final borderColor = conduitTheme.cardBorder.withValues( controller: _searchController,
alpha: isDark ? 0.65 : 0.55, focusNode: _searchFocusNode,
); onChanged: (_) => _onSearchChanged(),
style: AppTypography.standard.copyWith(
return ClipRRect( color: conduitTheme.textPrimary,
borderRadius: BorderRadius.circular(AppBorderRadius.pill),
child: BackdropFilter(
filter: ImageFilter.blur(sigmaX: 16, sigmaY: 16),
child: Container(
decoration: BoxDecoration(
color: backgroundColor.withValues(alpha: 0.85),
borderRadius: BorderRadius.circular(AppBorderRadius.pill),
border: Border.all(color: borderColor, width: BorderWidth.thin),
), ),
child: Material( decoration: InputDecoration(
color: Colors.transparent, isDense: true,
child: TextField( hintText: AppLocalizations.of(context)!.searchConversations,
controller: _searchController, hintStyle: AppTypography.standard.copyWith(
focusNode: _searchFocusNode, color: conduitTheme.textSecondary.withValues(alpha: 0.6),
onChanged: (_) => _onSearchChanged(), ),
style: AppTypography.standard.copyWith( prefixIcon: Icon(
color: conduitTheme.textPrimary, Platform.isIOS ? CupertinoIcons.search : Icons.search,
), color: conduitTheme.iconSecondary,
decoration: InputDecoration( size: IconSize.input,
isDense: true, ),
hintText: AppLocalizations.of(context)!.searchConversations, prefixIconConstraints: const BoxConstraints(
hintStyle: AppTypography.standard.copyWith( minWidth: TouchTarget.minimum,
color: conduitTheme.textSecondary.withValues(alpha: 0.6), minHeight: TouchTarget.minimum,
), ),
prefixIcon: Icon( suffixIcon: _query.isNotEmpty
Platform.isIOS ? CupertinoIcons.search : Icons.search, ? IconButton(
color: conduitTheme.iconSecondary, onPressed: () {
size: IconSize.input, _searchController.clear();
), setState(() => _query = '');
prefixIconConstraints: const BoxConstraints( _searchFocusNode.unfocus();
minWidth: TouchTarget.minimum, },
minHeight: TouchTarget.minimum, icon: Icon(
), Platform.isIOS
suffixIcon: _query.isNotEmpty ? CupertinoIcons.clear_circled_solid
? IconButton( : Icons.clear,
onPressed: () { color: conduitTheme.iconSecondary,
_searchController.clear(); size: IconSize.input,
setState(() => _query = ''); ),
_searchFocusNode.unfocus(); )
}, : null,
icon: Icon( suffixIconConstraints: const BoxConstraints(
Platform.isIOS minWidth: TouchTarget.minimum,
? CupertinoIcons.clear_circled_solid minHeight: TouchTarget.minimum,
: Icons.clear, ),
color: conduitTheme.iconSecondary, filled: false,
size: IconSize.input, border: InputBorder.none,
), enabledBorder: InputBorder.none,
) focusedBorder: InputBorder.none,
: null, contentPadding: const EdgeInsets.symmetric(
suffixIconConstraints: const BoxConstraints( horizontal: Spacing.md,
minWidth: TouchTarget.minimum, vertical: Spacing.sm,
minHeight: TouchTarget.minimum,
),
filled: false,
border: InputBorder.none,
enabledBorder: InputBorder.none,
focusedBorder: InputBorder.none,
contentPadding: const EdgeInsets.symmetric(
horizontal: Spacing.md,
vertical: Spacing.sm,
),
),
), ),
), ),
), ),
@@ -1700,9 +1679,7 @@ class _ChatsDrawerState extends ConsumerState<ChatsDrawer> {
} }
Widget _buildFloatingBottomSection(BuildContext context) { Widget _buildFloatingBottomSection(BuildContext context) {
final theme = Theme.of(context);
final conduitTheme = context.conduitTheme; final conduitTheme = context.conduitTheme;
final isDark = theme.brightness == Brightness.dark;
final authUser = ref.watch(currentUserProvider2); final authUser = ref.watch(currentUserProvider2);
final asyncUser = ref.watch(currentUserProvider); final asyncUser = ref.watch(currentUserProvider);
final user = asyncUser.maybeWhen( final user = asyncUser.maybeWhen(
@@ -1722,98 +1699,81 @@ class _ChatsDrawerState extends ConsumerState<ChatsDrawer> {
final initial = initialFor(displayName); final initial = initialFor(displayName);
final avatarUrl = resolveUserAvatarUrlForUser(api, user); final avatarUrl = resolveUserAvatarUrlForUser(api, user);
final backgroundColor = isDark
? Color.lerp(conduitTheme.cardBackground, Colors.white, 0.08)!
: Color.lerp(conduitTheme.inputBackground, Colors.black, 0.06)!;
final borderColor = conduitTheme.cardBorder.withValues(
alpha: isDark ? 0.65 : 0.55,
);
if (user == null) return const SizedBox.shrink(); if (user == null) return const SizedBox.shrink();
return ClipRRect( return FloatingAppBarPill(
borderRadius: BorderRadius.circular(AppBorderRadius.pill), child: Padding(
child: BackdropFilter( padding: const EdgeInsets.symmetric(
filter: ImageFilter.blur(sigmaX: 16, sigmaY: 16), horizontal: Spacing.sm,
child: Container( vertical: Spacing.xs,
padding: const EdgeInsets.symmetric( ),
horizontal: Spacing.sm, child: Row(
vertical: Spacing.xs, children: [
), Container(
decoration: BoxDecoration( width: 36,
color: backgroundColor.withValues(alpha: 0.85), height: 36,
borderRadius: BorderRadius.circular(AppBorderRadius.pill), decoration: BoxDecoration(
border: Border.all(color: borderColor, width: BorderWidth.thin), borderRadius: BorderRadius.circular(
), AppBorderRadius.avatar,
child: Row(
children: [
Container(
width: 36,
height: 36,
decoration: BoxDecoration(
borderRadius: BorderRadius.circular(
AppBorderRadius.avatar,
),
border: Border.all(
color: conduitTheme.buttonPrimary.withValues(alpha: 0.25),
width: BorderWidth.thin,
),
), ),
clipBehavior: Clip.hardEdge, border: Border.all(
child: UserAvatar( color: conduitTheme.buttonPrimary.withValues(alpha: 0.25),
size: 36, width: BorderWidth.thin,
imageUrl: avatarUrl,
fallbackText: initial,
), ),
), ),
const SizedBox(width: Spacing.sm), clipBehavior: Clip.hardEdge,
Expanded( child: UserAvatar(
child: Text( size: 36,
displayName, imageUrl: avatarUrl,
maxLines: 1, fallbackText: initial,
overflow: TextOverflow.ellipsis, ),
style: AppTypography.bodySmallStyle.copyWith( ),
color: conduitTheme.textPrimary, const SizedBox(width: Spacing.sm),
fontWeight: FontWeight.w600, Expanded(
decoration: TextDecoration.none, child: Text(
), displayName,
maxLines: 1,
overflow: TextOverflow.ellipsis,
style: AppTypography.bodySmallStyle.copyWith(
color: conduitTheme.textPrimary,
fontWeight: FontWeight.w600,
decoration: TextDecoration.none,
), ),
), ),
// Notes icon (hidden when feature is disabled) ),
if (notesEnabled) // Notes icon (hidden when feature is disabled)
IconButton( if (notesEnabled)
tooltip: AppLocalizations.of(context)!.notes,
onPressed: () {
Navigator.of(context).maybePop();
context.pushNamed(RouteNames.notes);
},
visualDensity: VisualDensity.compact,
icon: Icon(
Platform.isIOS
? CupertinoIcons.doc_text
: Icons.note_alt_outlined,
color: conduitTheme.iconPrimary,
size: IconSize.medium,
),
),
IconButton( IconButton(
tooltip: AppLocalizations.of(context)!.manage, tooltip: AppLocalizations.of(context)!.notes,
onPressed: () { onPressed: () {
Navigator.of(context).maybePop(); Navigator.of(context).maybePop();
context.pushNamed(RouteNames.profile); context.pushNamed(RouteNames.notes);
}, },
visualDensity: VisualDensity.compact, visualDensity: VisualDensity.compact,
icon: Icon( icon: Icon(
Platform.isIOS Platform.isIOS
? CupertinoIcons.settings ? CupertinoIcons.doc_text
: Icons.settings_rounded, : Icons.note_alt_outlined,
color: conduitTheme.iconPrimary, color: conduitTheme.iconPrimary,
size: IconSize.medium, size: IconSize.medium,
), ),
), ),
], IconButton(
), tooltip: AppLocalizations.of(context)!.manage,
onPressed: () {
Navigator.of(context).maybePop();
context.pushNamed(RouteNames.profile);
},
visualDensity: VisualDensity.compact,
icon: Icon(
Platform.isIOS
? CupertinoIcons.settings
: Icons.settings_rounded,
color: conduitTheme.iconPrimary,
size: IconSize.medium,
),
),
],
), ),
), ),
); );

View File

@@ -15,6 +15,7 @@ import '../../../core/providers/app_providers.dart';
import '../../../core/widgets/error_boundary.dart'; import '../../../core/widgets/error_boundary.dart';
import '../../../shared/theme/theme_extensions.dart'; import '../../../shared/theme/theme_extensions.dart';
import '../../../shared/utils/ui_utils.dart'; import '../../../shared/utils/ui_utils.dart';
import '../../../shared/widgets/conduit_components.dart';
import '../../../shared/widgets/improved_loading_states.dart'; import '../../../shared/widgets/improved_loading_states.dart';
import '../../../shared/widgets/middle_ellipsis_text.dart'; import '../../../shared/widgets/middle_ellipsis_text.dart';
import '../../../shared/widgets/themed_dialogs.dart'; import '../../../shared/widgets/themed_dialogs.dart';
@@ -815,55 +816,9 @@ class _NoteEditorPageState extends ConsumerState<NoteEditorPage> {
Widget child, { Widget child, {
bool isCircular = false, bool isCircular = false,
}) { }) {
final theme = Theme.of(context); return FloatingAppBarPill(
final isDark = theme.brightness == Brightness.dark; isCircular: isCircular,
child: child,
final backgroundColor = isDark
? Color.lerp(context.conduitTheme.cardBackground, Colors.white, 0.08)!
: Color.lerp(context.conduitTheme.inputBackground, Colors.black, 0.06)!;
final borderColor = context.conduitTheme.cardBorder.withValues(
alpha: isDark ? 0.65 : 0.55,
);
final borderRadius = isCircular
? BorderRadius.circular(100)
: BorderRadius.circular(AppBorderRadius.pill);
if (isCircular) {
return SizedBox(
width: 44,
height: 44,
child: ClipRRect(
borderRadius: borderRadius,
child: BackdropFilter(
filter: ImageFilter.blur(sigmaX: 16, sigmaY: 16),
child: Container(
decoration: BoxDecoration(
color: backgroundColor.withValues(alpha: 0.85),
borderRadius: borderRadius,
border: Border.all(color: borderColor, width: BorderWidth.thin),
),
child: Center(child: child),
),
),
),
);
}
return ClipRRect(
borderRadius: borderRadius,
child: BackdropFilter(
filter: ImageFilter.blur(sigmaX: 16, sigmaY: 16),
child: Container(
decoration: BoxDecoration(
color: backgroundColor.withValues(alpha: 0.85),
borderRadius: borderRadius,
border: Border.all(color: borderColor, width: BorderWidth.thin),
),
child: child,
),
),
); );
} }

View File

@@ -1,6 +1,5 @@
import 'dart:async'; import 'dart:async';
import 'dart:io' show Platform; import 'dart:io' show Platform;
import 'dart:ui' show ImageFilter;
import 'package:flutter/cupertino.dart'; import 'package:flutter/cupertino.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
@@ -15,7 +14,7 @@ import '../../../core/providers/app_providers.dart';
import '../../../core/services/navigation_service.dart'; import '../../../core/services/navigation_service.dart';
import '../../../core/widgets/error_boundary.dart'; import '../../../core/widgets/error_boundary.dart';
import '../../../shared/theme/theme_extensions.dart'; import '../../../shared/theme/theme_extensions.dart';
import '../../../shared/utils/ui_utils.dart'; import '../../../shared/widgets/conduit_components.dart';
import '../../../shared/widgets/improved_loading_states.dart'; import '../../../shared/widgets/improved_loading_states.dart';
import '../../../shared/widgets/loading_states.dart'; import '../../../shared/widgets/loading_states.dart';
import '../../../shared/widgets/themed_dialogs.dart'; import '../../../shared/widgets/themed_dialogs.dart';
@@ -130,123 +129,23 @@ class _NotesListPageState extends ConsumerState<NotesListPage> {
child: Scaffold( child: Scaffold(
backgroundColor: context.conduitTheme.surfaceBackground, backgroundColor: context.conduitTheme.surfaceBackground,
extendBodyBehindAppBar: true, extendBodyBehindAppBar: true,
appBar: PreferredSize( appBar: FloatingAppBar(
preferredSize: const Size.fromHeight(kToolbarHeight + 64), leading: canPop ? const FloatingAppBarBackButton() : null,
child: Container( title: FloatingAppBarTitle(
decoration: BoxDecoration( text: l10n.notes,
gradient: LinearGradient( icon: Platform.isIOS
begin: Alignment.topCenter, ? CupertinoIcons.doc_text_fill
end: Alignment.bottomCenter, : Icons.notes_rounded,
stops: const [0.0, 0.4, 1.0], ),
colors: [ bottomHeight: 64,
Theme.of(context).scaffoldBackgroundColor, bottom: Padding(
Theme.of(context).scaffoldBackgroundColor.withValues( padding: const EdgeInsets.fromLTRB(
alpha: 0.85, Spacing.inputPadding,
), Spacing.xs,
Theme.of(context).scaffoldBackgroundColor.withValues( Spacing.inputPadding,
alpha: 0.0, Spacing.sm,
),
],
),
),
child: SafeArea(
bottom: false,
child: Column(
mainAxisSize: MainAxisSize.min,
children: [
// App bar row with back button and title
SizedBox(
height: kToolbarHeight,
child: Row(
children: [
// Leading (back button)
if (canPop)
Padding(
padding: const EdgeInsets.only(
left: Spacing.inputPadding,
),
child: Center(
child: GestureDetector(
onTap: () => Navigator.of(context).maybePop(),
child: _buildAppBarPill(
context,
Icon(
UiUtils.platformIcon(
ios: CupertinoIcons.back,
android: Icons.arrow_back,
),
color: context.conduitTheme.textPrimary,
size: IconSize.appBar,
),
isCircular: true,
),
),
),
)
else
const SizedBox(width: Spacing.inputPadding),
// Title centered
Expanded(
child: Center(
child: _buildAppBarPill(
context,
Padding(
padding: const EdgeInsets.symmetric(
horizontal: Spacing.md,
vertical: Spacing.xs,
),
child: Row(
mainAxisSize: MainAxisSize.min,
children: [
Icon(
Platform.isIOS
? CupertinoIcons.doc_text_fill
: Icons.notes_rounded,
color: context.conduitTheme.textPrimary
.withValues(alpha: 0.7),
size: IconSize.md,
),
const SizedBox(width: Spacing.sm),
Text(
l10n.notes,
style:
AppTypography.headlineSmallStyle
.copyWith(
color: context
.conduitTheme
.textPrimary,
fontWeight: FontWeight.w600,
),
),
],
),
),
),
),
),
// Trailing spacer to balance
if (canPop)
const SizedBox(
width: 44 + Spacing.inputPadding,
)
else
const SizedBox(width: Spacing.inputPadding),
],
),
),
// Search bar directly below title
Padding(
padding: const EdgeInsets.fromLTRB(
Spacing.inputPadding,
Spacing.xs,
Spacing.inputPadding,
Spacing.sm,
),
child: _buildFloatingSearchField(context),
),
],
),
), ),
child: _buildFloatingSearchField(context),
), ),
), ),
body: _buildBody(context), body: _buildBody(context),
@@ -255,140 +154,62 @@ class _NotesListPageState extends ConsumerState<NotesListPage> {
); );
} }
Widget _buildAppBarPill(
BuildContext context,
Widget child, {
bool isCircular = false,
}) {
final theme = Theme.of(context);
final isDark = theme.brightness == Brightness.dark;
final backgroundColor = isDark
? Color.lerp(context.conduitTheme.cardBackground, Colors.white, 0.08)!
: Color.lerp(context.conduitTheme.inputBackground, Colors.black, 0.06)!;
final borderColor = context.conduitTheme.cardBorder.withValues(
alpha: isDark ? 0.65 : 0.55,
);
final borderRadius = isCircular
? BorderRadius.circular(100)
: BorderRadius.circular(AppBorderRadius.pill);
if (isCircular) {
return SizedBox(
width: 44,
height: 44,
child: ClipRRect(
borderRadius: borderRadius,
child: BackdropFilter(
filter: ImageFilter.blur(sigmaX: 16, sigmaY: 16),
child: Container(
decoration: BoxDecoration(
color: backgroundColor.withValues(alpha: 0.85),
borderRadius: borderRadius,
border: Border.all(color: borderColor, width: BorderWidth.thin),
),
child: Center(child: child),
),
),
),
);
}
return ClipRRect(
borderRadius: borderRadius,
child: BackdropFilter(
filter: ImageFilter.blur(sigmaX: 16, sigmaY: 16),
child: Container(
decoration: BoxDecoration(
color: backgroundColor.withValues(alpha: 0.85),
borderRadius: borderRadius,
border: Border.all(color: borderColor, width: BorderWidth.thin),
),
child: child,
),
),
);
}
Widget _buildFloatingSearchField(BuildContext context) { Widget _buildFloatingSearchField(BuildContext context) {
final theme = Theme.of(context);
final conduitTheme = context.conduitTheme; final conduitTheme = context.conduitTheme;
final isDark = theme.brightness == Brightness.dark;
final l10n = AppLocalizations.of(context)!; final l10n = AppLocalizations.of(context)!;
final backgroundColor = isDark return FloatingAppBarPill(
? Color.lerp(conduitTheme.cardBackground, Colors.white, 0.08)! child: Material(
: Color.lerp(conduitTheme.inputBackground, Colors.black, 0.06)!; color: Colors.transparent,
child: TextField(
final borderColor = conduitTheme.cardBorder.withValues( controller: _searchController,
alpha: isDark ? 0.65 : 0.55, focusNode: _searchFocusNode,
); onChanged: (_) => _onSearchChanged(),
style: AppTypography.standard.copyWith(
return ClipRRect( color: conduitTheme.textPrimary,
borderRadius: BorderRadius.circular(AppBorderRadius.pill),
child: BackdropFilter(
filter: ImageFilter.blur(sigmaX: 16, sigmaY: 16),
child: Container(
decoration: BoxDecoration(
color: backgroundColor.withValues(alpha: 0.85),
borderRadius: BorderRadius.circular(AppBorderRadius.pill),
border: Border.all(color: borderColor, width: BorderWidth.thin),
), ),
child: Material( decoration: InputDecoration(
color: Colors.transparent, isDense: true,
child: TextField( hintText: l10n.searchNotes,
controller: _searchController, hintStyle: AppTypography.standard.copyWith(
focusNode: _searchFocusNode, color: conduitTheme.textSecondary.withValues(alpha: 0.6),
onChanged: (_) => _onSearchChanged(), ),
style: AppTypography.standard.copyWith( prefixIcon: Icon(
color: conduitTheme.textPrimary, Platform.isIOS ? CupertinoIcons.search : Icons.search,
), color: conduitTheme.iconSecondary,
decoration: InputDecoration( size: IconSize.input,
isDense: true, ),
hintText: l10n.searchNotes, prefixIconConstraints: const BoxConstraints(
hintStyle: AppTypography.standard.copyWith( minWidth: TouchTarget.minimum,
color: conduitTheme.textSecondary.withValues(alpha: 0.6), minHeight: TouchTarget.minimum,
), ),
prefixIcon: Icon( suffixIcon: _query.isNotEmpty
Platform.isIOS ? CupertinoIcons.search : Icons.search, ? IconButton(
color: conduitTheme.iconSecondary, onPressed: () {
size: IconSize.input, _searchController.clear();
), setState(() => _query = '');
prefixIconConstraints: const BoxConstraints( _searchFocusNode.unfocus();
minWidth: TouchTarget.minimum, },
minHeight: TouchTarget.minimum, icon: Icon(
), Platform.isIOS
suffixIcon: _query.isNotEmpty ? CupertinoIcons.clear_circled_solid
? IconButton( : Icons.clear,
onPressed: () { color: conduitTheme.iconSecondary,
_searchController.clear(); size: IconSize.input,
setState(() => _query = ''); ),
_searchFocusNode.unfocus(); )
}, : null,
icon: Icon( suffixIconConstraints: const BoxConstraints(
Platform.isIOS minWidth: TouchTarget.minimum,
? CupertinoIcons.clear_circled_solid minHeight: TouchTarget.minimum,
: Icons.clear, ),
color: conduitTheme.iconSecondary, filled: false,
size: IconSize.input, border: InputBorder.none,
), enabledBorder: InputBorder.none,
) focusedBorder: InputBorder.none,
: null, contentPadding: const EdgeInsets.symmetric(
suffixIconConstraints: const BoxConstraints( horizontal: Spacing.md,
minWidth: TouchTarget.minimum, vertical: Spacing.sm,
minHeight: TouchTarget.minimum,
),
filled: false,
border: InputBorder.none,
enabledBorder: InputBorder.none,
focusedBorder: InputBorder.none,
contentPadding: const EdgeInsets.symmetric(
horizontal: Spacing.md,
vertical: Spacing.sm,
),
),
), ),
), ),
), ),

View File

@@ -1,6 +1,5 @@
import 'dart:async'; import 'dart:async';
import 'dart:io' show Platform; import 'dart:io' show Platform;
import 'dart:ui' show ImageFilter;
import 'package:flutter/cupertino.dart'; import 'package:flutter/cupertino.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
@@ -48,88 +47,12 @@ class AppCustomizationPage extends ConsumerWidget {
final canPop = ModalRoute.of(context)?.canPop ?? false; final canPop = ModalRoute.of(context)?.canPop ?? false;
final topPadding = MediaQuery.of(context).padding.top + kToolbarHeight + 24; final topPadding = MediaQuery.of(context).padding.top + kToolbarHeight + 24;
final theme = Theme.of(context);
final conduitTheme = context.conduitTheme;
return Scaffold( return Scaffold(
backgroundColor: conduitTheme.surfaceBackground, backgroundColor: context.conduitTheme.surfaceBackground,
extendBodyBehindAppBar: true, extendBodyBehindAppBar: true,
appBar: PreferredSize( appBar: FloatingAppBar(
preferredSize: const Size.fromHeight(kToolbarHeight + 8), leading: canPop ? const FloatingAppBarBackButton() : null,
child: Container( title: FloatingAppBarTitle(text: l10n.appCustomization),
decoration: BoxDecoration(
gradient: LinearGradient(
begin: Alignment.topCenter,
end: Alignment.bottomCenter,
stops: const [0.0, 0.4, 1.0],
colors: [
theme.scaffoldBackgroundColor,
theme.scaffoldBackgroundColor.withValues(alpha: 0.85),
theme.scaffoldBackgroundColor.withValues(alpha: 0.0),
],
),
),
child: SafeArea(
bottom: false,
child: SizedBox(
height: kToolbarHeight,
child: Row(
children: [
// Leading (back button)
if (canPop)
Padding(
padding: const EdgeInsets.only(left: Spacing.inputPadding),
child: Center(
child: GestureDetector(
onTap: () => Navigator.of(context).maybePop(),
child: _buildAppBarPill(
context,
Icon(
UiUtils.platformIcon(
ios: CupertinoIcons.back,
android: Icons.arrow_back,
),
color: conduitTheme.textPrimary,
size: IconSize.appBar,
),
isCircular: true,
),
),
),
)
else
const SizedBox(width: Spacing.inputPadding),
// Title centered
Expanded(
child: Center(
child: _buildAppBarPill(
context,
Padding(
padding: const EdgeInsets.symmetric(
horizontal: Spacing.md,
vertical: Spacing.xs,
),
child: Text(
l10n.appCustomization,
style: AppTypography.headlineSmallStyle.copyWith(
color: conduitTheme.textPrimary,
fontWeight: FontWeight.w600,
),
),
),
),
),
),
// Trailing spacer to balance
if (canPop)
const SizedBox(width: 44 + Spacing.inputPadding)
else
const SizedBox(width: Spacing.inputPadding),
],
),
),
),
),
), ),
body: ListView( body: ListView(
physics: const BouncingScrollPhysics( physics: const BouncingScrollPhysics(
@@ -170,63 +93,6 @@ class AppCustomizationPage extends ConsumerWidget {
); );
} }
Widget _buildAppBarPill(
BuildContext context,
Widget child, {
bool isCircular = false,
}) {
final theme = Theme.of(context);
final isDark = theme.brightness == Brightness.dark;
final backgroundColor = isDark
? Color.lerp(context.conduitTheme.cardBackground, Colors.white, 0.08)!
: Color.lerp(context.conduitTheme.inputBackground, Colors.black, 0.06)!;
final borderColor = context.conduitTheme.cardBorder.withValues(
alpha: isDark ? 0.65 : 0.55,
);
final borderRadius = isCircular
? BorderRadius.circular(100)
: BorderRadius.circular(AppBorderRadius.pill);
if (isCircular) {
return SizedBox(
width: 44,
height: 44,
child: ClipRRect(
borderRadius: borderRadius,
child: BackdropFilter(
filter: ImageFilter.blur(sigmaX: 16, sigmaY: 16),
child: Container(
decoration: BoxDecoration(
color: backgroundColor.withValues(alpha: 0.85),
borderRadius: borderRadius,
border: Border.all(color: borderColor, width: BorderWidth.thin),
),
child: Center(child: child),
),
),
),
);
}
return ClipRRect(
borderRadius: borderRadius,
child: BackdropFilter(
filter: ImageFilter.blur(sigmaX: 16, sigmaY: 16),
child: Container(
decoration: BoxDecoration(
color: backgroundColor.withValues(alpha: 0.85),
borderRadius: borderRadius,
border: Border.all(color: borderColor, width: BorderWidth.thin),
),
child: child,
),
),
);
}
Widget _buildThemesDropdownSection( Widget _buildThemesDropdownSection(
BuildContext context, BuildContext context,
WidgetRef ref, WidgetRef ref,

View File

@@ -9,7 +9,6 @@ import 'package:url_launcher/url_launcher_string.dart';
import 'package:conduit/l10n/app_localizations.dart'; import 'package:conduit/l10n/app_localizations.dart';
import '../../../core/widgets/error_boundary.dart'; import '../../../core/widgets/error_boundary.dart';
import '../../../shared/widgets/improved_loading_states.dart'; import '../../../shared/widgets/improved_loading_states.dart';
import 'dart:ui' show ImageFilter;
import '../../../shared/utils/ui_utils.dart'; import '../../../shared/utils/ui_utils.dart';
import '../../../shared/widgets/themed_dialogs.dart'; import '../../../shared/widgets/themed_dialogs.dart';
@@ -68,151 +67,19 @@ class ProfilePage extends ConsumerWidget {
Scaffold _buildScaffold(BuildContext context, {required Widget body}) { Scaffold _buildScaffold(BuildContext context, {required Widget body}) {
final canPop = ModalRoute.of(context)?.canPop ?? false; final canPop = ModalRoute.of(context)?.canPop ?? false;
final theme = Theme.of(context);
final l10n = AppLocalizations.of(context)!; final l10n = AppLocalizations.of(context)!;
final conduitTheme = context.conduitTheme;
return Scaffold( return Scaffold(
backgroundColor: conduitTheme.surfaceBackground, backgroundColor: context.conduitTheme.surfaceBackground,
extendBodyBehindAppBar: true, extendBodyBehindAppBar: true,
appBar: PreferredSize( appBar: FloatingAppBar(
preferredSize: const Size.fromHeight(kToolbarHeight + 8), leading: canPop ? const FloatingAppBarBackButton() : null,
child: Container( title: FloatingAppBarTitle(text: l10n.you),
decoration: BoxDecoration(
gradient: LinearGradient(
begin: Alignment.topCenter,
end: Alignment.bottomCenter,
stops: const [0.0, 0.4, 1.0],
colors: [
theme.scaffoldBackgroundColor,
theme.scaffoldBackgroundColor.withValues(alpha: 0.85),
theme.scaffoldBackgroundColor.withValues(alpha: 0.0),
],
),
),
child: SafeArea(
bottom: false,
child: SizedBox(
height: kToolbarHeight,
child: Row(
children: [
// Leading (back button)
if (canPop)
Padding(
padding: const EdgeInsets.only(left: Spacing.inputPadding),
child: Center(
child: GestureDetector(
onTap: () => Navigator.of(context).maybePop(),
child: _buildAppBarPill(
context,
Icon(
UiUtils.platformIcon(
ios: CupertinoIcons.back,
android: Icons.arrow_back,
),
color: conduitTheme.textPrimary,
size: IconSize.appBar,
),
isCircular: true,
),
),
),
)
else
const SizedBox(width: Spacing.inputPadding),
// Title centered
Expanded(
child: Center(
child: _buildAppBarPill(
context,
Padding(
padding: const EdgeInsets.symmetric(
horizontal: Spacing.md,
vertical: Spacing.xs,
),
child: Text(
l10n.you,
style: AppTypography.headlineSmallStyle.copyWith(
color: conduitTheme.textPrimary,
fontWeight: FontWeight.w600,
),
),
),
),
),
),
// Trailing spacer to balance
if (canPop)
const SizedBox(width: 44 + Spacing.inputPadding)
else
const SizedBox(width: Spacing.inputPadding),
],
),
),
),
),
), ),
body: body, body: body,
); );
} }
Widget _buildAppBarPill(
BuildContext context,
Widget child, {
bool isCircular = false,
}) {
final theme = Theme.of(context);
final isDark = theme.brightness == Brightness.dark;
final backgroundColor = isDark
? Color.lerp(context.conduitTheme.cardBackground, Colors.white, 0.08)!
: Color.lerp(context.conduitTheme.inputBackground, Colors.black, 0.06)!;
final borderColor = context.conduitTheme.cardBorder.withValues(
alpha: isDark ? 0.65 : 0.55,
);
final borderRadius = isCircular
? BorderRadius.circular(100)
: BorderRadius.circular(AppBorderRadius.pill);
if (isCircular) {
return SizedBox(
width: 44,
height: 44,
child: ClipRRect(
borderRadius: borderRadius,
child: BackdropFilter(
filter: ImageFilter.blur(sigmaX: 16, sigmaY: 16),
child: Container(
decoration: BoxDecoration(
color: backgroundColor.withValues(alpha: 0.85),
borderRadius: borderRadius,
border: Border.all(color: borderColor, width: BorderWidth.thin),
),
child: Center(child: child),
),
),
),
);
}
return ClipRRect(
borderRadius: borderRadius,
child: BackdropFilter(
filter: ImageFilter.blur(sigmaX: 16, sigmaY: 16),
child: Container(
decoration: BoxDecoration(
color: backgroundColor.withValues(alpha: 0.85),
borderRadius: borderRadius,
border: Border.all(color: borderColor, width: BorderWidth.thin),
),
child: child,
),
),
);
}
Widget _buildCenteredState(BuildContext context, Widget child) { Widget _buildCenteredState(BuildContext context, Widget child) {
final topPadding = MediaQuery.of(context).padding.top + kToolbarHeight + 24; final topPadding = MediaQuery.of(context).padding.top + kToolbarHeight + 24;
return Padding( return Padding(

View File

@@ -1,3 +1,5 @@
import 'dart:ui' show ImageFilter;
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter_riverpod/flutter_riverpod.dart'; import 'package:flutter_riverpod/flutter_riverpod.dart';
import '../theme/theme_extensions.dart'; import '../theme/theme_extensions.dart';
@@ -10,6 +12,307 @@ import '../../core/services/settings_service.dart';
/// Unified component library following Conduit design patterns /// Unified component library following Conduit design patterns
/// This provides consistent, reusable UI components throughout the app /// This provides consistent, reusable UI components throughout the app
// =============================================================================
// FLOATING APP BAR COMPONENTS
// =============================================================================
/// A pill-shaped container with blur effect for floating app bar elements.
/// Used for back buttons, titles, and action buttons in the floating app bar.
class FloatingAppBarPill extends StatelessWidget {
final Widget child;
final bool isCircular;
const FloatingAppBarPill({
super.key,
required this.child,
this.isCircular = false,
});
@override
Widget build(BuildContext context) {
final theme = Theme.of(context);
final conduitTheme = context.conduitTheme;
final isDark = theme.brightness == Brightness.dark;
final backgroundColor = isDark
? Color.lerp(conduitTheme.cardBackground, Colors.white, 0.08)!
: Color.lerp(conduitTheme.inputBackground, Colors.black, 0.06)!;
final borderColor = conduitTheme.cardBorder.withValues(
alpha: isDark ? 0.65 : 0.55,
);
final borderRadius = isCircular
? BorderRadius.circular(100)
: BorderRadius.circular(AppBorderRadius.pill);
if (isCircular) {
return SizedBox(
width: 44,
height: 44,
child: ClipRRect(
borderRadius: borderRadius,
child: BackdropFilter(
filter: ImageFilter.blur(sigmaX: 16, sigmaY: 16),
child: Container(
decoration: BoxDecoration(
color: backgroundColor.withValues(alpha: 0.85),
borderRadius: borderRadius,
border: Border.all(color: borderColor, width: BorderWidth.thin),
),
child: Center(child: child),
),
),
),
);
}
return ClipRRect(
borderRadius: borderRadius,
child: BackdropFilter(
filter: ImageFilter.blur(sigmaX: 16, sigmaY: 16),
child: Container(
decoration: BoxDecoration(
color: backgroundColor.withValues(alpha: 0.85),
borderRadius: borderRadius,
border: Border.all(color: borderColor, width: BorderWidth.thin),
),
child: child,
),
),
);
}
}
/// A floating app bar with gradient background and pill-shaped elements.
/// Provides a consistent app bar style across the app with blur effects.
///
/// Supports:
/// - Simple title with optional leading/actions
/// - Custom title widget for complex layouts
/// - Bottom widget for search bars or other content
/// - Flexible actions positioning
class FloatingAppBar extends StatelessWidget implements PreferredSizeWidget {
/// Leading widget (typically a back button or menu button)
final Widget? leading;
/// Title widget - can be a simple [FloatingAppBarTitle] or custom widget
final Widget title;
/// Action widgets displayed on the right side
final List<Widget>? actions;
/// Bottom widget displayed below the main row (e.g., search bar)
final Widget? bottom;
/// Height of the bottom widget (used for preferredSize calculation)
final double bottomHeight;
/// Whether to show a trailing spacer when there's a leading widget but no actions
/// Set to false if you want the title to use all available space
final bool balanceLeading;
const FloatingAppBar({
super.key,
this.leading,
required this.title,
this.actions,
this.bottom,
this.bottomHeight = 0,
this.balanceLeading = true,
});
@override
Size get preferredSize => Size.fromHeight(kToolbarHeight + bottomHeight);
@override
Widget build(BuildContext context) {
final theme = Theme.of(context);
return Container(
decoration: BoxDecoration(
gradient: LinearGradient(
begin: Alignment.topCenter,
end: Alignment.bottomCenter,
stops: const [0.0, 0.4, 1.0],
colors: [
theme.scaffoldBackgroundColor,
theme.scaffoldBackgroundColor.withValues(alpha: 0.85),
theme.scaffoldBackgroundColor.withValues(alpha: 0.0),
],
),
),
child: SafeArea(
bottom: false,
child: Column(
mainAxisSize: MainAxisSize.min,
children: [
SizedBox(
height: kToolbarHeight,
child: Row(
children: [
// Leading
if (leading != null)
Padding(
padding: const EdgeInsets.only(left: Spacing.inputPadding),
child: Center(child: leading),
)
else
const SizedBox(width: Spacing.inputPadding),
// Title centered
Expanded(
child: Center(child: title),
),
// Actions or trailing spacer
if (actions != null && actions!.isNotEmpty)
Row(
mainAxisSize: MainAxisSize.min,
children: actions!,
)
else if (leading != null && balanceLeading)
const SizedBox(width: 44 + Spacing.inputPadding)
else
const SizedBox(width: Spacing.inputPadding),
],
),
),
if (bottom != null) bottom!,
],
),
),
);
}
}
/// Helper to build a standard floating app bar title pill with text.
class FloatingAppBarTitle extends StatelessWidget {
final String text;
final IconData? icon;
const FloatingAppBarTitle({
super.key,
required this.text,
this.icon,
});
@override
Widget build(BuildContext context) {
final conduitTheme = context.conduitTheme;
return FloatingAppBarPill(
child: Padding(
padding: const EdgeInsets.symmetric(
horizontal: Spacing.md,
vertical: Spacing.xs,
),
child: Row(
mainAxisSize: MainAxisSize.min,
children: [
if (icon != null) ...[
Icon(
icon,
color: conduitTheme.textPrimary.withValues(alpha: 0.7),
size: IconSize.md,
),
const SizedBox(width: Spacing.sm),
],
Text(
text,
style: AppTypography.headlineSmallStyle.copyWith(
color: conduitTheme.textPrimary,
fontWeight: FontWeight.w600,
),
),
],
),
),
);
}
}
/// Helper to build a standard floating app bar back button.
class FloatingAppBarBackButton extends StatelessWidget {
final VoidCallback? onTap;
final IconData? icon;
const FloatingAppBarBackButton({
super.key,
this.onTap,
this.icon,
});
@override
Widget build(BuildContext context) {
final conduitTheme = context.conduitTheme;
final isIOS = Theme.of(context).platform == TargetPlatform.iOS;
return GestureDetector(
onTap: onTap ?? () => Navigator.of(context).maybePop(),
child: FloatingAppBarPill(
isCircular: true,
child: Icon(
icon ?? (isIOS ? Icons.arrow_back_ios_new : Icons.arrow_back),
color: conduitTheme.textPrimary,
size: IconSize.appBar,
),
),
);
}
}
/// Helper to build a floating app bar icon button (circular pill with icon).
class FloatingAppBarIconButton extends StatelessWidget {
final IconData icon;
final VoidCallback? onTap;
final Color? iconColor;
const FloatingAppBarIconButton({
super.key,
required this.icon,
this.onTap,
this.iconColor,
});
@override
Widget build(BuildContext context) {
final conduitTheme = context.conduitTheme;
return GestureDetector(
onTap: onTap,
child: FloatingAppBarPill(
isCircular: true,
child: Icon(
icon,
color: iconColor ?? conduitTheme.textPrimary,
size: IconSize.appBar,
),
),
);
}
}
/// Helper to build a floating app bar action with padding.
class FloatingAppBarAction extends StatelessWidget {
final Widget child;
const FloatingAppBarAction({
super.key,
required this.child,
});
@override
Widget build(BuildContext context) {
return Padding(
padding: const EdgeInsets.only(right: Spacing.inputPadding),
child: Center(child: child),
);
}
}
// =============================================================================
// EXISTING COMPONENTS
// =============================================================================
class ConduitButton extends ConsumerWidget { class ConduitButton extends ConsumerWidget {
final String text; final String text;
final VoidCallback? onPressed; final VoidCallback? onPressed;