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

feat(slider): Implement vertical orientation #11028

Open
wants to merge 15 commits into
base: dev
Choose a base branch
from

Conversation

josercarcamo
Copy link
Contributor

Related Issue: #5522

Summary

Implemented the vertical orientation for slider.

@github-actions github-actions bot added the enhancement Issues tied to a new feature or request. label Dec 11, 2024
josercarcamo and others added 3 commits December 11, 2024 10:35
@jcfranco jcfranco added the pr ready for visual snapshots Adding this label will run visual snapshot testing. label Dec 11, 2024
@benelan
Copy link
Member

benelan commented Dec 12, 2024

What happened to #9953? Closing a PR with a lot of context in favor of a new one makes reviewing more difficult because we don't know what's been addressed and what needs further changes/discussion.

Can you please try maintain a single branch/PR for an issue's changes instead of doing this branch-v1, branch-v2, etc. style of iteration? Thanks!

@benelan
Copy link
Member

benelan commented Dec 12, 2024

It looks like your v3 PR (#10652) is still open too, which one should we be reviewing?

Copy link
Contributor

This PR has been automatically marked as stale because it has not had recent activity. Please close your PR if it is no longer relevant. Thank you for your contributions.

@github-actions github-actions bot added the Stale Issues or pull requests that have not had recent activity. label Dec 26, 2024
@github-actions github-actions bot removed the Stale Issues or pull requests that have not had recent activity. label Jan 3, 2025
Copy link
Member

@jcfranco jcfranco left a comment

Choose a reason for hiding this comment

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

Thanks, @josercarcamo! A few changes are needed before we can merge this.

@@ -56,6 +56,10 @@
:host {
@apply block;

&:host([layout="vertical"]) {
Copy link
Member

Choose a reason for hiding this comment

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

The parent selector here is redundant. Let’s move it outside this block to simplify. Otherwise, it results in: :host:host([layout="vertical"]).

@@ -56,6 +56,10 @@
:host {
@apply block;

&:host([layout="vertical"]) {
transform: rotate(-90deg);
Copy link
Member

Choose a reason for hiding this comment

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

I'm noticing some layout issues. It seems the vertical layout isn’t respecting the explicitly set height on the component.

Screenshot 2025-01-06 at 10 34 32 AM
<body>
    <style>
      calcite-slider {
        border: 1px solid red;
        &[layout="horizontal"] {
          width: 100px;
        }
        &[layout="vertical"] {
          height: 100px;
        }
      }
    </style>
    <span>top</span>
    <calcite-slider
      layout="vertical"
      min-value="25"
      max-value="75"
    ></calcite-slider>
    <calcite-slider
      min-value="25"
      max-value="75"
    ></calcite-slider>
    <span>bottom</span>
  </body>

Copy link
Contributor Author

@josercarcamo josercarcamo Jan 14, 2025

Choose a reason for hiding this comment

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

Although the content box has been rotated via the rule:

:host([layout="vertical"]) { transform: rotate(-90deg); }

CSS interprets the properties as if the scroll bar had not been rotated, so "width" controls the height and "height" controls the width when vertical. Please advise if you know of a way to change the "orientation" of the properties as well. I'm thinking we set a watcher on "width" and "height" and set the properties correctly when vertical.

Copy link
Member

Choose a reason for hiding this comment

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

I'm not sure I follow the proposed approach, but let's discuss this further today.

@@ -138,6 +142,7 @@
content: "\2014";
display: inline-block;
inline-size: 1em;
margin-inline-start: 0.1875rem;
Copy link
Member

Choose a reason for hiding this comment

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

Nice! Is there a token we could reference instead of hardcoding it? We have the following ones available: --calcite-spacing-xxs (4px), --calcite-spacing-base (2px). cc @ashetland

<span
aria-hidden="true"
class={{
[thumbLabelClasses]: true,
Copy link
Member

Choose a reason for hiding this comment

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

Can you move the classes from the template string into the class object?

Copy link
Contributor Author

@josercarcamo josercarcamo Jan 14, 2025

Choose a reason for hiding this comment

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

Do you mean to move the classes here?

const thumbLabelClasses = ${CSS.handleLabel} ${ isMinThumb ? CSS.handleLabelMinValue : CSS.handleLabelValue };

To here?

const labels = isLabeled
? [
<span
aria-hidden="true"
class={{
[thumbLabelClasses]: true,
}}
>
{displayedValue}
,
<span ariaHidden="true" class={${thumbLabelClasses} ${CSS.static}}>
{displayedValue}
,
<span ariaHidden="true" class={${thumbLabelClasses} ${CSS.transformed}}>
{displayedValue}
,
]
: [];

Copy link
Member

Choose a reason for hiding this comment

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

I meant something like this:

const labelClasses = {
  [CSS.handleLabel]: true,
  [CSS.handleLabelMinValue]: isMinThumb
  [CSS.handleLabelValue]: isMinThumb
};

// ...

<span
  aria-hidden="true"
  class={labelClasses}
>
  {displayedValue}
</span>,
<span ariaHidden="true" class={{
  [CSS.static]: true,
  ...labelClasses
}}>
  {displayedValue}
</span>,
<span ariaHidden="true" class={{
 [CSS.transformed]: true, 
 ...labelClasses,
}}>
  {displayedValue}
</span>
```

@@ -320,6 +322,9 @@ export class Slider
/** The component's value. */
@property({ type: Number, reflect: true }) value: null | number | number[] = 0;

/** The orientation of the slider */
Copy link
Member

Choose a reason for hiding this comment

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

We avoid using the actual name of components in the doc. Can you change this to follow existing props (e.g., segmented-conttrol#layout)? cc @geospatialem @DitwanP

describe("mouse interaction", () => {
it("single handle: clicking the track changes value on mousedown, emits on mouseup", async () => {
const page = await newE2EPage({
html: `<calcite-slider snap style="width:${sliderWidthFor1To1PixelValueTrack}" layout=${layout}></calcite-slider>`,
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 using height, right?

Copy link
Contributor Author

Choose a reason for hiding this comment

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

No. "width = height" in vertical orientation.

Copy link
Member

Choose a reason for hiding this comment

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

This is confusing, possibly because the implementation is relying heavily on the transform. From a user-perspective, width and height should not change semantics based on an option.

Let's discuss this further today.

expect(await slider.getProperty("maxValue")).toBe(75);
it("single handle: clicking and dragging the track changes and emits the value", async () => {
const page = await newE2EPage({
html: `<calcite-slider snap style="width:${sliderWidthFor1To1PixelValueTrack}; margin-top: 100px" layout=${layout}></calcite-slider>`,
Copy link
Member

Choose a reason for hiding this comment

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

margin-top shouldn't be necessary here. This seems related to the layout issue mentioned earlier.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

It is for aesthetics.

Copy link
Member

Choose a reason for hiding this comment

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

E2E tests aren't visual by default, so aesthetics aren't a concern in this context.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

I guess I misspoke: a part of the vertical slider could be outside the bounds of the window and not be clickable, so it must be moved downward.


const [trackX, trackY] = await getElementXY(page, "calcite-slider", ".track");
expect(await slider.getProperty("value")).toBe(5);
expect(inputEvent).toHaveReceivedEventTimes(layout === "horizontal" ? 5 : 6);
Copy link
Member

@jcfranco jcfranco Jan 6, 2025

Choose a reason for hiding this comment

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

Could you look into this? Changing layout shouldn't affect event behavior.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

The problem is that the upper-left hand corner given by "getElementXY" for horizontal is at value 0 of the slider, but it is at 100 for vertical. This causes one extra event:

image

image

Copy link
Member

Choose a reason for hiding this comment

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

Oh, I see. In that case, the starting position for this test for the vertical case should be adjusted to start at the same value.


expect(isMaxThumbFocused).toBe(true);
await assertValuesUnchanged(5);
layout === "horizontal"
Copy link
Member

Choose a reason for hiding this comment

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

Can you DRY this up by storing the test x, y variables based on layout and passing them to click? Applies to similar test steps.

max="1"
min-value="0"
max-value="0"
layout={layout}`;
Copy link
Member

Choose a reason for hiding this comment

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

Shouldn't this be layout=${layout} to use the layout var from the describe.each block? This would eliminate the need to use replace below and keeps tests consistent.

@jcfranco jcfranco requested a review from ashetland January 6, 2025 20:24
@josercarcamo josercarcamo requested a review from benelan as a code owner January 15, 2025 23:14
@josercarcamo
Copy link
Contributor Author

After meeting with @jcfranco, we've decided to seek an alternative implementation and pause this issue at this time.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
enhancement Issues tied to a new feature or request. pr ready for visual snapshots Adding this label will run visual snapshot testing.
Projects
None yet
Development

Successfully merging this pull request may close these issues.

4 participants