Merge pull request #265 from cogwheel0/refactor-chat-input-styling
refactor-chat-input-styling
This commit is contained in:
File diff suppressed because it is too large
Load Diff
@@ -9,7 +9,6 @@ import 'package:flutter_riverpod/flutter_riverpod.dart';
|
|||||||
|
|
||||||
import 'dart:io' show Platform;
|
import 'dart:io' show Platform;
|
||||||
import 'dart:async';
|
import 'dart:async';
|
||||||
import 'dart:ui';
|
|
||||||
import 'dart:math' as math;
|
import 'dart:math' as math;
|
||||||
import '../providers/chat_providers.dart';
|
import '../providers/chat_providers.dart';
|
||||||
import '../services/clipboard_attachment_service.dart';
|
import '../services/clipboard_attachment_service.dart';
|
||||||
@@ -1073,10 +1072,10 @@ class _ModernChatInputState extends ConsumerState<ModernChatInput>
|
|||||||
|
|
||||||
final Brightness brightness = Theme.of(context).brightness;
|
final Brightness brightness = Theme.of(context).brightness;
|
||||||
final bool isActive = _focusNode.hasFocus || _hasText;
|
final bool isActive = _focusNode.hasFocus || _hasText;
|
||||||
final Color composerSurface = context.conduitTheme.inputBackground;
|
// Use high-contrast background for floating input
|
||||||
final Color composerBackground = brightness == Brightness.dark
|
final Color composerBackground = brightness == Brightness.dark
|
||||||
? composerSurface.withValues(alpha: 0.78)
|
? Color.lerp(context.conduitTheme.cardBackground, Colors.white, 0.08)!
|
||||||
: context.conduitTheme.surfaceContainerHighest;
|
: Color.lerp(context.conduitTheme.inputBackground, Colors.black, 0.06)!;
|
||||||
final Color placeholderBase = context.conduitTheme.inputText.withValues(
|
final Color placeholderBase = context.conduitTheme.inputText.withValues(
|
||||||
alpha: 0.64,
|
alpha: 0.64,
|
||||||
);
|
);
|
||||||
@@ -1087,7 +1086,7 @@ class _ModernChatInputState extends ConsumerState<ModernChatInput>
|
|||||||
context.conduitTheme.inputBorder,
|
context.conduitTheme.inputBorder,
|
||||||
context.conduitTheme.inputBorderFocused,
|
context.conduitTheme.inputBorderFocused,
|
||||||
isActive ? 1.0 : 0.0,
|
isActive ? 1.0 : 0.0,
|
||||||
)!.withValues(alpha: brightness == Brightness.dark ? 0.55 : 0.45);
|
)!.withValues(alpha: brightness == Brightness.dark ? 0.65 : 0.55);
|
||||||
final Color shellShadowColor = context.conduitTheme.cardShadow.withValues(
|
final Color shellShadowColor = context.conduitTheme.cardShadow.withValues(
|
||||||
alpha: brightness == Brightness.dark
|
alpha: brightness == Brightness.dark
|
||||||
? 0.22 + (isActive ? 0.08 : 0.0)
|
? 0.22 + (isActive ? 0.08 : 0.0)
|
||||||
@@ -1209,21 +1208,17 @@ class _ModernChatInputState extends ConsumerState<ModernChatInput>
|
|||||||
);
|
);
|
||||||
|
|
||||||
final BoxDecoration shellDecoration = BoxDecoration(
|
final BoxDecoration shellDecoration = BoxDecoration(
|
||||||
color: showCompactComposer ? Colors.transparent : composerBackground,
|
color: composerBackground,
|
||||||
borderRadius: shellRadius,
|
borderRadius: shellRadius,
|
||||||
border: showCompactComposer
|
border: Border.all(color: outlineColor, width: BorderWidth.thin),
|
||||||
? null
|
boxShadow: <BoxShadow>[
|
||||||
: Border.all(color: outlineColor, width: BorderWidth.thin),
|
BoxShadow(
|
||||||
boxShadow: showCompactComposer
|
color: shellShadowColor,
|
||||||
? const <BoxShadow>[]
|
blurRadius: 12 + (isActive ? 4 : 0),
|
||||||
: <BoxShadow>[
|
spreadRadius: -2,
|
||||||
BoxShadow(
|
offset: const Offset(0, -2),
|
||||||
color: shellShadowColor,
|
),
|
||||||
blurRadius: 12 + (isActive ? 4 : 0),
|
],
|
||||||
spreadRadius: -2,
|
|
||||||
offset: const Offset(0, -2),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
);
|
);
|
||||||
|
|
||||||
final List<Widget> composerChildren = <Widget>[
|
final List<Widget> composerChildren = <Widget>[
|
||||||
@@ -1238,82 +1233,7 @@ class _ModernChatInputState extends ConsumerState<ModernChatInput>
|
|||||||
),
|
),
|
||||||
child: _buildPromptOverlay(context),
|
child: _buildPromptOverlay(context),
|
||||||
),
|
),
|
||||||
if (showCompactComposer)
|
if (!showCompactComposer) ...[
|
||||||
Padding(
|
|
||||||
key: const ValueKey('composer-compact'),
|
|
||||||
padding: const EdgeInsets.fromLTRB(
|
|
||||||
Spacing.screenPadding,
|
|
||||||
Spacing.xs,
|
|
||||||
Spacing.screenPadding,
|
|
||||||
Spacing.sm,
|
|
||||||
),
|
|
||||||
child: Row(
|
|
||||||
crossAxisAlignment: CrossAxisAlignment.end,
|
|
||||||
children: [
|
|
||||||
_buildOverflowButton(
|
|
||||||
tooltip: AppLocalizations.of(context)!.more,
|
|
||||||
webSearchActive: webSearchEnabled,
|
|
||||||
imageGenerationActive: imageGenEnabled,
|
|
||||||
toolsActive: selectedToolIds.isNotEmpty,
|
|
||||||
filtersActive: selectedFilterIds.isNotEmpty,
|
|
||||||
),
|
|
||||||
const SizedBox(width: Spacing.sm),
|
|
||||||
Expanded(
|
|
||||||
child: ConstrainedBox(
|
|
||||||
constraints: BoxConstraints(
|
|
||||||
maxHeight: MediaQuery.of(context).size.height * 0.25,
|
|
||||||
),
|
|
||||||
child: AnimatedContainer(
|
|
||||||
duration: const Duration(milliseconds: 180),
|
|
||||||
curve: Curves.easeOutCubic,
|
|
||||||
padding: const EdgeInsets.symmetric(horizontal: Spacing.md),
|
|
||||||
constraints: const BoxConstraints(
|
|
||||||
minHeight: TouchTarget.input,
|
|
||||||
),
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
color: composerSurface.withValues(
|
|
||||||
alpha: brightness == Brightness.dark ? 0.9 : 0.2,
|
|
||||||
),
|
|
||||||
borderRadius: BorderRadius.circular(_composerRadius),
|
|
||||||
border: Border.all(
|
|
||||||
color: outlineColor.withValues(
|
|
||||||
alpha: brightness == Brightness.dark ? 0.32 : 0.2,
|
|
||||||
),
|
|
||||||
width: BorderWidth.micro,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
child: Row(
|
|
||||||
children: [
|
|
||||||
Expanded(
|
|
||||||
child: _buildComposerTextField(
|
|
||||||
brightness: brightness,
|
|
||||||
sendOnEnter: sendOnEnter,
|
|
||||||
placeholderBase: placeholderBase,
|
|
||||||
placeholderFocused: placeholderFocused,
|
|
||||||
contentPadding: const EdgeInsets.symmetric(
|
|
||||||
vertical: Spacing.xs,
|
|
||||||
),
|
|
||||||
isActive: isActive,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
if (!_hasText && voiceAvailable && !isGenerating)
|
|
||||||
_buildInlineMicIcon(voiceAvailable),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(width: Spacing.sm),
|
|
||||||
_buildPrimaryButton(
|
|
||||||
_hasText,
|
|
||||||
isGenerating,
|
|
||||||
stopGeneration,
|
|
||||||
voiceAvailable,
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
)
|
|
||||||
else ...[
|
|
||||||
Padding(
|
Padding(
|
||||||
key: const ValueKey('composer-expanded-input'),
|
key: const ValueKey('composer-expanded-input'),
|
||||||
padding: const EdgeInsets.fromLTRB(
|
padding: const EdgeInsets.fromLTRB(
|
||||||
@@ -1359,7 +1279,7 @@ class _ModernChatInputState extends ConsumerState<ModernChatInput>
|
|||||||
Spacing.inputPadding,
|
Spacing.inputPadding,
|
||||||
0,
|
0,
|
||||||
Spacing.inputPadding,
|
Spacing.inputPadding,
|
||||||
0,
|
Spacing.sm,
|
||||||
),
|
),
|
||||||
child: Row(
|
child: Row(
|
||||||
children: [
|
children: [
|
||||||
@@ -1405,29 +1325,91 @@ class _ModernChatInputState extends ConsumerState<ModernChatInput>
|
|||||||
],
|
],
|
||||||
];
|
];
|
||||||
|
|
||||||
|
// For compact mode, render text field shell with floating buttons on sides
|
||||||
|
if (showCompactComposer) {
|
||||||
|
// Build the text field shell
|
||||||
|
Widget textFieldShell = AnimatedContainer(
|
||||||
|
duration: const Duration(milliseconds: 180),
|
||||||
|
curve: Curves.easeOutCubic,
|
||||||
|
padding: const EdgeInsets.symmetric(horizontal: Spacing.md),
|
||||||
|
constraints: const BoxConstraints(minHeight: TouchTarget.input),
|
||||||
|
decoration: shellDecoration,
|
||||||
|
child: ConstrainedBox(
|
||||||
|
constraints: BoxConstraints(
|
||||||
|
maxHeight: MediaQuery.of(context).size.height * 0.25,
|
||||||
|
),
|
||||||
|
child: Row(
|
||||||
|
children: [
|
||||||
|
Expanded(
|
||||||
|
child: _buildComposerTextField(
|
||||||
|
brightness: brightness,
|
||||||
|
sendOnEnter: sendOnEnter,
|
||||||
|
placeholderBase: placeholderBase,
|
||||||
|
placeholderFocused: placeholderFocused,
|
||||||
|
contentPadding: const EdgeInsets.symmetric(
|
||||||
|
vertical: Spacing.xs,
|
||||||
|
),
|
||||||
|
isActive: isActive,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
if (!_hasText && voiceAvailable && !isGenerating)
|
||||||
|
_buildInlineMicIcon(voiceAvailable),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
|
||||||
|
final bottomPadding = MediaQuery.of(context).viewPadding.bottom;
|
||||||
|
return Padding(
|
||||||
|
padding: EdgeInsets.fromLTRB(
|
||||||
|
Spacing.screenPadding,
|
||||||
|
0,
|
||||||
|
Spacing.screenPadding,
|
||||||
|
bottomPadding + Spacing.md,
|
||||||
|
),
|
||||||
|
child: Row(
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.end,
|
||||||
|
children: [
|
||||||
|
_buildOverflowButton(
|
||||||
|
tooltip: AppLocalizations.of(context)!.more,
|
||||||
|
webSearchActive: webSearchEnabled,
|
||||||
|
imageGenerationActive: imageGenEnabled,
|
||||||
|
toolsActive: selectedToolIds.isNotEmpty,
|
||||||
|
filtersActive: selectedFilterIds.isNotEmpty,
|
||||||
|
),
|
||||||
|
const SizedBox(width: Spacing.sm),
|
||||||
|
Expanded(child: textFieldShell),
|
||||||
|
const SizedBox(width: Spacing.sm),
|
||||||
|
_buildPrimaryButton(
|
||||||
|
_hasText,
|
||||||
|
isGenerating,
|
||||||
|
stopGeneration,
|
||||||
|
voiceAvailable,
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
// For expanded mode with quick pills, use the full shell
|
||||||
Widget shell = AnimatedContainer(
|
Widget shell = AnimatedContainer(
|
||||||
duration: const Duration(milliseconds: 180),
|
duration: const Duration(milliseconds: 180),
|
||||||
curve: Curves.easeOutCubic,
|
curve: Curves.easeOutCubic,
|
||||||
decoration: shellDecoration,
|
decoration: shellDecoration,
|
||||||
width: double.infinity,
|
child: ConstrainedBox(
|
||||||
child: SafeArea(
|
constraints: BoxConstraints(
|
||||||
top: false,
|
maxHeight: MediaQuery.of(context).size.height * 0.4,
|
||||||
bottom: true,
|
),
|
||||||
child: ConstrainedBox(
|
child: AnimatedSize(
|
||||||
constraints: BoxConstraints(
|
duration: const Duration(milliseconds: 160),
|
||||||
maxHeight: MediaQuery.of(context).size.height * 0.4,
|
curve: Curves.easeOutCubic,
|
||||||
),
|
alignment: Alignment.topCenter,
|
||||||
child: AnimatedSize(
|
child: SingleChildScrollView(
|
||||||
duration: const Duration(milliseconds: 160),
|
physics: const ClampingScrollPhysics(),
|
||||||
curve: Curves.easeOutCubic,
|
child: RepaintBoundary(
|
||||||
alignment: Alignment.topCenter,
|
child: Column(
|
||||||
child: SingleChildScrollView(
|
mainAxisSize: MainAxisSize.min,
|
||||||
physics: const ClampingScrollPhysics(),
|
children: composerChildren,
|
||||||
child: RepaintBoundary(
|
|
||||||
child: Column(
|
|
||||||
mainAxisSize: MainAxisSize.min,
|
|
||||||
children: composerChildren,
|
|
||||||
),
|
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
@@ -1435,20 +1417,16 @@ class _ModernChatInputState extends ConsumerState<ModernChatInput>
|
|||||||
),
|
),
|
||||||
);
|
);
|
||||||
|
|
||||||
if (brightness == Brightness.dark && !showCompactComposer) {
|
// Wrap with padding for floating effect, accounting for safe area
|
||||||
shell = ClipRRect(
|
final bottomPadding = MediaQuery.of(context).viewPadding.bottom;
|
||||||
borderRadius: shellRadius,
|
return Padding(
|
||||||
child: BackdropFilter(
|
padding: EdgeInsets.fromLTRB(
|
||||||
filter: ImageFilter.blur(sigmaX: 12, sigmaY: 12),
|
Spacing.screenPadding,
|
||||||
child: shell,
|
0,
|
||||||
),
|
Spacing.screenPadding,
|
||||||
);
|
bottomPadding + Spacing.md,
|
||||||
}
|
),
|
||||||
|
child: shell,
|
||||||
return Container(
|
|
||||||
color: Colors.transparent,
|
|
||||||
padding: EdgeInsets.zero,
|
|
||||||
child: Column(mainAxisSize: MainAxisSize.min, children: [shell]),
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1688,9 +1666,11 @@ class _ModernChatInputState extends ConsumerState<ModernChatInput>
|
|||||||
: (activeColor ??
|
: (activeColor ??
|
||||||
context.conduitTheme.textPrimary.withValues(alpha: Alpha.strong));
|
context.conduitTheme.textPrimary.withValues(alpha: Alpha.strong));
|
||||||
|
|
||||||
|
// Use high-contrast background for floating button
|
||||||
final Brightness brightness = Theme.of(context).brightness;
|
final Brightness brightness = Theme.of(context).brightness;
|
||||||
final Color baseBackground = context.conduitTheme.inputBackground
|
final Color baseBackground = brightness == Brightness.dark
|
||||||
.withValues(alpha: brightness == Brightness.dark ? 0.9 : 0.2);
|
? Color.lerp(context.conduitTheme.cardBackground, Colors.white, 0.08)!
|
||||||
|
: Color.lerp(context.conduitTheme.inputBackground, Colors.black, 0.06)!;
|
||||||
final Color backgroundColor = !enabled
|
final Color backgroundColor = !enabled
|
||||||
? baseBackground.withValues(alpha: Alpha.disabled)
|
? baseBackground.withValues(alpha: Alpha.disabled)
|
||||||
: isActive
|
: isActive
|
||||||
|
|||||||
@@ -2,6 +2,7 @@ import 'dart:math' as math;
|
|||||||
|
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:flutter/cupertino.dart';
|
import 'package:flutter/cupertino.dart';
|
||||||
|
import 'package:flutter/services.dart';
|
||||||
import 'package:flutter_animate/flutter_animate.dart';
|
import 'package:flutter_animate/flutter_animate.dart';
|
||||||
import 'theme_extensions.dart';
|
import 'theme_extensions.dart';
|
||||||
import 'tweakcn_themes.dart';
|
import 'tweakcn_themes.dart';
|
||||||
@@ -115,6 +116,13 @@ class AppTheme {
|
|||||||
elevation: Elevation.none,
|
elevation: Elevation.none,
|
||||||
backgroundColor: surfaces.background,
|
backgroundColor: surfaces.background,
|
||||||
foregroundColor: tokens.neutralOnSurface,
|
foregroundColor: tokens.neutralOnSurface,
|
||||||
|
systemOverlayStyle: SystemUiOverlayStyle(
|
||||||
|
statusBarBrightness: brightness,
|
||||||
|
statusBarIconBrightness: isDark ? Brightness.light : Brightness.dark,
|
||||||
|
systemNavigationBarIconBrightness: isDark
|
||||||
|
? Brightness.light
|
||||||
|
: Brightness.dark,
|
||||||
|
),
|
||||||
),
|
),
|
||||||
bottomSheetTheme: BottomSheetThemeData(
|
bottomSheetTheme: BottomSheetThemeData(
|
||||||
backgroundColor: surfaces.card,
|
backgroundColor: surfaces.card,
|
||||||
|
|||||||
Reference in New Issue
Block a user