Skip to content

WJSoftware/wjfe-dataview

Repository files navigation

@wjfe/dataview

Svelte v5 table component suitable for examination of extensive tabular data.

Overview

The data view component renders a table with functionality suitable for close examination of the presented data. It provides conveniences like pinnable columns and row highlighting on hover, useful to users when following data with their eyes.

Demo Website

The component tries to be as unopinionated as possible in terms of styling and tries to provide as little styling as possible. Certain features, however, impose some appearance requirements. For example, pinning columns in the grid requires opaque background colors or else the data from other columns will be seen through the pinned columns when the data view is scrolled horizontally.

To theme the table to your needs or otherwise make it play well with your styling framework of choice (Bootstrap, Tailwind CSS, Bulma, etc.), you may style it in accordance to what is shown in the Theming the Data View section.

Quickstart

Install the package:

npm i @wjfe/dataview

Now import the data view component and use it:

import { WjDataView } from '@wjfe/dataview';

The only two required properties are columns and data. The former defines the columns in the data view; the latter provides the data that shows in each column. By default, the key property of each column is treated as the key to retrieve data from the data row, but this can be overridden by providing get functions.

Each column must have the key and the text properties. Any other property is optional.

Each data object (the rows) must have an id property of type number or string, and the wjdv property, which is an object that holds operational data for each row. The defineData() function is a helper function that helps you fulfill these 2 requirements. The id values are meant to be unique amongst the data set.

IMPORTANT: Always bind columns and data. The data view control will mutate properties inside this data, so the best practice is to bind.

<script lang="ts">
    import { WjDataView, defineData } from '@wjfe/dataview';
    import { type MyDataModel } from 'path/to/my-model-types.js';

    type MyColumn = WjDvColumn<MyDataModel>;
    const columns = $state<MyColumn[]>([
        {
            key: 'id',
            text: 'ID'
        },
        {
            key: 'tagName',
            text: 'Tag'
        }
    ]);
    // Obtain the data somehow.  This could be part of the results of the universal or server load() SvelteKit 
    // function, or could be obtained in non-SvelteKit projects with a fetch() call.
    const dataFromApi = getDataSomehow();
    // Now ensure the data fulfills the component requirements using "defineData":
    let data = $state(defineData<MyDataModel>(dataFromApi));
    // The model type here ------^^^^^^^^^^^ may not be needed if you have properly typed the `getDataSomehow()` 
    // function or the `dataFromApi` variable.
</script>

...
<WjDataView bind:columns bind:data>
    <!-- snippets go here -->
</WjDataView>

This example would render the data view with two columns, whose captions will read ID and Tag. The data shown in each column will be extracted from the MyDataModel.id and MyDataModel.tagName properties of each of the data objects in the data array.

The defineData() function mutates the data for performance reasons. It works the items directly as opposed to cloning them. Also, the same array is returned for the same performance reasons.

Theming the Data View

As stated in the overview, the data view's appearance can be customized. The component has been styled using CSS variables with some defaults. If you wish, you may redefine the variables at will (the variables are listed at the end of the section), but it is best to use the theme component WjDataViewTheme that provides a friendlier way than directly defining the CSS variables.

The WjDataViewTheme Component

This component is a mere convenience to setting up themes for data view components. It works by defining the CSS variables in a <div> element with its CSS display attribute set to contents that wraps the target WjDataView component.

TIP: The WjDataViewTheme component doesn't have to be the immediate parent of a WjDataView. It can be placed higher in the hierarchy to, for example, cover more than one WjDataView component.

The theme component has a single theme property of type Theme:

export type ComponentColor = {
    backgroundColor?: string;
    opacity?: number;
    color?: string;
};

export type ResizerColor = {
    backgroundColor?: string;
    borderColor?: string;
};

export type BorderDefinition = {
    width?: string;
    style?: 'dashed' | 'dotted' | 'double' | 'groove' | 'inset' | 'outset' | 'ridge' | 'solid' | 'unset';
    color?: string;
};

export type Theme = {
    table?: ComponentColor;
    stripes?: ComponentColor;
    rowTracking?: ComponentColor;
    rowSelection?: ComponentColor;
    pinnedColumnsDivider?: BorderDefinition,
    resizer?: {
        width?: string;
        overlay?: {
            opacity?: number;
            item?: ResizerColor;
            positiveDelta?: ResizerColor;
            negativeDelta?: ResizerColor;
        }
    };
    gridBorders?: BorderDefinition;
};

While the amount of properties is large, each one of them are optional. Simply set the properties that you wish to customize. The properties that aren't set will take the default value documented in the table further down.

For example, Bootstrap consumers might want to ensure that the data view always uses the body's background color. In this case, we could create the following theme in a dataViewThemes.ts (potential) file:

import { type Theme } from '@wjfe/dataview';

export const bootstrapTheme: Theme = {
    table: {
        backgroundColor: 'var(--bs-body-bg-rgb)'
    },
    stripes: {
        backgroundColor: 'var(--bs-emphasis-color-rgb)'
    },
    rowTracking: {
        backgroundColor: 'var(--bs-primary-rgb)',
        opacity: 0.2
    },
    rowSelection: {
        backgroundColor: 'var(--bs-row-selection-bg-color-rgb)',
    }
};

This is the actual modification set in the demonstration page. The first three variables are provided by Bootstrap, while the last one (--bs-row-selection-bg-color-rgb) is defined inside the demo project.

As seen, one can take advantage of CSS variables to define values. Bootstrap provides light and dark modes, and these variables have different definitions depending on the mode, making the data view's theme immediately responsive to mode selection changes.

This is not perfect, however, because Bootstrap doesn't have -rgb variables for every color, so not everything goes as smoothly. Create CSS variables that adjust to the color mode to perfect the theme. For example, the last one has been defined as:

.theme-def {
    --bs-row-selection-bg-color-rgb: 221, 235, 255;
    
    :global([data-bs-theme="dark"]) & {
        --bs-row-selection-bg-color-rgb: 21, 35, 55;
    }
}

With this technique, we can create fully responsive themes for the data view component.

IMPORTANT: All background colors are composed using the provided color and an opacity value. This is why the color must be specified in RGB format, or with a CSS variable that defines it in RGB format. Formats like #rrggbb simply won't work.

Use the WjDataViewTheme component as a wrapper for any WjDataView components that you may have. This wrapper doesn't have to be the immediate parent, so put it wherever is best according to your needs.

<script lang="ts">
    import { bootstrapTheme } from '../dataViewThemes.js';
</script>

<WjDataViewTheme theme={bootstrapTheme}>
    <WjDataView ...>
        <!-- Snippets go here -->
    </WjDataView>
</WjDataViewTheme>

The complete list of CSS variables that can be set for the data view component are:

CSS Variable Light Default Dark Default Description
--wjdv-bg-color-rgb 255, 255, 255 0, 0, 0 Data view's background color.
--wjdv-bg-opacity 1 1 Background's opacity.
--wjdv-fg-color inherit inherit Foreground (or text) color. Usually this one doesn't need to be set.
--wjdv-striping-bg-color-rgb 0, 0, 0 255, 255, 255 Striping background color. Set the opacity as well.
--wjdv-striping-bg-opacity 0.04 0.07 Striping background color's opacity.
--wjdv-striping-fg-color inherit inherit Foreground (or text) color for striped rows.
--wjdv-rowtracking-bg-color-rgb 0, 0, 0 255, 255, 255 Background color for row tracking.
--wjdv-rowtracking-bg-opacity 0.07 0.15 Opacity for row tracking. Usually set higher than the striping one or the effect doesn't look very good.
--wjdv-rowtracking-fg-color inherit inherit Foreground (or text) color for tracked rows.
--wjdv-sticky-divider-width 0.1em 0.1em Width of the border that divides pinned columns from unpinned ones.
--wjdv-sticky-divider-style solid solid Style of the border that divides pinned columns from unpinned ones.
--wjdv-sticky-divider-color darkgray lightgray Color of the border that divides pinned columns from unpinned ones.
--wjdv-resizer-width 0.4em 0.4em Column resizer's width.
--wjdv-resizer-overlay-opacity 0.7 0.7 Opacity of the entire resizer overlay.
--wjdv-resizer-overlay-bg-color lightblue #0578ea Background color of the overlay section that represents the original column's size.
--wjdv-resizer-overlay-border-color blue #13aeff Border color of the overlay section that represents the original column's size.
--wjdv-resizer-deltapos-bg-color lightgreen lightgreen Background color of the overlay section that represents the column's size increase.
--wjdv-resizer-deltapos-border-color green green Border color of the overlay section that represents the column's size increase.
--wjdv-resizer-deltaneg-bg-color pink pink Background color of the overlay section that represents the column's size reduction.
--wjdv-resizer-deltaneg-border-color red red Border color of the overlay section that represents the column's size reduction.
--wjdv-selected-bg-color-rgb 227, 240, 254 15, 25, 74 Background color of rows that have been selected.
--wjdv-selected-bgopacity 1 1 Background opacity of rows that have been selected.
--wjdv-selected-fg-color inherit inherit Foreground color of rows that have been selected.
--wjdv-grid-line-width 0.01em 0.01em Width of the table's border lines.
--wjdv-grid-line-style solid solid Style used in the table's border lines.
--wjdv-grid-line-color currentColor currentColor Color used in the table's border lines.
--wjdv-i-filler-color #ddd #ddd Color used to render the pattern in filler columns.
--wjdv-filler-pattern-size 5px 5px Size of the pattern in filler columns.

Styling the Header Row

The header row, as row, cannot be styled. What can be styled is the individual header cells. This is due to the nature of the structure around the pinnable columns feature.

The most common thing to do here is to add some background coloring to the header cells, so they appear different to the data cells. This in itself is complicated because pinned cells must be guaranteed a fully opaque background color or else the non-pinned columns will be seen through them when scrolling horizontally.

But when there's will there's a way: I learned from the Bootstrap team that one can simulate a background color using box-shadow (this technique is revealed in Intellisense on the headerClass property of the component):

<script lang="ts">
    import { bootstrapTheme } from '../dataViewThemes.js';
</script>

<WjDataViewTheme theme={bootstrapTheme} headerClass="header-background">
    <WjDataView ...>
        <!-- Snippets go here -->
    </WjDataView>
</WjDataViewTheme>

<style>
    :global(.header-background) {
        box-shadow: 0 9999px 9999px rgba(0, 0, 0, 0.07) inset;
    }
</style>

This will shade the header cells darker. On a white background, this will make the cells look gray; on a black background the color selection must be a light one (not black as in the example) so the cells are shaded lighter and can stand out.

If you must set background-color, you'll have to fight the component using !important:

<style>
    :global(.header-background) {
        background-color: #ddd !important;
    }
</style>

Remember: If you set the background color, it must be opaque so pinned columns work as expected.

Cross Column Synchronization

Since v0.13.0

It is possible to synchronize any property of a column with the same property in the column object of another dataview instance, up to the point where they can be visually aligned. This is a desirable feature in master-child scenarios because users can have related columns perfectly aligned between master and child tables. See this detailed README to learn how to make use of this one-in-a-kind feature.

Reference

Props

Property Type Default Value Bindable Description
columns WjDvColumn<TCol, TRow>[] (none) Yes Defines the columns the data view component will create.
data WjDvRow<TRow>[] (none) Yes The data that is shown by the data view component.
get (row: TRow, key: string) => any (function) Function that retrieves a column's value using the row and provided key for columns that don't provide one.
defaultWidth number 10 The width for colums that don't specify its own width, in em's.
rowTracking boolean true Turns the row tracking feature on and off.
rowSelectionBg boolean true Turns the row-highlighting-on-selection feature on and off.
striped boolean true Turns the striping of rows on and off.
pinnedDivider boolean true Turns the divider between pinned and unpinned columns on and off.
class string undefined Additional CSS classes that are applied to the data view's viewport (the top-level element).
headerClass string undefined Adds additional CSS classes to the individual header cells.
controlColumn ControlColumn<TRow, TCol> undefined Yes Specifies the shape of the control column, which an extra column that is always the first pinned column.
noViewport boolean false Allows the exclusion of the component's viewport.
propSpreadingTarget PropSpreadingTarget root Establishes the target for property spreading.
fillerPattern 'checkered' | 'diagstriped' | 'triangles' 'diagstriped' Sets the desired pattern for the filler columns that may appear when columns are cross-synchronized.

Snippets

Name Parameters Description
headerCell ColumnContext<TRow, TCol> Renders header cells' content. The snippet is passed the column definition.
dataCell DataCellContext<TRow, TCol> Renders data cells' content. The snippet is passed the column definition and the data object for the row being rendered.
rowExpansion RowContext<TRow> Renders arbitrary content immediately below the data cells of the row. It is only rendered when WjDvRow<TRow>.wjdv.expanded is true.
controlHeaderCell (none) Renders the contents of the control column's header cell.
controlDataCell RowContext<TRow> Renders the contents of the control column's data cells.
caption (none) Renders the content of the data view's caption.

Events

None.

Roadmap

  • Scrollable viewport
  • Striped look
  • Row highlighting effect on hover
  • Column alignment
  • Text wrap control
  • Hideable columns
  • Pinnable columns
  • Customizable appearance
  • Theme component
  • headerCell snippet
  • dataCell snippet
  • Resizable columns
  • Expansible rows
  • Row selection
  • Control column
  • Header styling
  • Cross Column Synchronization
  • Make cell/row/column padding themeable
  • dataRow snippet (complex)