-
Notifications
You must be signed in to change notification settings - Fork 305
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
base: main
Are you sure you want to change the base?
Conversation
enum ZulipWebUiKitButtonIntent { | ||
// neutral, | ||
// warning, | ||
// danger, | ||
info, | ||
// success, | ||
// brand, |
There was a problem hiding this comment.
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.
There was a problem hiding this 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.
lib/widgets/theme.dart
Outdated
required this.btnLabelAttMediumIntInfo, | ||
required this.btnLabelAttMediumIntDanger, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
nit: order
required this.btnLabelAttMediumIntInfo, | |
required this.btnLabelAttMediumIntDanger, | |
required this.btnLabelAttMediumIntDanger, | |
required this.btnLabelAttMediumIntInfo, |
There was a problem hiding this comment.
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), |
There was a problem hiding this comment.
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.
There was a problem hiding this comment.
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.
There was a problem hiding this comment.
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.
lib/widgets/button.dart
Outdated
|
||
return AnimatedScaleOnTap( | ||
scaleEnd: 0.96, | ||
duration: Duration(milliseconds: 50), |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
There was a problem hiding this comment.
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 :)
lib/widgets/button.dart
Outdated
final ZulipWebUiKitButtonAttention attention; | ||
final ZulipWebUiKitButtonIntent intent; | ||
final String label; | ||
final VoidCallback? onPressed; |
There was a problem hiding this comment.
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).
lib/widgets/theme.dart
Outdated
@@ -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), |
There was a problem hiding this comment.
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.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thanks for the catch!
771493d
to
091c102
Compare
Thanks for the review! Revision pushed. |
Thanks for the update! This LGTM. |
There was a problem hiding this 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.
lib/widgets/button.dart
Outdated
return TextStyle( | ||
color: _labelColor(designVariables), | ||
fontSize: 17, | ||
height: 1.20, | ||
letterSpacing: proportionalLetterSpacing(context, | ||
0.006, baseFontSize: 17), | ||
).merge(weightVariableTextStyle(context, wght: 600)); |
There was a problem hiding this comment.
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;
There was a problem hiding this comment.
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;
There was a problem hiding this comment.
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.
lib/widgets/button.dart
Outdated
borderRadius: BorderRadius.circular(4)), | ||
splashFactory: NoSplash.splashFactory, | ||
tapTargetSize: MaterialTapTargetSize.shrinkWrap, | ||
minimumSize: Size(kMinInteractiveDimension, 28), |
There was a problem hiding this comment.
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?
091c102
to
5c59015
Compare
5c59015
to
c88b217
Compare
Thanks for the review! Revision pushed. |
There was a problem hiding this 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!; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Huh, neat mechanism.
final textScaleFactorVariants = ValueVariant(Set.of(kTextScaleFactors)); | ||
testWidgets('button and touch-target heights', (tester) async { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
nit:
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 { |
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 |
There was a problem hiding this comment.
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.
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. |
There was a problem hiding this comment.
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
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); |
There was a problem hiding this comment.
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.
for (double y = 0; y < 44; y++) { | ||
await tester.tapAt(Offset(buttonCenter.dx, y + buttonTopLeft.dy)); |
There was a problem hiding this comment.
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?
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)); |
check(numTapsHandled).equals(numTaps); | ||
|
||
final textScaler = TextScaler.linear(textScaleFactorVariants.currentValue!) |
There was a problem hiding this comment.
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.
// 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; |
There was a problem hiding this comment.
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.
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
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)
Save button ("info" intent, "high" attention)
Related: #126