import type {LayoutAlign, SignalRef} from 'vega'; import {BinParams} from '../bin.js'; import {ChannelDef, Field, FieldName, TypedFieldDef} from '../channeldef.js'; import {ExprRef} from '../expr.js'; import {Header} from '../header.js'; import {EncodingSortField, SortArray, SortOrder} from '../sort.js'; import {StandardType} from '../type.js'; import {BaseSpec, GenericCompositionLayoutWithColumns, ResolveMixins} from './base.js'; import {GenericLayerSpec, NormalizedLayerSpec} from './layer.js'; import {GenericUnitSpec, NormalizedUnitSpec} from './unit.js'; import {hasProperty} from '../util.js'; export interface FacetFieldDef extends TypedFieldDef { /** * An object defining properties of a facet's header. */ header?: Header | null; // Note: `"sort"` for facet field def is different from encoding field def as it does not support `SortByEncoding` /** * Sort order for the encoded field. * * For continuous fields (quantitative or temporal), `sort` can be either `"ascending"` or `"descending"`. * * For discrete fields, `sort` can be one of the following: * - `"ascending"` or `"descending"` -- for sorting by the values' natural order in JavaScript. * - [A sort field definition](https://vega.github.io/vega-lite/docs/sort.html#sort-field) for sorting by another field. * - [An array specifying the field values in preferred order](https://vega.github.io/vega-lite/docs/sort.html#sort-array). In this case, the sort order will obey the values in the array, followed by any unspecified values in their original order. For discrete time field, values in the sort array can be [date-time definition objects](types#datetime). In addition, for time units `"month"` and `"day"`, the values can be the month or day names (case insensitive) or their 3-letter initials (e.g., `"Mon"`, `"Tue"`). * - `null` indicating no sort. * * __Default value:__ `"ascending"` * * __Note:__ `null` is not supported for `row` and `column`. */ sort?: SortArray | SortOrder | EncodingSortField | null; } export type FacetEncodingFieldDef< F extends Field, ES extends ExprRef | SignalRef = ExprRef | SignalRef, > = FacetFieldDef & GenericCompositionLayoutWithColumns; export interface RowColumnEncodingFieldDef extends FacetFieldDef { // Manually declarae this separated from GenericCompositionLayout as we don't support RowCol object in RowColumnEncodingFieldDef /** * The alignment to apply to row/column facet's subplot. * The supported string values are `"all"`, `"each"`, and `"none"`. * * - For `"none"`, a flow layout will be used, in which adjacent subviews are simply placed one after the other. * - For `"each"`, subviews will be aligned into a clean grid structure, but each row or column may be of variable size. * - For `"all"`, subviews will be aligned and each row or column will be sized identically based on the maximum observed size. String values for this property will be applied to both grid rows and columns. * * __Default value:__ `"all"`. */ align?: LayoutAlign; /** * Boolean flag indicating if facet's subviews should be centered relative to their respective rows or columns. * * __Default value:__ `false` */ center?: boolean; /** * The spacing in pixels between facet's sub-views. * * __Default value__: Depends on `"spacing"` property of [the view composition configuration](https://vega.github.io/vega-lite/docs/config.html#view-config) (`20` by default) */ spacing?: number; } export interface FacetMapping< F extends Field, FD extends FacetFieldDef = FacetFieldDef, > { /** * A field definition for the vertical facet of trellis plots. */ row?: FD; /** * A field definition for the horizontal facet of trellis plots. */ column?: FD; } export function isFacetMapping( f: FacetFieldDef | FacetMapping, ): f is FacetMapping { return hasProperty(f, 'row') || hasProperty(f, 'column'); } /** * Facet mapping for encoding macro */ export interface EncodingFacetMapping extends FacetMapping> { /** * A field definition for the (flexible) facet of trellis plots. * * If either `row` or `column` is specified, this channel will be ignored. */ facet?: FacetEncodingFieldDef; } export function isFacetFieldDef(channelDef: ChannelDef): channelDef is FacetFieldDef { return hasProperty(channelDef, 'header'); } /** * Base interface for a facet specification. */ export interface GenericFacetSpec, L extends GenericLayerSpec, F extends Field> extends BaseSpec, GenericCompositionLayoutWithColumns, ResolveMixins { /** * Definition for how to facet the data. One of: * 1) [a field definition for faceting the plot by one field](https://vega.github.io/vega-lite/docs/facet.html#field-def) * 2) [An object that maps `row` and `column` channels to their field definitions](https://vega.github.io/vega-lite/docs/facet.html#mapping) */ facet: FacetFieldDef | FacetMapping; /** * A specification of the view that gets faceted. */ spec: L | U; // TODO: replace this with GenericSpec once we support all cases; } /** * A facet specification without any shortcut / expansion syntax */ export type NormalizedFacetSpec = GenericFacetSpec; export function isFacetSpec(spec: BaseSpec): spec is GenericFacetSpec { return hasProperty(spec, 'facet'); }