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

support for Add Filter button #3671

Merged
merged 23 commits into from
Dec 18, 2023
Merged
Show file tree
Hide file tree
Changes from 18 commits
Commits
Show all changes
23 commits
Select commit Hold shift + click to select a range
c00f552
rework of footer to match new design and improve readability
briangregoryholmes Dec 8, 2023
8478425
adding filter button functionality, plus design tweaks to relevant menus
briangregoryholmes Dec 8, 2023
71acf1f
new icon
briangregoryholmes Dec 8, 2023
8f3b637
Merge branch 'filter' of github.com:rilldata/rill into filter
briangregoryholmes Dec 8, 2023
c68da3f
chip should not be removed when deselecting the only selected value
briangregoryholmes Dec 9, 2023
0bb7d85
removing limit from filter list query
briangregoryholmes Dec 9, 2023
8cc77d3
changed searchedValues to allValues, simplified display logic
briangregoryholmes Dec 9, 2023
f1399b0
updated tests to meet updated requirements
briangregoryholmes Dec 9, 2023
d2b0de5
prettier unused vars build fix
briangregoryholmes Dec 9, 2023
0c7384d
merge main and resolve conflicts
briangregoryholmes Dec 14, 2023
0f0290e
remove keep-alive event
briangregoryholmes Dec 14, 2023
bdac8f8
update let directive for named slot, add timeout to dashboard.spec
briangregoryholmes Dec 14, 2023
90d5085
remove unused function
briangregoryholmes Dec 14, 2023
c8af4eb
change to active logic when mounting chip component
briangregoryholmes Dec 14, 2023
94e0355
resolve name collision
briangregoryholmes Dec 14, 2023
b01f3c8
update page wait from timeout to selector
briangregoryholmes Dec 15, 2023
164a092
added specific method for adding a dimension name without a value
briangregoryholmes Dec 15, 2023
154da92
only unselected dimension names are shown in the add filter dropdown
briangregoryholmes Dec 15, 2023
ed91037
add back limit
briangregoryholmes Dec 15, 2023
8f7d720
toggleDimensionNameSelection is now actually a toggling function
briangregoryholmes Dec 15, 2023
1de1107
dispatch remove event rather than handle remove directly, plus bind t…
briangregoryholmes Dec 15, 2023
2d52fe9
use state manager action and prop cleanup
briangregoryholmes Dec 15, 2023
b5f794f
add back limit
briangregoryholmes Dec 15, 2023
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
21 changes: 17 additions & 4 deletions web-common/src/components/chip/removable-list-chip/Footer.svelte
Original file line number Diff line number Diff line change
@@ -1,5 +1,18 @@
<div
class="flex flex-row mt-auto items-center justify-between gap-x-2 px-2 py-1 sticky bottom-0 border-t bg-gray-50 dark:bg-gray-600 border-gray-200 dark:border-gray-500"
>
<footer>
<slot />
</div>
</footer>

<style lang="postcss">
footer {
@apply flex flex-row mt-auto items-center justify-end;
@apply bg-slate-100;
@apply border-t border-slate-300;
@apply bottom-0 sticky;
@apply gap-x-2 p-2 px-3.5;
}

footer:is(.dark) {
@apply bg-gray-800;
@apply border-gray-700;
}
</style>
Original file line number Diff line number Diff line change
Expand Up @@ -11,9 +11,9 @@ the name and then move the cursor to the right to cancel it.
existing elements in the lib as well as changing the type (include, exclude) and enabling list search. The implementation of these parts
are details left to the consumer of the component; this component should remain pure-ish (only internal state) if possible.
-->
<script lang="ts">
import { createEventDispatcher } from "svelte";
import { Writable, writable } from "svelte/store";

<script context="module" lang="ts">
import { createEventDispatcher, onMount } from "svelte";
import { fly } from "svelte/transition";
import WithTogglableFloatingElement from "../../floating-element/WithTogglableFloatingElement.svelte";
import Tooltip from "../../tooltip/Tooltip.svelte";
Expand All @@ -23,28 +23,39 @@ are details left to the consumer of the component; this component should remain
import { Chip } from "../index";
import RemovableListBody from "./RemovableListBody.svelte";
import RemovableListMenu from "./RemovableListMenu.svelte";
import { clearFilterForDimension } from "@rilldata/web-common/features/dashboards/actions";
import { getStateManagers } from "@rilldata/web-common/features/dashboards/state-managers/state-managers";
</script>

<script lang="ts">
export let name: string;
export let selectedValues: string[];
export let searchedValues: string[] | null;
export let allValues: string[] | null;

/** an optional type label that will appear in the tooltip */
export let typeLabel: string;
export let excludeMode;
export let excludeMode: boolean;
export let colors: ChipColors = defaultChipColors;
export let label: string | undefined = undefined;
export let dimensionName: string;

const initiallyActive = !selectedValues.length;

const dispatch = createEventDispatcher();

const excludeStore: Writable<boolean> = writable(excludeMode);
$: excludeStore.set(excludeMode);
const StateManagers = getStateManagers();

onMount(() => {
dispatch("mount");
});
</script>

<WithTogglableFloatingElement
let:toggleFloatingElement
let:active
distance={8}
alignment="start"
active={initiallyActive}
>
<Tooltip
location="bottom"
Expand All @@ -55,7 +66,10 @@ are details left to the consumer of the component; this component should remain
>
<Chip
removable
on:click={toggleFloatingElement}
on:click={() => {
toggleFloatingElement();
dispatch("click");
}}
on:remove={() => dispatch("remove")}
{active}
{...colors}
Expand Down Expand Up @@ -91,15 +105,27 @@ are details left to the consumer of the component; this component should remain
</div>
</Tooltip>
<RemovableListMenu
{excludeStore}
slot="floating-element"
let:toggleFloatingElement
on:escape={toggleFloatingElement}
on:click-outside={toggleFloatingElement}
slot="floating-element"
{excludeMode}
{allValues}
{selectedValues}
on:escape={() => {
if (!selectedValues.length) {
clearFilterForDimension(StateManagers, dimensionName, !excludeMode);
} else {
toggleFloatingElement();
}
}}
on:click-outside={() => {
if (!selectedValues.length) {
clearFilterForDimension(StateManagers, dimensionName, !excludeMode);
} else {
toggleFloatingElement();
}
}}
on:apply
on:search
on:toggle
{selectedValues}
{searchedValues}
/>
</WithTogglableFloatingElement>
Original file line number Diff line number Diff line change
@@ -1,28 +1,31 @@
import RemovableListMenu from "./RemovableListMenu.svelte";
import { describe, it, expect, vi } from "vitest";
import { render, waitFor, fireEvent, screen } from "@testing-library/svelte";
import { writable } from "svelte/store";

describe("RemovableListMenu", () => {
it("renders selected values by default", async () => {
it("does not render selected values if not in all values", async () => {
const { unmount } = render(RemovableListMenu, {
excludeStore: writable(false),
selectedValues: ["foo", "bar"],
searchedValues: null,
excludeMode: false,
selectedValues: ["x"],
allValues: ["foo", "bar"],
});

const foo = screen.getByText("foo");
const bar = screen.getByText("bar");
expect(foo).toBeDefined();
expect(bar).toBeDefined();

const x = screen.queryByText("x");
expect(x).toBeNull();

unmount();
});

it("renders selected values if search text is empty", async () => {
it("renders all values if search text is empty", async () => {
const { unmount } = render(RemovableListMenu, {
excludeStore: writable(false),
selectedValues: ["foo", "bar"],
searchedValues: ["x"],
excludeMode: false,
selectedValues: [],
allValues: ["foo", "bar"],
});

const foo = screen.getByText("foo");
Expand All @@ -34,9 +37,9 @@ describe("RemovableListMenu", () => {

it("renders search values if search text is populated", async () => {
const { unmount } = render(RemovableListMenu, {
excludeStore: writable(false),
excludeMode: false,
selectedValues: ["foo", "bar"],
searchedValues: ["x"],
allValues: ["x"],
});

const searchInput = screen.getByRole("textbox", { name: "Search list" });
Expand All @@ -51,35 +54,33 @@ describe("RemovableListMenu", () => {
});

it("should render switch based on exclude store", async () => {
const excludeStore = writable(false);
const { unmount } = render(RemovableListMenu, {
excludeStore,
const { unmount, component } = render(RemovableListMenu, {
excludeMode: false,
selectedValues: ["foo", "bar"],
searchedValues: ["x"],
allValues: ["x"],
});

const switchInput = screen.getByRole<HTMLInputElement>("switch");
expect(switchInput.checked).toBe(false);
const switchInput = screen.getByText("Exclude");
expect(switchInput).toBeDefined();

excludeStore.set(true);
await waitFor(() => {
expect(switchInput.checked).toBe(true);
});
await component.$set({ excludeMode: true });

const includeButton = screen.getByText("Include");
expect(includeButton).toBeDefined();

unmount();
});

it("should dispatch toggle, apply, and search events", async () => {
const excludeStore = writable(false);
const { unmount, component } = render(RemovableListMenu, {
excludeStore,
selectedValues: ["foo", "bar"],
searchedValues: ["x"],
excludeMode: false,
selectedValues: [],
allValues: ["foo", "bar"],
});

const toggleSpy = vi.fn();
component.$on("toggle", toggleSpy);
const switchInput = screen.getByRole<HTMLInputElement>("switch");
const switchInput = screen.getByText("Exclude");
await fireEvent.click(switchInput);
expect(toggleSpy).toHaveBeenCalledOnce();

Expand Down
Original file line number Diff line number Diff line change
@@ -1,17 +1,16 @@
<script lang="ts">
import { createEventDispatcher } from "svelte";
import type { Writable } from "svelte/store";
import { Switch } from "../../button";
import Cancel from "../../icons/Cancel.svelte";
import Check from "../../icons/Check.svelte";
import Spacer from "../../icons/Spacer.svelte";
import { Menu, MenuItem } from "../../menu";
import { Search } from "../../search";
import Footer from "./Footer.svelte";
import Button from "../../button/Button.svelte";

export let excludeStore: Writable<boolean>;
export let excludeMode: boolean;
export let selectedValues: string[];
export let searchedValues: string[] | null = [];
export let allValues: string[] | null = [];

let searchText = "";

Expand All @@ -21,33 +20,20 @@
dispatch("search", searchText);
}

function onToggleHandler() {
dispatch("toggle");
function toggleValue(value: string) {
dispatch("apply", value);
}

/** On instantiation, only take the exact current selectedValues, so that
* when the user unchecks a menu item, it still persists in the FilterMenu
* until the user closes.
*/
let candidateValues = [...selectedValues];
let valuesToDisplay = [...candidateValues];

// If searchedValues === null, search has not finished yet. So continue rendering the previous list
$: if (searchText && searchedValues !== null) {
valuesToDisplay = [...searchedValues];
} else if (!searchText) valuesToDisplay = [...candidateValues];

$: numSelectedNotInSearch = selectedValues.filter(
(v) => !valuesToDisplay.includes(v)
).length;

function toggleValue(value) {
dispatch("apply", value);
function toggleSelectAll() {
allValues?.forEach((value) => {
if (!allSelected && selectedValues.includes(value)) return;

if (!candidateValues.includes(value)) {
candidateValues = [...candidateValues, value];
}
toggleValue(value);
});
}

$: allSelected =
selectedValues?.length && allValues?.length === selectedValues.length;
</script>

<Menu
Expand All @@ -62,8 +48,7 @@
on:click-outside
>
<!-- the min-height is set to have about 3 entries in it -->

<div class="px-1 pb-1">
<div class="px-3 py-2">
<Search
bind:value={searchText}
on:input={onSearch}
Expand All @@ -74,8 +59,8 @@

<!-- apply a wrapped flex element to ensure proper bottom spacing between body and footer -->
<div class="flex flex-col flex-1 overflow-auto w-full pb-1">
{#if valuesToDisplay.length}
{#each valuesToDisplay as value}
{#if allValues?.length}
{#each allValues.sort() as value}
<MenuItem
icon
animateSelect={false}
Expand All @@ -85,17 +70,17 @@
}}
>
<svelte:fragment slot="icon">
{#if selectedValues.includes(value) && !$excludeStore}
{#if selectedValues.includes(value) && !excludeMode}
<Check size="20px" color="#15141A" />
{:else if selectedValues.includes(value) && $excludeStore}
{:else if selectedValues.includes(value) && excludeMode}
<Cancel size="20px" color="#15141A" />
{:else}
<Spacer size="20px" />
{/if}
</svelte:fragment>
<span
class:ui-copy-disabled={selectedValues.includes(value) &&
$excludeStore}
excludeMode}
>
{#if value?.length > 240}
{value.slice(0, 240)}...
Expand All @@ -110,17 +95,20 @@
{/if}
</div>
<Footer>
<span class="ui-copy">
<Switch on:click={() => onToggleHandler()} checked={$excludeStore}>
<Button type="text" on:click={toggleSelectAll}>
{#if allSelected}
Deselect all
{:else}
Select all
{/if}
</Button>

<Button type="secondary" on:click={() => dispatch("toggle")}>
{#if excludeMode}
Include
{:else}
Exclude
</Switch>
</span>
{#if numSelectedNotInSearch}
<div class="ui-label">
{numSelectedNotInSearch} other value{numSelectedNotInSearch > 1
? "s"
: ""} selected
</div>
{/if}
{/if}
</Button>
</Footer>
</Menu>
19 changes: 19 additions & 0 deletions web-common/src/components/icons/ChevronRight.svelte
Original file line number Diff line number Diff line change
@@ -0,0 +1,19 @@
<script>
export let size = "1em";
export let color = "currentColor";
</script>

<svg
width={size}
viewBox="0 0 24 24"
fill="none"
xmlns="http://www.w3.org/2000/svg"
>
<path
d="M9 18L15 12L9 6"
stroke={color}
stroke-width="2"
stroke-linecap="round"
stroke-linejoin="round"
/>
</svg>
Loading