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

Conversation

chrisbobbe
Copy link
Collaborator

@chrisbobbe chrisbobbe commented Mar 21, 2025

This PR implements some variants of the "Button" component in the "Zulip Web UI kit" in Figma: https://www.figma.com/design/msWyAJ8cnMHgOMPxi7BUvA/Zulip-Web-UI-kit?node-id=1-8&p=f&m=dev

The Figma uses this component for the "Cancel" and "Save" buttons in the edit-message compose box:

https://www.figma.com/design/1JTNtYo9memgW7vV6d0ygq/Zulip-Mobile?node-id=3988-38200&m=dev

Light Dark
image image

Not all values of the "intent" and "attention" params are implemented here; just the "info" intent with "medium" and "high" attention. In the edit-message compose box banner, both buttons have the "info" intent; the cancel button has "medium" attention and the save button has "high" attention.

Here are screenshots of the buttons, made with these commits and also #1430 (plus the _EditMessageBanner class in that PR's description):

Cancel button ("info" intent, "medium" attention)

Light; normal state Light; pressed state Dark; normal state Dark; pressed state
image image image image

Save button ("info" intent, "high" attention)

Light; normal state Light; pressed state Dark; normal state Dark; pressed state
image image image image

Related: #126

@chrisbobbe chrisbobbe added the maintainer review PR ready for review by Zulip maintainers label Mar 21, 2025
@chrisbobbe chrisbobbe requested a review from PIG208 March 21, 2025 22:58
Comment on lines +116 to +150
enum ZulipWebUiKitButtonIntent {
// neutral,
// warning,
// danger,
info,
// success,
// brand,
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.

Copy link
Member

@PIG208 PIG208 left a comment

Choose a reason for hiding this comment

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

This is neat. Thanks for working on this! Left some comments.

required this.btnLabelAttMediumIntInfo,
required this.btnLabelAttMediumIntDanger,
Copy link
Member

Choose a reason for hiding this comment

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

nit: order

Suggested change
required this.btnLabelAttMediumIntInfo,
required this.btnLabelAttMediumIntDanger,
required this.btnLabelAttMediumIntDanger,
required this.btnLabelAttMediumIntInfo,

Copy link
Collaborator Author

Choose a reason for hiding this comment

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

🫣

).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.


return AnimatedScaleOnTap(
scaleEnd: 0.96,
duration: Duration(milliseconds: 50),
Copy link
Member

Choose a reason for hiding this comment

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

Looks like duration in Figma is 100ms:

image

Copy link
Collaborator Author

Choose a reason for hiding this comment

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

Thanks for spotting that! I didn't see it, so instead found an animation in the web app with Chrome dev tools; probably ought to have made a comment about that :)

final ZulipWebUiKitButtonAttention attention;
final ZulipWebUiKitButtonIntent intent;
final String label;
final VoidCallback? onPressed;
Copy link
Member

Choose a reason for hiding this comment

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

Looks like the Figma design does not offer style for disabled buttons. Widgets like TextButton are disabled when onPressed is null, but we don't have special styling to indicate that (and AnimatedScaleOnTap still works).

@@ -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.22),
Copy link
Member

Choose a reason for hiding this comment

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

This should be 0.12 as well.

Copy link
Collaborator Author

Choose a reason for hiding this comment

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

Thanks for the catch!

@chrisbobbe
Copy link
Collaborator Author

Thanks for the review! Revision pushed.

@PIG208
Copy link
Member

PIG208 commented Apr 1, 2025

Thanks for the update! This LGTM.

@PIG208 PIG208 assigned gnprice and unassigned PIG208 Apr 1, 2025
@PIG208 PIG208 added integration review Added by maintainers when PR may be ready for integration and removed maintainer review PR ready for review by Zulip maintainers labels Apr 1, 2025
@PIG208 PIG208 requested a review from gnprice April 1, 2025 14:54
Copy link
Member

@gnprice gnprice left a comment

Choose a reason for hiding this comment

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

Thanks! This structure looks good; a couple of comments below.

Comment on lines 55 to 61
return TextStyle(
color: _labelColor(designVariables),
fontSize: 17,
height: 1.20,
letterSpacing: proportionalLetterSpacing(context,
0.006, baseFontSize: 17),
).merge(weightVariableTextStyle(context, wght: 600));
Copy link
Member

Choose a reason for hiding this comment

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

Where do these details come from?

Looking at one of the buttons in Figma:
https://www.figma.com/design/msWyAJ8cnMHgOMPxi7BUvA/Zulip-Web-UI-kit?node-id=1-2960&m=dev
I see:

overflow: hidden;
color: var(--btn-label-att_high, #FFF);
text-align: center;
text-overflow: ellipsis;
font-family: "Source Sans 3";
font-size: 16px;
font-style: normal;
font-weight: 500;
line-height: 20px; /* 125% */
letter-spacing: 0.096px;

Copy link
Collaborator Author

Choose a reason for hiding this comment

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

Odd, I took these details from the buttons as they appear in the compose-box frame:

https://www.figma.com/design/1JTNtYo9memgW7vV6d0ygq/Zulip-Mobile?node-id=3988-38201&m=dev

overflow: hidden;
color: var(--btn-label-att_high, #FFF);
text-align: center;
text-overflow: ellipsis;
font-family: "Source Sans 3";
font-size: 17px;
font-style: normal;
font-weight: 600;
line-height: 120%; /* 20.4px */
letter-spacing: 0.102px;

Copy link
Member

Choose a reason for hiding this comment

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

Huh. Yeah, I guess leave a comment, then. We can leave this code specialized to what that bit of the mobile design calls for, unless we find that other designs call for something different in which case we can generalize as needed.

borderRadius: BorderRadius.circular(4)),
splashFactory: NoSplash.splashFactory,
tapTargetSize: MaterialTapTargetSize.shrinkWrap,
minimumSize: Size(kMinInteractiveDimension, 28),
Copy link
Member

Choose a reason for hiding this comment

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

Hmm, 28px is awfully small for a touch target.

The Figma design says:

People might say that 44px is recommended minimum for touch screens but we assume that this is for web app.

Here we are indeed talking about a touch screen, though 🙂 So being just 28px high, rather than 44px or 48px, seems likely to be frustrating.

It'd be fine if the visual button is on the short side, as long as the tap target is a good height. So in the edit-message design, it calls for the buttons to be 28px high but then the whole banner is 40px. So making the banner only slightly taller than the design would make room for a 44px tap target, which could still enclose a 28px visual button.

These Material buttons have a facility for padding out the tap target beyond the visual button — see the call site of _InputPadding in button_style_button.dart (in _ButtonStyleState.build). Can we use that to get that effect?

@chrisbobbe
Copy link
Collaborator Author

Thanks for the review! Revision pushed.

Copy link
Member

@gnprice gnprice left a comment

Choose a reason for hiding this comment

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

Thanks for the revision! The implementation all looks good. Small comments from reading through the tests.

final textScaleFactorVariants = ValueVariant(Set.of(kTextScaleFactors));
testWidgets('button and touch-target heights', (tester) async {
addTearDown(testBinding.reset);
tester.platformDispatcher.textScaleFactorTestValue = textScaleFactorVariants.currentValue!;
Copy link
Member

Choose a reason for hiding this comment

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

Huh, neat mechanism.

Comment on lines +19 to +20
final textScaleFactorVariants = ValueVariant(Set.of(kTextScaleFactors));
testWidgets('button and touch-target heights', (tester) async {
Copy link
Member

Choose a reason for hiding this comment

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

nit:

Suggested change
final textScaleFactorVariants = ValueVariant(Set.of(kTextScaleFactors));
testWidgets('button and touch-target heights', (tester) async {
final textScaleFactorVariants = ValueVariant(Set.of(kTextScaleFactors));
testWidgets('button and touch-target heights', (tester) async {

Comment on lines +36 to +40
final renderObject = element.renderObject as RenderBox;
final width = renderObject.size.width;
final buttonTopLeft = tester.getTopLeft(buttonFinder);
final buttonCenter = tester.getCenter(buttonFinder);
check(element).size.isNotNull().height.equals(44); // includes outer padding
Copy link
Member

Choose a reason for hiding this comment

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

Is there a difference between element.size and (element.renderObject as RenderBox).size? I notice this code is using one of those to find the height and the other for the width.

Comment on lines +38 to +42
final buttonTopLeft = tester.getTopLeft(buttonFinder);
final buttonCenter = tester.getCenter(buttonFinder);
check(element).size.isNotNull().height.equals(44); // includes outer padding

// Outer padding responds to taps, not just the painted part.
Copy link
Member

Choose a reason for hiding this comment

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

nit: these two are used only here in this next stanza, so move them there

Suggested change
final buttonTopLeft = tester.getTopLeft(buttonFinder);
final buttonCenter = tester.getCenter(buttonFinder);
check(element).size.isNotNull().height.equals(44); // includes outer padding
// Outer padding responds to taps, not just the painted part.
check(element).size.isNotNull().height.equals(44); // includes outer padding
// Outer padding responds to taps, not just the painted part.
final buttonTopLeft = tester.getTopLeft(buttonFinder);
final buttonCenter = tester.getCenter(buttonFinder);

Copy link
Member

Choose a reason for hiding this comment

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

Or maybe what I really want here is to move renderObject and width down to where they're used.

Comment on lines +44 to +45
for (double y = 0; y < 44; y++) {
await tester.tapAt(Offset(buttonCenter.dx, y + buttonTopLeft.dy));
Copy link
Member

Choose a reason for hiding this comment

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

nit: Perhaps this would be clearer in this form?

Suggested change
for (double y = 0; y < 44; y++) {
await tester.tapAt(Offset(buttonCenter.dx, y + buttonTopLeft.dy));
for (double y = -22; y < 22; y++) {
await tester.tapAt(buttonCenter + Offset(0, y));

Comment on lines +48 to +50
check(numTapsHandled).equals(numTaps);

final textScaler = TextScaler.linear(textScaleFactorVariants.currentValue!)
Copy link
Member

Choose a reason for hiding this comment

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

I think this test would break up pretty cleanly into two tests at this spot: one checks the response to taps, and one checks the painting. That would probably make them both a little easier to read.

Comment on lines +94 to +99
// 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;
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.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
integration review Added by maintainers when PR may be ready for integration
Projects
None yet
Development

Successfully merging this pull request may close these issues.

3 participants