- From: CSS Meeting Bot via GitHub <sysbot+gh@w3.org>
- Date: Tue, 06 Apr 2021 17:03:04 +0000
- To: public-css-archive@w3.org
The CSS Working Group just discussed `[css-values-4] Add lvh+lvw values`, and agreed to the following: * `RESOLVED: Work on the dynamic unit.` <details><summary>The full IRC log of that discussion</summary> <fantasai> topic: [css-values-4] Add lvh+lvw values<br> <fantasai> github: https://github.com/w3c/csswg-drafts/issues/6113<br> <fantasai> github: https://github.com/w3c/csswg-drafts/issues/4329<br> <TabAtkins> fantasai: 6113 is a specific part of 4329<br> <TabAtkins> fantasai: 4329 is a long discussiona bout how people are really, *really*, REALLY unhappy how on mobile the viewport units don't represent the size of hte viewport<br> <TabAtkins> fantasai: And the reason is that there are parts of the browser chrome that appear and disappear dynamically as you scroll<br> <astearns> +1 to unhappiness<br> <TabAtkins> fantasai: so a bunch of UAs decided that the viewport units are the size of the viewport with all the dynamic chrome hidden, so a 100vh element will be partially obscured if anything *is* showing<br> <TabAtkins> fantasai: Which is fine for some content, and really obnoxious for others (like toolbars or headers that have to be attached at the top/bottom of the screen)<br> <TabAtkins> fantasai: So we've had a dsicussion about what values authors need to get the useful behavior.<br> <TabAtkins> fantasai: Three thrings.<br> <TabAtkins> fantasai: One is size of the viewport with chrome present, so they can size into that minimal space and never get things overlapped.<br> <TabAtkins> fantasai: Another is the size without the chrome, so they can size to the maximal space and get all the space filled when it's all hidden.<br> <TabAtkins> fantasai: And finally, the dynamically-changing current size with whatever chrome is currently showing, so it'll perfectly fit the viewport in all cases.<br> <TabAtkins> fantasai: Currently we are only providing #2, and that's a problem.<br> <TabAtkins> iank_: I thought vhc did change?<br> <TabAtkins> fantasai: No it's static, and also it's not defined yet because we haven't decided on a name.<br> <smfr> q+<br> <TabAtkins> fantasai: So the issue I tagged here is someone saying "how about we introduce lvh/lvw/etc" and that would represent the dynamic size of the viewport.<br> <cbiesinger> what does lv stand for?<br> <TabAtkins> fantasai: And then vhc would be the with-chrome static size, and existing vh would be without-chrome static size.<br> <emilio> q+<br> <astearns> cbiesinger: Layout Viewport height<br> <cbiesinger> ah<br> <TabAtkins> fantasai: In A Coruna we discussed this, and a suggestion was becuase of the way viewport units compute, and to emphasize the dynamic-ness, possibly exposing the dynamic version as an env() value instead of a unit.<br> <TabAtkins> fantasai: And that might be less scary to implement.<br> <TabAtkins> fantasai: Viewport units have to be computed whenever the window changes; making it happen on scroll would be worse.<br> <TabAtkins> smfr: Drill down on "dynamic change"<br> <astearns> s/Coruna/Coruña/<br> <TabAtkins> smfr: in iOS, when the chrome animates in, I don't think we'd animate the unit; we'd snap it.<br> <TabAtkins> smfr: We ahve a notion of the UI being in an unstable state.<br> <TabAtkins> smfr: Like, position:fixed is magic that happens in the compositor.<br> <astearns> ack smfr<br> <TabAtkins> smfr: Web content doesn't see the dynamic change, it sees things flip, and the dynamic stuff happens behind the scenes.<br> <TabAtkins> fantasai: I think that's fine, as long as the layout is correct when UI is stable.<br> <TabAtkins> fantasai: If the in-between state is optimized in ways that's probably fine.<br> <TabAtkins> fantasai: Someday animating it might be nice and performant, but for now it's fine. From the author's POV, it's the end states that matter.<br> <TabAtkins> TabAtkins: I agree, it seems like the end states are th emost important here<br> <astearns> ack emilio<br> <TabAtkins> emilio: Same about end states, and that Firefox Mobile uses similar concepts. Webview only knows about the final states, not intermediate.<br> <TabAtkins> emilio: Also this sounds like fullscreen, a bit. We ahve display-mode MQ which tells you some stuff about the browser screen.<br> <TabAtkins> emilio: Would having these units and an MQ to switch between them work?<br> <chrishtr> q+<br> <TabAtkins> TabAtkins: Don't understand what you mean.<br> <TabAtkins> emilio: Instead of having a dynamic unit, just using the vh or vhc based on an MQ.<br> <TabAtkins> fantasai: I do want to keep it open for the future to be animated.<br> <TabAtkins> TabAtkins: I think this supports that, if we don't do dynamic *at all* right now.<br> <TabAtkins> fantasai: The computed-ness is still problematic<br> <TabAtkins> emilio: Computed value will *definitely* change with env(), they get substituted at var() time<br> <plinss> q+<br> <TabAtkins> fantasai: Or we have a unit that is used-value time. I don't think people want computed behavior here.<br> <TabAtkins> emilio: We ahve this already with %s on a fixpos<br> <TabAtkins> iank_: Yeah, or % height on <html>, they dynamically change based on chrome showing. So this concept is already exposed.<br> <fantasai> s/want/need/<br> <TabAtkins> iank_: So given the existence of height:100% on these elements, I don't know if we should try to avoid exposing this by using the MQ path.<br> <TabAtkins> smfr: height:100% doesn't change on iOS<br> <TabAtkins> emilio: It does for fixpos on Gecko, but unsure about on <html><br> <TabAtkins> emilio: This is historically not the most interoperable bit of mobile engines...<br> <TabAtkins> emilio: Could we spend some effort figuring out what we want the existing primitives to do?<br> <TabAtkins> fantasai: Yes, but I think we also need to add the new ones.<br> <TabAtkins> fantasai: Either way we'll need these three kinds of values.<br> <iank_> here is a testpage https://bokand.github.io/demo/urlbarsize.html<br> <TabAtkins> fantasai: Deciding which one the height:100% maps to is probably a little more acceptable once we have the ability to change to another one, via new units.<br> <astearns> ack chrishtr<br> <TabAtkins> fantasai: But right now people are just very unhappy<br> <TabAtkins> chrishtr: Is you set vh units it's the full size, ignoring chrome.<br> <TabAtkins> chrishtr: If you use vhc it subtracts chrome, so if yous croll up you see a gap<br> <TabAtkins> chrishtr: lvh theoretically animates, but at least shows the right value on stable. If you don't animate it'll show a gap during unstable, and then snap.<br> <TabAtkins> fantasai: One thing you can do is make the animation always snap to the larger size.<br> <TabAtkins> fantasai: It doesn't have to be the size you started at, just one of them; using the larger size always sounds better.<br> <TabAtkins> chrishtr: So when you start the animation, snap to the larger thing, do a main-thread relayout and possible introduce jank to the browser's interaction model.<br> <TabAtkins> chrishtr: Layout could take time, there could be a rAF callback...<br> <TabAtkins> fantasai: Right, but I think making it pretty is secondary to making it usable at all.<br> <TabAtkins> chrishtr: I find the whole area pretty confusing for ideal UX.<br> <TabAtkins> fantasai: Example: I've got a coding tutorial page, with pretty pictures.<br> <TabAtkins> fantasai: Long scrolling page.<br> <TabAtkins> fantasai: The elements I want to be viewport height are pictures and code blocks, both scrollable.<br> <TabAtkins> fantasai: It's *really awkward* to have a scrollable block taller than the viewport.<br> <TabAtkins> fantasai: So for usability reasons I want the code block to be at most th eheight of the viewport.<br> <TabAtkins> fantasai: But I want the viewport size *with chrome factored in*, the smaller size, so it'll always show the scrollbar when needed.<br> <TabAtkins> fantasai: I don't want larger (overlaps) and I don't want dynamic; I just want to make sure the entire box is visible when it's on screen regardless, so I can access both scrollbars.<br> <TabAtkins> fantasai: But for the pictures, I want them to fill the viewport, but it's okay to ahve them be cropped a bit.<br> <TabAtkins> fantasai: So I want them to fill the whole viewport when the chrome is hiding. And it's okay if the chrome comes in and obscures a little of the top or bottom of the image.<br> <TabAtkins> fantasai: And similarly I don't want this to be dynamic, because it'll cause the page to shift around, I want stable.<br> <TabAtkins> fantasai: And then *thirdly* I could have a sidebar that wants to fill the height of the page, and it needs to be dynamic - using all the height that's available when chrome is hidden, but not cropping anything when chrome is showing.<br> <TabAtkins> fantasai: So we want all three behaviors; we want to make sure that dynamic behavior is *clear* and won't get reached for accidentally.<br> <TabAtkins> chrishtr: So the sidebar example wants the dynamic example, right?<br> <TabAtkins> fantasai: Yeah.<br> <fantasai> TabAtkins: Have problems with the toolbars currently, even with position:fixed. [gives example]<br> <TabAtkins> iank_: People do dialogs, for example, that want it to fill the viewport.<br> <TabAtkins> iank_: Ther'es a lot of examples.<br> <fantasai> s/viewport/viewport, but too deep in the DOM to use 100%/<br> <TabAtkins> chrishtr: So a fullscreen, or fullscreen minus margin, dialog, you probably want it to dynamically resize when you hide the omnibox?<br> <TabAtkins> iank_: Yes.<br> <astearns> ack plinss<br> <plinss> https://github.com/MicrosoftEdge/MSEdgeExplainers/blob/main/VirtualKeyboardAPI/explainer.md<br> <TabAtkins> plinss: Microsoft is doing some virtual keyboard API (link), wanna make sure we coordinate with this work<br> <TabAtkins> iank_: I think this is a slightly different usecase here<br> <TabAtkins> iank_: I don't think you want lvh to change on keyboard showing<br> <TabAtkins> fantasai: I agree with Ian, and think we discussed this in A Coruna as well and came to the same conclusion.<br> <TabAtkins> fantasai: Wanting keyboard to change your layout is something much less common you should opt yourself into.<br> <TabAtkins> iank_: Especially like adding an extra keyboard above the UI keyboard<br> <TabAtkins> astearns: If that's the case, I'd want to see noted the reasons why we're explicitly not doing that.<br> <TabAtkins> fantasai: Sure, we can add a note for that.<br> <TabAtkins> smfr: I think there's a diff on Android vs iOS for whether keyboard is an overlay vs viewport shrink<br> <TabAtkins> smfr: There's also the visual viewport API that tells you when it changes due to the keyboard<br> <TabAtkins> dlibby: There's three modes - keyboard resizes the layout viewport (Android), keyboard resizes the visual viewport (Windows/iOS), and virtual keyboard is third (nothing page-observable happens besides some events firing)<br> <astearns> ack dbaron<br> <TabAtkins> dbaron: One thing to be careful about with dynamic unit is that there's an assumption that users will make some scroll gesture, and the toolbar will appear or disappear, and the unit will resize things, and somehow there's magic that'll happen that puts things in the positions they're expected to be in.<br> <TabAtkins> dbaron: I think we shoudl be careful not to assume that magic will just work, because it's actually quite hard. When things resize, things move.<br> <TabAtkins> chrishtr: You mean hard for the dev to make sure things move in predictable ways?<br> <smfr> q+<br> <TabAtkins> dbaron: Possibly both. I think there will be unexpected results where they use the dynamic unit, get the right size, but not the right position.<br> <TabAtkins> florian: Right, and I don't think in general you can't guarantee anyway because too much of the layout has changed.<br> <TabAtkins> chrishtr: What do you mean by position of content?<br> <TabAtkins> florian: If your page is just a linear stream of content, and you're just using the unit for height, you can expect the position of th eresized bit to not change.<br> <TabAtkins> florian: But if you use the unit in width, and this changes how many columns fit, etc, there might be arbitrary layout changes on the page that make it undefined what it means to be "same position"<br> <fantasai> TabAtkins: 2 items, both 100lvh<br> <fantasai> TabAtkins: both change, and bottom one ...<br> <fantasai> TabAtkins: causes unexpected position changes<br> <TabAtkins> smfr: Scroll anchoring can probably help ther<br> <TabAtkins> TabAtkins: Agree<br> <TabAtkins> smfr: If this is a unit that changes value, we need an event to inform you of the change. Possibly this is the resize event; if so, we need to sync this unit changing with th eresize event firing<br> <TabAtkins> fantasai: Does resize change due to this?<br> <TabAtkins> smfr: We do fire the resize event....<br> <TabAtkins> smfr: It fires on inner height, which does match this I think<br> <dlibby> visualViewport resize does fire already, I believe<br> <TabAtkins> smfr: There's several things that need to be accounted for.<br> <TabAtkins> iank_: I posted a testcase a bit ago, that should show in a particular browser when things change based on user gesture<br> <TabAtkins> iank_: I think resize event should fire when innerHeight changes<br> <TabAtkins> florian: Could this get us into loops, if the content using the viewport units gets smaller than the viewport height?<br> <chrishtr> q+<br> <smfr> q-<br> <TabAtkins> fantasai: We can get into the details for that after we decide whether to do this or not at all ^_^<br> <TabAtkins> fantasai: So we need to decide whether to add these functionalities or not<br> <TabAtkins> fantasai: If so, add them as units, as env(), as something else?<br> <TabAtkins> astearns: Are we lacking a second stable resolution?<br> <TabAtkins> fantasai: We have a resolution, just not a name.<br> <TabAtkins> fantasai: We don't ahve a resolution for the dynamic version.<br> <TabAtkins> fantasai: We should have consistency between them.<br> <TabAtkins> fantasai: The proposal for dynamic uses a prefix; we might want the small static one use a prefix as well, so cvh rather than vhc<br> <astearns> ack chrishtr<br> <TabAtkins> chrishtr: Coming back to use-case for dynamic, a 100% height sidebar<br> <TabAtkins> chrishtr: Why can't that be position:fixed top/bottom:0?<br> <TabAtkins> iank_: It can, but often people will embed sidebars deep in other elements<br> <TabAtkins> iank_: It'll break their site layout if it's fixpos<br> <TabAtkins> iank_: There can be a fixpos containing block between it and the root<br> <TabAtkins> iank_: People will often develop for desktop-first sidebars, and handle mobile after, and their page structure is already set.<br> <TabAtkins> florian: Another is you might not necessarily want to be on top of things. Think app, not document, with toolbars around a content area - a grid with toolbars in side cells and an overflowable center area<br> <TabAtkins> florian: Entire UI is supposed to fit the screen at all times. If browser chrome pops in and out, the page's UI shifts, but it should all always be visible.<br> <TabAtkins> chrishtr: So this is a site that doesn't scroll?<br> <TabAtkins> florian: Yes, the overall site doesn't scroll, just the regions.<br> <TabAtkins> jensimmons: Right, fixpos would work *only* if people were already using abspos for their layout, not if they were using Grid/etc.<br> <TabAtkins> jensimmons: Could we do a thing where we have units to measure the constant states, and then use env() for the dynamic things?<br> <iank_> (the units aren't constant if you change the window size :)<br> <fantasai> TabAtkins: That was fantasai's suggestion, to make it really obvious that this is dynamic and different from the others, make sure you really want it<br> <TabAtkins> chrishtr: Could one of y'all share an example that would help explain it?<br> <TabAtkins> florian: [shows off a music editing app on his phone, with a toolbar on bottom with sheet music taking up the content area]<br> <TabAtkins> florian: I always want the toolbar to be showing regardless of chrome showing or not, and I don't want the toolbar to overlap the end of the sheet music if I scroll to the bottom.<br> <fremy> @florian: isn't that position:sticky?<br> <TabAtkins> florian: So probably the whole thing is a Grid, with a height of env(viewport-height) or something, and the toolbar's grid cell is fixed height, etc<br> <TabAtkins> fremy: Wouldn't stickypos do this, too?<br> <TabAtkins> dlibby: yeah, seems like stickypos too<br> <dlibby> s/dlibby/sanketj/<br> <TabAtkins> florian: My claim isn't that you couldn't do it with stickypos or whatever, but rather if you're thinking about your entire layout in Grid, which you might very well do because it's modern and convenient, you can express this all in Grid, but then if it fails on mobile because the viewport units don't work...<br> <astearns> q?<br> <TabAtkins> iank_: Important to think about is that web devs might start by developing with a grid, which is deep in the DOM; to size it to the viewport they need to size all the elements in the ancestor chain carefully. Using vh is a very simple and easy tool there. It's a big change in theinking to instead switch to fixpos or stickypos.<br> <TabAtkins> iank_: So If they started with this as a grid, and were sizing to `calc(100vh - 10px)` or whatever, it would be a big change.<br> <fantasai> TabAtkins: ...<br> <fantasai> TabAtkins: ...<br> <fantasai> TabAtkins: ...<br> <fantasai> TabAtkins: Want to make sure that we don't create cliffs<br> <TabAtkins> florian: And if you have stickypos, the toolbar needs to be a child of the scroller; in Grid it's a sibling of the scroller<br> <TabAtkins> jensimmons: The use-cases we've seen, attaching to the bottom, is *one* of *many* use-cases for viewport units; I've demo'd viewport unit a bunch on stage and have never even used this case.<br> <TabAtkins> jensimmons: And 100vh isn't the whole thing - 50vh items stacking is reasonable.<br> <TabAtkins> chrishtr: Returning to florian's app example, the root scroller doesn't have overflow, right?<br> <TabAtkins> florian: In a simplified version, yes, it's a flexbox that fills the viewport. The content area that contains the sheet music has overflow:auto.<br> <TabAtkins> chrishtr: So the interaction model they want is you can scroll the music, nothing else; you can always tap the play button.<br> <TabAtkins> florian: Yeah.<br> <TabAtkins> chrishtr: So in this case, when it first loads on chrome mobile, the url bar will show and then never disappear, since it only disappears if you scroll the root scroller.<br> <TabAtkins> cbiesinger: I think apps sometimes do a scrollTo(1px) to cause the URL bar to autohide<br> <TabAtkins> florian: That might explain why the Firefox behavior is so confusing/uncertain<br> <TabAtkins> chrishtr: Yeah there's bugs around that.<br> <TabAtkins> chrishtr: So the site could use vhc perhaps, but if they use the dynamic version they'll never see it adjust.<br> <TabAtkins> chrishtr: So I was concerned about this responding to user gestures hiding the bar - that needs to be smooth.<br> <fremy> totally agree with cbiesinger here. if you use a fixed grid over the screen, there is no way to auto-hide the chrome as a user, and the problem just doesn't exist in that case<br> <TabAtkins> chrishtr: But if it just needs to be resized to the correct new area, I can see the use for this dynamic unit<br> <TabAtkins> I know there's a proposal for a root scroller designation API floating around somewhere...<br> <TabAtkins> florian: So there's definitely a bunch of issues, but does it seem reasonable to take a resolution to pursue the dynamic thing, and possible scal eit back afterwards?<br> <fremy> @TabAtkins: Ah, yeah if such API existed, that could change things indeed<br> <TabAtkins> astearns: And are we shooting for a unit or env() ?<br> <TabAtkins> iank_: I'd err towards units, since that's waht webdevs are already used to.<br> <TabAtkins> iank_: I don't find the argument about dynamic changes compelling - the other units already dynamically change on window resize.<br> <TabAtkins> astearns: I think it's much more common for webdevs to resize the tab to check it responds properly, less experience with showing/hiding chrome on a mobile device.<br> <TabAtkins> fantasai: The nice thing about env() is that it's more obnoxious to use, so it's more of an explicit decision.<br> <TabAtkins> fantasai: On desktop it doesn't have this dynamic problem, only on mobile.<br> <TabAtkins> fantasai: So I feel like we'll get less people doing it by default.<br> <TabAtkins> astearns: People who teach CSS, do y'all have an opinion?<br> <TabAtkins> jensimmons: I could go both ways.<br> <florian> +1 to fantasai's obnoxious-on-purpose syntax argument<br> <TabAtkins> jensimmons: vh means the most pixels, vTBA means the least pixels, and both are fixed.<br> <TabAtkins> jensimmons: A third unit meaning a dynamic changeable value, that could be confusing to people.<br> <TabAtkins> jensimmons: But iank_'s point is compelling, where authors are already used to these as units.<br> <TabAtkins> jensimmons: But elika's point about this being an env() so people ahve to go look for it might make sense.<br> <TabAtkins> rachelandrew: My gut is a unit, same as the others, but I get how we don't want it to be the default people use, and making it look different can help there.<br> <TabAtkins> rachelandrew: So as a teacher, units are easier to teach, but I get the reasoning behind the other.<br> <TabAtkins> leaverou: I think it makes more sense as a unit, due to symmetry.<br> <TabAtkins> leaverou: Might be easier to read as an env() because it can have a longer name; unit names are cryptic.<br> <TabAtkins> leaverou: But env() names can get pretty long, especially in calculations.<br> <TabAtkins> iank_: There's already polyfills for this; they act like a unit setting a css variable.<br> <TabAtkins> That sounds like it's similar to env(), then?<br> <TabAtkins> astearns: I propose that we resolve to work on this dynamic unit. Get spec text together, with examples, and open separate issues for moving it to env(), what to do with animations, etc.<br> <TabAtkins> chrishtr: I think that sounds great, one more thing, I hope we can go forward with vhc as well.<br> <TabAtkins> astearns: We do have a resolution for that, we just haven't gotten it into the spec<br> <jensimmo_> I always find answering questions like this (unit vs environmental var) are easier to understand when I code some demos. Actually write out the code.<br> <TabAtkins> fantasai: And haven't decided on the name. And it might make sense to make sure these are consistent.<br> <TabAtkins> fantasai: Might all work best with a prefix.<br> <iank_> https://www.npmjs.com/package/postcss-viewport-height-correction is one example of subsitution via postcss<br> <TabAtkins> fantasai: I propose to use "s" as a prefix for the small, static size, and "d" as a prefix for dynamic one.<br> <TabAtkins> RESOLVED: Work on the dynamic unit.<br> <TabAtkins> chrishtr: Not only do we hear a lot that this is a problem, Chrome is def interested in exploring this and working with devs.<br> <TabAtkins> chrishtr: If anyone has partners who can work with us, or has more examples, please contact me.<br> </details> -- GitHub Notification of comment by css-meeting-bot Please view or discuss this issue at https://github.com/w3c/csswg-drafts/issues/4329#issuecomment-814281379 using your GitHub account -- Sent via github-notify-ml as configured in https://github.com/w3c/github-notify-ml-config
Received on Tuesday, 6 April 2021 17:03:08 UTC