Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Edit-message (3/n): Implement ZulipWebUiKitButton, for "Cancel" / "Save" buttons #1432

Open
wants to merge 2 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
197 changes: 197 additions & 0 deletions lib/widgets/button.dart
Original file line number Diff line number Diff line change
@@ -0,0 +1,197 @@
import 'package:flutter/material.dart';

import 'color.dart';
import 'text.dart';
import 'theme.dart';

/// The "Button" component from Zulip Web UI kit,
/// plus outer vertical padding to make the touch target 44px tall.
///
/// The Figma uses this for the "Cancel" and "Save" buttons in the compose box
/// for editing an already-sent message.
///
/// See Figma:
/// * Component: https://www.figma.com/design/msWyAJ8cnMHgOMPxi7BUvA/Zulip-Web-UI-kit?node-id=1-2780&t=Wia0D0i1I0GXdD9z-0
/// * Edit-message compose box: https://www.figma.com/design/1JTNtYo9memgW7vV6d0ygq/Zulip-Mobile?node-id=3988-38201&m=dev
class ZulipWebUiKitButton extends StatelessWidget {
const ZulipWebUiKitButton({
super.key,
this.attention = ZulipWebUiKitButtonAttention.medium,
this.intent = ZulipWebUiKitButtonIntent.info,
required this.label,
required this.onPressed,
});

final ZulipWebUiKitButtonAttention attention;
final ZulipWebUiKitButtonIntent intent;
final String label;
final VoidCallback onPressed;

WidgetStateColor _backgroundColor(DesignVariables designVariables) {
switch ((attention, intent)) {
case (ZulipWebUiKitButtonAttention.medium, ZulipWebUiKitButtonIntent.info):
return WidgetStateColor.fromMap({
WidgetState.pressed: designVariables.btnBgAttMediumIntInfoActive,
~WidgetState.pressed: designVariables.btnBgAttMediumIntInfoNormal,
});
case (ZulipWebUiKitButtonAttention.high, ZulipWebUiKitButtonIntent.info):
return WidgetStateColor.fromMap({
WidgetState.pressed: designVariables.btnBgAttHighIntInfoActive,
~WidgetState.pressed: designVariables.btnBgAttHighIntInfoNormal,
});
}
}

Color _labelColor(DesignVariables designVariables) {
switch ((attention, intent)) {
case (ZulipWebUiKitButtonAttention.medium, ZulipWebUiKitButtonIntent.info):
return designVariables.btnLabelAttMediumIntInfo;
case (ZulipWebUiKitButtonAttention.high, ZulipWebUiKitButtonIntent.info):
return designVariables.btnLabelAttHigh;
}
}

TextStyle _labelStyle(BuildContext context, {required TextScaler textScaler}) {
final designVariables = DesignVariables.of(context);
// Values chosen from the Figma frame for zulip-flutter's compose box:
// https://www.figma.com/design/1JTNtYo9memgW7vV6d0ygq/Zulip-Mobile?node-id=3988-38201&m=dev
// Commented values come from the Figma page "Zulip Web UI kit":
// https://www.figma.com/design/msWyAJ8cnMHgOMPxi7BUvA/Zulip-Web-UI-kit?node-id=1-8&p=f&m=dev
// Discussion:
// https://github.com/zulip/zulip-flutter/pull/1432#discussion_r2023880851
return TextStyle(
color: _labelColor(designVariables),
fontSize: 17, // 16
height: 1.20, // 1.25
letterSpacing: proportionalLetterSpacing(context, textScaler: textScaler,
0.006,
baseFontSize: 17), // 16
).merge(weightVariableTextStyle(context,
wght: 600)); // 500
}

BorderSide _borderSide(DesignVariables designVariables) {
switch (attention) {
case ZulipWebUiKitButtonAttention.medium:
// TODO inner shadow effect like `box-shadow: inset`, following Figma;
// needs Flutter support for something like that:
// https://github.com/flutter/flutter/issues/18636
// https://github.com/flutter/flutter/issues/52999
// For now, we just use a solid-stroke border with half the opacity
// and half the width.
return BorderSide(
color: designVariables.btnShadowAttMed.withFadedAlpha(0.5),
width: 0.5);
case ZulipWebUiKitButtonAttention.high:
return BorderSide.none;
}
}

@override
Widget build(BuildContext context) {
final designVariables = DesignVariables.of(context);

// With [MaterialTapTargetSize.padded],
// make [TextButton] set 44 instead of 48 for the touch-target height.
final visualDensity = VisualDensity(vertical: -1);
// A value that [TextButton] adds to some of its layout parameters;
// we can cancel out those adjustments by subtracting it.
final densityVerticalAdjustment = visualDensity.baseSizeAdjustment.dy;
Comment on lines +94 to +99
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Cool, glad you found a way to control this.


// An upper limit when the text-size setting is large
// - helps prioritize more important content (like message content); #1023
// - prevents the vertical padding added by [MaterialTapTargetSize.padded]
// from shrinking to zero as the button grows to accommodate a larger label
final textScaler = MediaQuery.textScalerOf(context).clamp(maxScaleFactor: 1.5);

return AnimatedScaleOnTap(
scaleEnd: 0.96,
duration: Duration(milliseconds: 100),
child: TextButton(
style: TextButton.styleFrom(
padding: EdgeInsets.symmetric(horizontal: 10, vertical: 4 - densityVerticalAdjustment),
foregroundColor: _labelColor(designVariables),
shape: RoundedRectangleBorder(
side: _borderSide(designVariables),
borderRadius: BorderRadius.circular(4)),
splashFactory: NoSplash.splashFactory,

// These three arguments make the button 28px tall vertically,
// but with vertical padding to make the touch target 44px tall:
// https://github.com/zulip/zulip-flutter/pull/1432#discussion_r2023907300
visualDensity: visualDensity,
tapTargetSize: MaterialTapTargetSize.padded,
minimumSize: Size(kMinInteractiveDimension, 28 - densityVerticalAdjustment),
).copyWith(backgroundColor: _backgroundColor(designVariables)),
onPressed: onPressed,
child: ConstrainedBox(
constraints: BoxConstraints(maxWidth: 240),
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I wonder if we need a different value for the mobile app. Presumably this is for web.

Copy link
Collaborator Author

@chrisbobbe chrisbobbe Apr 1, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

If 240 doesn't seem OK, could you say more about why? If it's OK but just not ideal, I think it's fine to leave it as-is and tune later based on feedback or more design work.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Usually when we have wider-than-usual buttons, they occupy the screen-width or the width of the parent; for my phone a 240px button would be a bit narrower than my screen. I think leaving it this way and tuning later sounds like a good plan.

child: Text(label,
textScaler: textScaler,
maxLines: 1,
style: _labelStyle(context, textScaler: textScaler),
textAlign: TextAlign.center,
overflow: TextOverflow.ellipsis))));
}
}

enum ZulipWebUiKitButtonAttention {
high,
medium,
// low,
}

enum ZulipWebUiKitButtonIntent {
// neutral,
// warning,
// danger,
info,
// success,
// brand,
Comment on lines +144 to +150
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

One good reason to add these only lazily, as you're doing here: to keep down how many different colors we have to maintain in theme.dart, until we finish #831.

}

/// Apply [Transform.scale] to the child widget when tapped, and reset its scale
/// when released, while animating the transitions.
class AnimatedScaleOnTap extends StatefulWidget {
const AnimatedScaleOnTap({
super.key,
required this.scaleEnd,
required this.duration,
required this.child,
});

/// The terminal scale to animate to.
final double scaleEnd;

/// The duration over which to animate the scale change.
final Duration duration;

final Widget child;

@override
State<AnimatedScaleOnTap> createState() => _AnimatedScaleOnTapState();
}

class _AnimatedScaleOnTapState extends State<AnimatedScaleOnTap> {
double _scale = 1;

void _changeScale(double scale) {
setState(() {
_scale = scale;
});
}

@override
Widget build(BuildContext context) {
return GestureDetector(
behavior: HitTestBehavior.translucent,
onTapDown: (_) => _changeScale(widget.scaleEnd),
onTapUp: (_) => _changeScale(1),
onTapCancel: () => _changeScale(1),
child: AnimatedScale(
scale: _scale,
duration: widget.duration,
curve: Curves.easeOut,
child: widget.child));
}
}
47 changes: 1 addition & 46 deletions lib/widgets/home.dart
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@ import 'about_zulip.dart';
import 'action_sheet.dart';
import 'app.dart';
import 'app_bar.dart';
import 'button.dart';
import 'color.dart';
import 'content.dart';
import 'icons.dart';
Expand Down Expand Up @@ -601,49 +602,3 @@ class _AboutZulipButton extends _MenuButton {
Navigator.of(context).push(AboutZulipPage.buildRoute(context));
}
}

/// Apply [Transform.scale] to the child widget when tapped, and reset its scale
/// when released, while animating the transitions.
class AnimatedScaleOnTap extends StatefulWidget {
const AnimatedScaleOnTap({
super.key,
required this.scaleEnd,
required this.duration,
required this.child,
});

/// The terminal scale to animate to.
final double scaleEnd;

/// The duration over which to animate the scale change.
final Duration duration;

final Widget child;

@override
State<AnimatedScaleOnTap> createState() => _AnimatedScaleOnTapState();
}

class _AnimatedScaleOnTapState extends State<AnimatedScaleOnTap> {
double _scale = 1;

void _changeScale(double scale) {
setState(() {
_scale = scale;
});
}

@override
Widget build(BuildContext context) {
return GestureDetector(
behavior: HitTestBehavior.translucent,
onTapDown: (_) => _changeScale(widget.scaleEnd),
onTapUp: (_) => _changeScale(1),
onTapCancel: () => _changeScale(1),
child: AnimatedScale(
scale: _scale,
duration: widget.duration,
curve: Curves.easeOut,
child: widget.child));
}
}
49 changes: 49 additions & 0 deletions lib/widgets/theme.dart
Original file line number Diff line number Diff line change
Expand Up @@ -138,8 +138,15 @@ class DesignVariables extends ThemeExtension<DesignVariables> {
bgTopBar: const Color(0xfff5f5f5),
borderBar: Colors.black.withValues(alpha: 0.2),
borderMenuButtonSelected: Colors.black.withValues(alpha: 0.2),
btnBgAttHighIntInfoActive: const Color(0xff1e41d3),
btnBgAttHighIntInfoNormal: const Color(0xff3c6bff),
btnBgAttMediumIntInfoActive: const Color(0xff3c6bff).withValues(alpha: 0.22),
btnBgAttMediumIntInfoNormal: const Color(0xff3c6bff).withValues(alpha: 0.12),
btnLabelAttHigh: const Color(0xffffffff),
btnLabelAttLowIntDanger: const Color(0xffc0070a),
btnLabelAttMediumIntDanger: const Color(0xffac0508),
btnLabelAttMediumIntInfo: const Color(0xff1027a6),
btnShadowAttMed: const Color(0xff000000).withValues(alpha: 0.20),
composeBoxBg: const Color(0xffffffff),
contextMenuCancelText: const Color(0xff222222),
contextMenuItemBg: const Color(0xff6159e1),
Expand Down Expand Up @@ -188,8 +195,15 @@ class DesignVariables extends ThemeExtension<DesignVariables> {
bgTopBar: const Color(0xff242424),
borderBar: const Color(0xffffffff).withValues(alpha: 0.1),
borderMenuButtonSelected: Colors.white.withValues(alpha: 0.1),
btnBgAttHighIntInfoActive: const Color(0xff1e41d3),
btnBgAttHighIntInfoNormal: const Color(0xff1e41d3),
btnBgAttMediumIntInfoActive: const Color(0xff97b6fe).withValues(alpha: 0.12),
btnBgAttMediumIntInfoNormal: const Color(0xff97b6fe).withValues(alpha: 0.12),
btnLabelAttHigh: const Color(0xffffffff).withValues(alpha: 0.85),
btnLabelAttLowIntDanger: const Color(0xffff8b7c),
btnLabelAttMediumIntDanger: const Color(0xffff8b7c),
btnLabelAttMediumIntInfo: const Color(0xff97b6fe),
btnShadowAttMed: const Color(0xffffffff).withValues(alpha: 0.21),
composeBoxBg: const Color(0xff0f0f0f),
contextMenuCancelText: const Color(0xffffffff).withValues(alpha: 0.75),
contextMenuItemBg: const Color(0xff7977fe),
Expand Down Expand Up @@ -246,8 +260,15 @@ class DesignVariables extends ThemeExtension<DesignVariables> {
required this.bgTopBar,
required this.borderBar,
required this.borderMenuButtonSelected,
required this.btnBgAttHighIntInfoActive,
required this.btnBgAttHighIntInfoNormal,
required this.btnBgAttMediumIntInfoActive,
required this.btnBgAttMediumIntInfoNormal,
required this.btnLabelAttHigh,
required this.btnLabelAttLowIntDanger,
required this.btnLabelAttMediumIntDanger,
required this.btnLabelAttMediumIntInfo,
required this.btnShadowAttMed,
required this.composeBoxBg,
required this.contextMenuCancelText,
required this.contextMenuItemBg,
Expand Down Expand Up @@ -305,8 +326,15 @@ class DesignVariables extends ThemeExtension<DesignVariables> {
final Color bgTopBar;
final Color borderBar;
final Color borderMenuButtonSelected;
final Color btnBgAttHighIntInfoActive;
final Color btnBgAttHighIntInfoNormal;
final Color btnBgAttMediumIntInfoActive;
final Color btnBgAttMediumIntInfoNormal;
final Color btnLabelAttHigh;
final Color btnLabelAttLowIntDanger;
final Color btnLabelAttMediumIntDanger;
final Color btnLabelAttMediumIntInfo;
final Color btnShadowAttMed;
final Color composeBoxBg;
final Color contextMenuCancelText;
final Color contextMenuItemBg;
Expand Down Expand Up @@ -359,8 +387,15 @@ class DesignVariables extends ThemeExtension<DesignVariables> {
Color? bgTopBar,
Color? borderBar,
Color? borderMenuButtonSelected,
Color? btnBgAttHighIntInfoActive,
Color? btnBgAttHighIntInfoNormal,
Color? btnBgAttMediumIntInfoActive,
Color? btnBgAttMediumIntInfoNormal,
Color? btnLabelAttHigh,
Color? btnLabelAttLowIntDanger,
Color? btnLabelAttMediumIntDanger,
Color? btnLabelAttMediumIntInfo,
Color? btnShadowAttMed,
Color? composeBoxBg,
Color? contextMenuCancelText,
Color? contextMenuItemBg,
Expand Down Expand Up @@ -408,8 +443,15 @@ class DesignVariables extends ThemeExtension<DesignVariables> {
bgTopBar: bgTopBar ?? this.bgTopBar,
borderBar: borderBar ?? this.borderBar,
borderMenuButtonSelected: borderMenuButtonSelected ?? this.borderMenuButtonSelected,
btnBgAttHighIntInfoActive: btnBgAttHighIntInfoActive ?? this.btnBgAttHighIntInfoActive,
btnBgAttHighIntInfoNormal: btnBgAttHighIntInfoNormal ?? this.btnBgAttHighIntInfoNormal,
btnBgAttMediumIntInfoActive: btnBgAttMediumIntInfoActive ?? this.btnBgAttMediumIntInfoActive,
btnBgAttMediumIntInfoNormal: btnBgAttMediumIntInfoNormal ?? this.btnBgAttMediumIntInfoNormal,
btnLabelAttHigh: btnLabelAttHigh ?? this.btnLabelAttHigh,
btnLabelAttLowIntDanger: btnLabelAttLowIntDanger ?? this.btnLabelAttLowIntDanger,
btnLabelAttMediumIntDanger: btnLabelAttMediumIntDanger ?? this.btnLabelAttMediumIntDanger,
btnLabelAttMediumIntInfo: btnLabelAttMediumIntInfo ?? this.btnLabelAttMediumIntInfo,
btnShadowAttMed: btnShadowAttMed ?? this.btnShadowAttMed,
composeBoxBg: composeBoxBg ?? this.composeBoxBg,
contextMenuCancelText: contextMenuCancelText ?? this.contextMenuCancelText,
contextMenuItemBg: contextMenuItemBg ?? this.contextMenuItemBg,
Expand Down Expand Up @@ -464,8 +506,15 @@ class DesignVariables extends ThemeExtension<DesignVariables> {
bgTopBar: Color.lerp(bgTopBar, other.bgTopBar, t)!,
borderBar: Color.lerp(borderBar, other.borderBar, t)!,
borderMenuButtonSelected: Color.lerp(borderMenuButtonSelected, other.borderMenuButtonSelected, t)!,
btnBgAttHighIntInfoActive: Color.lerp(btnBgAttHighIntInfoActive, other.btnBgAttHighIntInfoActive, t)!,
btnBgAttHighIntInfoNormal: Color.lerp(btnBgAttHighIntInfoNormal, other.btnBgAttHighIntInfoNormal, t)!,
btnBgAttMediumIntInfoActive: Color.lerp(btnBgAttMediumIntInfoActive, other.btnBgAttMediumIntInfoActive, t)!,
btnBgAttMediumIntInfoNormal: Color.lerp(btnBgAttMediumIntInfoNormal, other.btnBgAttMediumIntInfoNormal, t)!,
btnLabelAttHigh: Color.lerp(btnLabelAttHigh, other.btnLabelAttHigh, t)!,
btnLabelAttLowIntDanger: Color.lerp(btnLabelAttLowIntDanger, other.btnLabelAttLowIntDanger, t)!,
btnLabelAttMediumIntDanger: Color.lerp(btnLabelAttMediumIntDanger, other.btnLabelAttMediumIntDanger, t)!,
btnLabelAttMediumIntInfo: Color.lerp(btnLabelAttMediumIntInfo, other.btnLabelAttMediumIntInfo, t)!,
btnShadowAttMed: Color.lerp(btnShadowAttMed, other.btnShadowAttMed, t)!,
composeBoxBg: Color.lerp(composeBoxBg, other.composeBoxBg, t)!,
contextMenuCancelText: Color.lerp(contextMenuCancelText, other.contextMenuCancelText, t)!,
contextMenuItemBg: Color.lerp(contextMenuItemBg, other.contextMenuItemBg, t)!,
Expand Down
Loading