Module

x/vega_lite/src/scale.ts

A concise grammar of interactive graphics, built on Vega.
Latest
File
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856
import { RangeEnum, ScaleBins, ScaleInterpolateEnum, ScaleInterpolateParams, SignalRef, TimeInterval, TimeIntervalStep} from 'vega';import {isString, toSet} from 'vega-util';import * as CHANNEL from './channel';import {Channel, isColorChannel} from './channel';import {DateTime} from './datetime';import {ExprRef} from './expr';import * as log from './log';import {SelectionExtent} from './selection';import {NOMINAL, ORDINAL, QUANTITATIVE, TEMPORAL, Type} from './type';import {contains, Flag, keys} from './util';
export const ScaleType = { // Continuous - Quantitative LINEAR: 'linear', LOG: 'log', POW: 'pow', SQRT: 'sqrt', SYMLOG: 'symlog',
IDENTITY: 'identity', SEQUENTIAL: 'sequential',
// Continuous - Time TIME: 'time', UTC: 'utc',
// Discretizing scales QUANTILE: 'quantile', QUANTIZE: 'quantize', THRESHOLD: 'threshold', BIN_ORDINAL: 'bin-ordinal',
// Discrete scales ORDINAL: 'ordinal', POINT: 'point', BAND: 'band'} as const;
type ValueOf<T> = T[keyof T];export type ScaleType = ValueOf<typeof ScaleType>;
/** * Index for scale categories -- only scale of the same categories can be merged together. * Current implementation is trying to be conservative and avoid merging scale type that might not work together */export const SCALE_CATEGORY_INDEX: Record<ScaleType, ScaleType | 'numeric' | 'ordinal-position' | 'discretizing'> = { linear: 'numeric', log: 'numeric', pow: 'numeric', sqrt: 'numeric', symlog: 'numeric', identity: 'numeric', sequential: 'numeric', time: 'time', utc: 'time', ordinal: 'ordinal', 'bin-ordinal': 'bin-ordinal', // TODO: should bin-ordinal support merging with other point: 'ordinal-position', band: 'ordinal-position', quantile: 'discretizing', quantize: 'discretizing', threshold: 'discretizing'};
export const SCALE_TYPES = keys(SCALE_CATEGORY_INDEX) as ScaleType[];
/** * Whether the two given scale types can be merged together. */export function scaleCompatible(scaleType1: ScaleType, scaleType2: ScaleType) { const scaleCategory1 = SCALE_CATEGORY_INDEX[scaleType1]; const scaleCategory2 = SCALE_CATEGORY_INDEX[scaleType2]; return ( scaleCategory1 === scaleCategory2 || (scaleCategory1 === 'ordinal-position' && scaleCategory2 === 'time') || (scaleCategory2 === 'ordinal-position' && scaleCategory1 === 'time') );}
/** * Index for scale precedence -- high score = higher priority for merging. */const SCALE_PRECEDENCE_INDEX: Record<ScaleType, number> = { // numeric linear: 0, log: 1, pow: 1, sqrt: 1, symlog: 1, identity: 1, sequential: 1, // time time: 0, utc: 0, // ordinal-position -- these have higher precedence than continuous scales as they support more types of data point: 10, band: 11, // band has higher precedence as it is better for interaction // non grouped types ordinal: 0, 'bin-ordinal': 0, quantile: 0, quantize: 0, threshold: 0};
/** * Return scale categories -- only scale of the same categories can be merged together. */export function scaleTypePrecedence(scaleType: ScaleType): number { return SCALE_PRECEDENCE_INDEX[scaleType];}
export const CONTINUOUS_TO_CONTINUOUS_SCALES: ScaleType[] = ['linear', 'log', 'pow', 'sqrt', 'symlog', 'time', 'utc'];const CONTINUOUS_TO_CONTINUOUS_INDEX = toSet(CONTINUOUS_TO_CONTINUOUS_SCALES);
export const QUANTITATIVE_SCALES: ScaleType[] = ['linear', 'log', 'pow', 'sqrt', 'symlog'];
const QUANTITATIVE_SCALES_INDEX = toSet(QUANTITATIVE_SCALES);
export function isQuantitative(type: ScaleType): type is 'linear' | 'log' | 'pow' | 'sqrt' | 'symlog' { return type in QUANTITATIVE_SCALES_INDEX;}
export const CONTINUOUS_TO_DISCRETE_SCALES: ScaleType[] = ['quantile', 'quantize', 'threshold'];const CONTINUOUS_TO_DISCRETE_INDEX = toSet(CONTINUOUS_TO_DISCRETE_SCALES);
export const CONTINUOUS_DOMAIN_SCALES: ScaleType[] = CONTINUOUS_TO_CONTINUOUS_SCALES.concat([ 'quantile', 'quantize', 'threshold', 'sequential', 'identity']);const CONTINUOUS_DOMAIN_INDEX = toSet(CONTINUOUS_DOMAIN_SCALES);
export const DISCRETE_DOMAIN_SCALES: ScaleType[] = ['ordinal', 'bin-ordinal', 'point', 'band'];const DISCRETE_DOMAIN_INDEX = toSet(DISCRETE_DOMAIN_SCALES);
export const TIME_SCALE_TYPES: ScaleType[] = ['time', 'utc'];
export function hasDiscreteDomain(type: ScaleType): type is 'ordinal' | 'bin-ordinal' | 'point' | 'band' { return type in DISCRETE_DOMAIN_INDEX;}
export function hasContinuousDomain( type: ScaleType): type is 'linear' | 'log' | 'pow' | 'sqrt' | 'symlog' | 'time' | 'utc' | 'quantile' | 'quantize' | 'threshold' { return type in CONTINUOUS_DOMAIN_INDEX;}
export function isContinuousToContinuous( type: ScaleType): type is 'linear' | 'log' | 'pow' | 'sqrt' | 'symlog' | 'time' | 'utc' { return type in CONTINUOUS_TO_CONTINUOUS_INDEX;}
export function isContinuousToDiscrete(type: ScaleType): type is 'quantile' | 'quantize' | 'threshold' { return type in CONTINUOUS_TO_DISCRETE_INDEX;}
export interface ScaleConfig<ES extends ExprRef | SignalRef> { /** * If true, rounds numeric output values to integers. * This can be helpful for snapping to the pixel grid. * (Only available for `x`, `y`, and `size` scales.) */ round?: boolean | ES;
/** * If true, values that exceed the data domain are clamped to either the minimum or maximum range value */ clamp?: boolean | ES;
/** * Default inner padding for `x` and `y` band-ordinal scales. * * __Default value:__ * - `barBandPaddingInner` for bar marks (`0.1` by default) * - `rectBandPaddingInner` for rect and other marks (`0` by default) * * @minimum 0 * @maximum 1 */ bandPaddingInner?: number | ES;
/** * Default outer padding for `x` and `y` band-ordinal scales. * * __Default value:__ `paddingInner/2` (which makes _width/height = number of unique values * step_) * * @minimum 0 * @maximum 1 */ bandPaddingOuter?: number | ES;
/** * Default inner padding for `x` and `y` band-ordinal scales of `"bar"` marks. * * __Default value:__ `0.1` * * @minimum 0 * @maximum 1 */ barBandPaddingInner?: number | ES;
/** * Default inner padding for `x` and `y` band-ordinal scales of `"rect"` marks. * * __Default value:__ `0` * * @minimum 0 * @maximum 1 */ rectBandPaddingInner?: number | ES;
/** * Default padding for continuous scales. * * __Default:__ `5` for continuous x-scale of a vertical bar and continuous y-scale of a horizontal bar.; `0` otherwise. * * @minimum 0 */ continuousPadding?: number | ES;
/** * Default outer padding for `x` and `y` point-ordinal scales. * * __Default value:__ `0.5` (which makes _width/height = number of unique values * step_) * * @minimum 0 * @maximum 1 */ pointPadding?: number | ES;
/** * Use the source data range before aggregation as scale domain instead of aggregated data for aggregate axis. * * This is equivalent to setting `domain` to `"unaggregate"` for aggregated _quantitative_ fields by default. * * This property only works with aggregate functions that produce values within the raw data domain (`"mean"`, `"average"`, `"median"`, `"q1"`, `"q3"`, `"min"`, `"max"`). For other aggregations that produce values outside of the raw data domain (e.g. `"count"`, `"sum"`), this property is ignored. * * __Default value:__ `false` */ useUnaggregatedDomain?: boolean;
// nice should depends on type (quantitative or temporal), so // let's not make a config.
// Configs for Range
/** * The default max value for mapping quantitative fields to bar's size/bandSize. * * If undefined (default), we will use the axis's size (width or height) - 1. * @minimum 0 */ maxBandSize?: number;
/** * The default min value for mapping quantitative fields to bar and tick's size/bandSize scale with zero=false. * * __Default value:__ `2` * * @minimum 0 */ minBandSize?: number;
/** * The default max value for mapping quantitative fields to text's size/fontSize. * * __Default value:__ `40` * * @minimum 0 */ maxFontSize?: number;
/** * The default min value for mapping quantitative fields to tick's size/fontSize scale with zero=false * * __Default value:__ `8` * * @minimum 0 */ minFontSize?: number;
/** * Default minimum opacity for mapping a field to opacity. * * __Default value:__ `0.3` * * @minimum 0 * @maximum 1 */ minOpacity?: number;
/** * Default max opacity for mapping a field to opacity. * * __Default value:__ `0.8` * * @minimum 0 * @maximum 1 */ maxOpacity?: number;
/** * Default minimum value for point size scale with zero=false. * * __Default value:__ `9` * * @minimum 0 */ minSize?: number;
/** * Default max value for point size scale. * @minimum 0 */ maxSize?: number;
/** * Default minimum strokeWidth for the scale of strokeWidth for rule and line marks and of size for trail marks with zero=false. * * __Default value:__ `1` * * @minimum 0 */ minStrokeWidth?: number;
/** * Default max strokeWidth for the scale of strokeWidth for rule and line marks and of size for trail marks. * * __Default value:__ `4` * * @minimum 0 */ maxStrokeWidth?: number;
/** * Default range cardinality for [`quantile`](https://vega.github.io/vega-lite/docs/scale.html#quantile) scale. * * __Default value:__ `4` * * @minimum 0 */ quantileCount?: number;
/** * Default range cardinality for [`quantize`](https://vega.github.io/vega-lite/docs/scale.html#quantize) scale. * * __Default value:__ `4` * * @minimum 0 */ quantizeCount?: number;
/** * Reverse x-scale by default (useful for right-to-left charts). */ xReverse?: boolean | ES;}
export const defaultScaleConfig: ScaleConfig<SignalRef> = { pointPadding: 0.5,
barBandPaddingInner: 0.1, rectBandPaddingInner: 0,
minBandSize: 2,
minFontSize: 8, maxFontSize: 40,
minOpacity: 0.3, maxOpacity: 0.8,
// FIXME: revise if these *can* become ratios of width/height step minSize: 9, // Point size is area. For square point, 9 = 3 pixel ^ 2, not too small!
minStrokeWidth: 1, maxStrokeWidth: 4, quantileCount: 4, quantizeCount: 4};
export interface SchemeParams { /** * A color scheme name for ordinal scales (e.g., `"category10"` or `"blues"`). * * For the full list of supported schemes, please refer to the [Vega Scheme](https://vega.github.io/vega/docs/schemes/#reference) reference. */ name: string | SignalRef;
/** * The extent of the color range to use. For example `[0.2, 1]` will rescale the color scheme such that color values in the range _[0, 0.2)_ are excluded from the scheme. */ extent?: (number | SignalRef)[] | SignalRef;
/** * The number of colors to use in the scheme. This can be useful for scale types such as `"quantize"`, which use the length of the scale range to determine the number of discrete bins for the scale domain. */ count?: number | SignalRef;}
export type Domain = | (null | string | number | boolean | DateTime | SignalRef)[] | 'unaggregated' | SelectionExtent | SignalRef | DomainUnionWith;
export type Scheme = string | SchemeParams;
export function isExtendedScheme(scheme: Scheme | SignalRef): scheme is SchemeParams { return !isString(scheme) && !!scheme['name'];}
export function isSelectionDomain(domain: Domain): domain is SelectionExtent { return domain?.['selection'];}
export interface DomainUnionWith { /** * Customized domain values to be union with the field's values. * * 1) `domain` for _quantitative_ fields can take one of the following forms: * * - a two-element array with minimum and maximum values. * - an array with more than two entries, for [Piecewise quantitative scales](https://vega.github.io/vega-lite/docs/scale.html#piecewise). (Alternatively, the `domainMid` property can be set for a diverging scale.) * - a string value `"unaggregated"`, if the input field is aggregated, to indicate that the domain should include the raw data values prior to the aggregation. * * 2) `domain` for _temporal_ fields can be a two-element array minimum and maximum values, in the form of either timestamps or the [DateTime definition objects](https://vega.github.io/vega-lite/docs/types.html#datetime). * * 3) `domain` for _ordinal_ and _nominal_ fields can be an array that lists valid input values. */ unionWith: number[] | string[] | boolean[] | DateTime[];}
export function isDomainUnionWith(domain: Domain): domain is DomainUnionWith { return domain && domain['unionWith'];}
export interface Scale<ES extends ExprRef | SignalRef = ExprRef | SignalRef> { /** * The type of scale. Vega-Lite supports the following categories of scale types: * * 1) [**Continuous Scales**](https://vega.github.io/vega-lite/docs/scale.html#continuous) -- mapping continuous domains to continuous output ranges ([`"linear"`](https://vega.github.io/vega-lite/docs/scale.html#linear), [`"pow"`](https://vega.github.io/vega-lite/docs/scale.html#pow), [`"sqrt"`](https://vega.github.io/vega-lite/docs/scale.html#sqrt), [`"symlog"`](https://vega.github.io/vega-lite/docs/scale.html#symlog), [`"log"`](https://vega.github.io/vega-lite/docs/scale.html#log), [`"time"`](https://vega.github.io/vega-lite/docs/scale.html#time), [`"utc"`](https://vega.github.io/vega-lite/docs/scale.html#utc). * * 2) [**Discrete Scales**](https://vega.github.io/vega-lite/docs/scale.html#discrete) -- mapping discrete domains to discrete ([`"ordinal"`](https://vega.github.io/vega-lite/docs/scale.html#ordinal)) or continuous ([`"band"`](https://vega.github.io/vega-lite/docs/scale.html#band) and [`"point"`](https://vega.github.io/vega-lite/docs/scale.html#point)) output ranges. * * 3) [**Discretizing Scales**](https://vega.github.io/vega-lite/docs/scale.html#discretizing) -- mapping continuous domains to discrete output ranges [`"bin-ordinal"`](https://vega.github.io/vega-lite/docs/scale.html#bin-ordinal), [`"quantile"`](https://vega.github.io/vega-lite/docs/scale.html#quantile), [`"quantize"`](https://vega.github.io/vega-lite/docs/scale.html#quantize) and [`"threshold"`](https://vega.github.io/vega-lite/docs/scale.html#threshold). * * __Default value:__ please see the [scale type table](https://vega.github.io/vega-lite/docs/scale.html#type). */ type?: ScaleType;
/** * Customized domain values in the form of constant values or dynamic values driven by a selection. * * 1) Constant `domain` for _quantitative_ fields can take one of the following forms: * * - A two-element array with minimum and maximum values. To create a diverging scale, this two-element array can be combined with the `domainMid` property. * - An array with more than two entries, for [Piecewise quantitative scales](https://vega.github.io/vega-lite/docs/scale.html#piecewise). * - A string value `"unaggregated"`, if the input field is aggregated, to indicate that the domain should include the raw data values prior to the aggregation. * * 2) Constant `domain` for _temporal_ fields can be a two-element array with minimum and maximum values, in the form of either timestamps or the [DateTime definition objects](https://vega.github.io/vega-lite/docs/types.html#datetime). * * 3) Constant `domain` for _ordinal_ and _nominal_ fields can be an array that lists valid input values. * * 4) To combine (union) specified constant domain with the field's values, `domain` can be an object with a `unionWith` property that specify constant domain to be combined. For example, `domain: {unionWith: [0, 100]}` for a quantitative scale means that the scale domain always includes `[0, 100]`, but will include other values in the fields beyond `[0, 100]`. * * 5) Domain can also takes an object defining a field or encoding of a selection that [interactively determines](https://vega.github.io/vega-lite/docs/selection.html#scale-domains) the scale domain. */ domain?: | (null | string | number | boolean | DateTime | ES)[] | 'unaggregated' | SelectionExtent | DomainUnionWith | ES;
/** * Inserts a single mid-point value into a two-element domain. The mid-point value must lie between the domain minimum and maximum values. This property can be useful for setting a midpoint for [diverging color scales](https://vega.github.io/vega-lite/docs/scale.html#piecewise). The domainMid property is only intended for use with scales supporting continuous, piecewise domains. */ domainMid?: number | ES;
/** * Sets the maximum value in the scale domain, overriding the `domain` property. This property is only intended for use with scales having continuous domains. */ domainMax?: number | DateTime | ES;
/** * Sets the minimum value in the scale domain, overriding the domain property. This property is only intended for use with scales having continuous domains. */ domainMin?: number | DateTime | ES;
/** * If true, reverses the order of the scale range. * __Default value:__ `false`. */ reverse?: boolean | ES;
/** * The range of the scale. One of: * * - A string indicating a [pre-defined named scale range](https://vega.github.io/vega-lite/docs/scale.html#range-config) (e.g., example, `"symbol"`, or `"diverging"`). * * - For [continuous scales](https://vega.github.io/vega-lite/docs/scale.html#continuous), two-element array indicating minimum and maximum values, or an array with more than two entries for specifying a [piecewise scale](https://vega.github.io/vega-lite/docs/scale.html#piecewise). * * - For [discrete](https://vega.github.io/vega-lite/docs/scale.html#discrete) and [discretizing](https://vega.github.io/vega-lite/docs/scale.html#discretizing) scales, an array of desired output values or an object with a `field` property representing the range values. For example, if a field `color` contains CSS color names, we can set `range` to `{field: "color"}`. * * __Notes:__ * * 1) For color scales you can also specify a color [`scheme`](https://vega.github.io/vega-lite/docs/scale.html#scheme) instead of `range`. * * 2) Any directly specified `range` for `x` and `y` channels will be ignored. Range can be customized via the view's corresponding [size](https://vega.github.io/vega-lite/docs/size.html) (`width` and `height`). */ range?: RangeEnum | (number | string | number[] | ES)[] | {field: string};
/** * Sets the maximum value in the scale range, overriding the `range` property or the default range. This property is only intended for use with scales having continuous ranges. */ rangeMax?: number | string | ES;
/** * Sets the minimum value in the scale range, overriding the `range` property or the default range. This property is only intended for use with scales having continuous ranges. */ rangeMin?: number | string | ES;
// ordinal
/** * A string indicating a color [scheme](https://vega.github.io/vega-lite/docs/scale.html#scheme) name (e.g., `"category10"` or `"blues"`) or a [scheme parameter object](https://vega.github.io/vega-lite/docs/scale.html#scheme-params). * * Discrete color schemes may be used with [discrete](https://vega.github.io/vega-lite/docs/scale.html#discrete) or [discretizing](https://vega.github.io/vega-lite/docs/scale.html#discretizing) scales. Continuous color schemes are intended for use with color scales. * * For the full list of supported schemes, please refer to the [Vega Scheme](https://vega.github.io/vega/docs/schemes/#reference) reference. */ scheme?: string | SchemeParams | ES;
/** * The alignment of the steps within the scale range. * * This value must lie in the range `[0,1]`. A value of `0.5` indicates that the steps should be centered within the range. A value of `0` or `1` may be used to shift the bands to one side, say to position them adjacent to an axis. * * __Default value:__ `0.5` */ align?: number | ES;
/** * Bin boundaries can be provided to scales as either an explicit array of bin boundaries or as a bin specification object. The legal values are: * - An [array](../types/#Array) literal of bin boundary values. For example, `[0, 5, 10, 15, 20]`. The array must include both starting and ending boundaries. The previous example uses five values to indicate a total of four bin intervals: [0-5), [5-10), [10-15), [15-20]. Array literals may include signal references as elements. * - A [bin specification object](https://vega.github.io/vega-lite/docs/scale.html#bins) that indicates the bin _step_ size, and optionally the _start_ and _stop_ boundaries. * - An array of bin boundaries over the scale domain. If provided, axes and legends will use the bin boundaries to inform the choice of tick marks and text labels. */ // TODO: add - A [signal reference](../types/#Signal) that resolves to either an array or bin specification object. bins?: ScaleBins;
/** * If `true`, rounds numeric output values to integers. This can be helpful for snapping to the pixel grid. * * __Default value:__ `false`. */ round?: boolean | ES;
/** * For _[continuous](https://vega.github.io/vega-lite/docs/scale.html#continuous)_ scales, expands the scale domain to accommodate the specified number of pixels on each of the scale range. The scale range must represent pixels for this parameter to function as intended. Padding adjustment is performed prior to all other adjustments, including the effects of the `zero`, `nice`, `domainMin`, and `domainMax` properties. * * For _[band](https://vega.github.io/vega-lite/docs/scale.html#band)_ scales, shortcut for setting `paddingInner` and `paddingOuter` to the same value. * * For _[point](https://vega.github.io/vega-lite/docs/scale.html#point)_ scales, alias for `paddingOuter`. * * __Default value:__ For _continuous_ scales, derived from the [scale config](https://vega.github.io/vega-lite/docs/scale.html#config)'s `continuousPadding`. * For _band and point_ scales, see `paddingInner` and `paddingOuter`. By default, Vega-Lite sets padding such that _width/height = number of unique values * step_. * * @minimum 0 */ padding?: number | ES;
/** * The inner padding (spacing) within each band step of band scales, as a fraction of the step size. This value must lie in the range [0,1]. * * For point scale, this property is invalid as point scales do not have internal band widths (only step sizes between bands). * * __Default value:__ derived from the [scale config](https://vega.github.io/vega-lite/docs/scale.html#config)'s `bandPaddingInner`. * * @minimum 0 * @maximum 1 */ paddingInner?: number | ES;
/** * The outer padding (spacing) at the ends of the range of band and point scales, * as a fraction of the step size. This value must lie in the range [0,1]. * * __Default value:__ derived from the [scale config](https://vega.github.io/vega-lite/docs/scale.html#config)'s `bandPaddingOuter` for band scales and `pointPadding` for point scales. * By default, Vega-Lite sets outer padding such that _width/height = number of unique values * step_. * * @minimum 0 * @maximum 1 */ paddingOuter?: number | ES;
// typical /** * If `true`, values that exceed the data domain are clamped to either the minimum or maximum range value * * __Default value:__ derived from the [scale config](https://vega.github.io/vega-lite/docs/config.html#scale-config)'s `clamp` (`true` by default). */ clamp?: boolean | ES;
/** * Extending the domain so that it starts and ends on nice round values. This method typically modifies the scale’s domain, and may only extend the bounds to the nearest round value. Nicing is useful if the domain is computed from data and may be irregular. For example, for a domain of _[0.201479…, 0.996679…]_, a nice domain might be _[0.2, 1.0]_. * * For quantitative scales such as linear, `nice` can be either a boolean flag or a number. If `nice` is a number, it will represent a desired tick count. This allows greater control over the step size used to extend the bounds, guaranteeing that the returned ticks will exactly cover the domain. * * For temporal fields with time and utc scales, the `nice` value can be a string indicating the desired time interval. Legal values are `"millisecond"`, `"second"`, `"minute"`, `"hour"`, `"day"`, `"week"`, `"month"`, and `"year"`. Alternatively, `time` and `utc` scales can accept an object-valued interval specifier of the form `{"interval": "month", "step": 3}`, which includes a desired number of interval steps. Here, the domain would snap to quarter (Jan, Apr, Jul, Oct) boundaries. * * __Default value:__ `true` for unbinned _quantitative_ fields; `false` otherwise. * */ nice?: boolean | number | TimeInterval | TimeIntervalStep | ES;
/** * The logarithm base of the `log` scale (default `10`). */ base?: number | ES;
/** * The exponent of the `pow` scale. */ exponent?: number | ES;
/** * A constant determining the slope of the symlog function around zero. Only used for `symlog` scales. * * __Default value:__ `1` */ constant?: number | ES;
/** * If `true`, ensures that a zero baseline value is included in the scale domain. * * __Default value:__ `true` for x and y channels if the quantitative field is not binned and no custom `domain` is provided; `false` otherwise. * * __Note:__ Log, time, and utc scales do not support `zero`. */ zero?: boolean | ES;
/** * The interpolation method for range values. By default, a general interpolator for numbers, dates, strings and colors (in HCL space) is used. For color ranges, this property allows interpolation in alternative color spaces. Legal values include `rgb`, `hsl`, `hsl-long`, `lab`, `hcl`, `hcl-long`, `cubehelix` and `cubehelix-long` ('-long' variants use longer paths in polar coordinate spaces). If object-valued, this property accepts an object with a string-valued _type_ property and an optional numeric _gamma_ property applicable to rgb and cubehelix interpolators. For more, see the [d3-interpolate documentation](https://github.com/d3/d3-interpolate). * * * __Default value:__ `hcl` */ interpolate?: ScaleInterpolateEnum | ES | ScaleInterpolateParams;}
const SCALE_PROPERTY_INDEX: Flag<keyof Scale<any>> = { type: 1, domain: 1, domainMax: 1, domainMin: 1, domainMid: 1, align: 1, range: 1, rangeMax: 1, rangeMin: 1, scheme: 1, bins: 1, // Other properties reverse: 1, round: 1, // quantitative / time clamp: 1, nice: 1, // quantitative base: 1, exponent: 1, constant: 1, interpolate: 1, zero: 1, // zero depends on domain // band/point padding: 1, paddingInner: 1, paddingOuter: 1};
export const SCALE_PROPERTIES = keys(SCALE_PROPERTY_INDEX);
const { type, domain, range, rangeMax, rangeMin, scheme, ...NON_TYPE_DOMAIN_RANGE_VEGA_SCALE_PROPERTY_INDEX} = SCALE_PROPERTY_INDEX;
export const NON_TYPE_DOMAIN_RANGE_VEGA_SCALE_PROPERTIES = keys(NON_TYPE_DOMAIN_RANGE_VEGA_SCALE_PROPERTY_INDEX);
export function scaleTypeSupportProperty(scaleType: ScaleType, propName: keyof Scale): boolean { switch (propName) { case 'type': case 'domain': case 'reverse': case 'range': return true; case 'scheme': case 'interpolate': return !contains(['point', 'band', 'identity'], scaleType); case 'bins': return !contains(['point', 'band', 'identity', 'ordinal'], scaleType); case 'round': return isContinuousToContinuous(scaleType) || scaleType === 'band' || scaleType === 'point'; case 'padding': case 'rangeMin': case 'rangeMax': return isContinuousToContinuous(scaleType) || contains(['point', 'band'], scaleType); case 'paddingOuter': case 'align': return contains(['point', 'band'], scaleType); case 'paddingInner': return scaleType === 'band'; case 'domainMax': case 'domainMid': case 'domainMin': case 'clamp': return isContinuousToContinuous(scaleType); case 'nice': return isContinuousToContinuous(scaleType) || scaleType === 'quantize' || scaleType === 'threshold'; case 'exponent': return scaleType === 'pow'; case 'base': return scaleType === 'log'; case 'constant': return scaleType === 'symlog'; case 'zero': return ( hasContinuousDomain(scaleType) && !contains( [ 'log', // log scale cannot have zero value 'time', 'utc', // zero is not meaningful for time 'threshold', // threshold requires custom domain so zero does not matter 'quantile' // quantile depends on distribution so zero does not matter ], scaleType ) ); }}
/** * Returns undefined if the input channel supports the input scale property name */export function channelScalePropertyIncompatability(channel: Channel, propName: keyof Scale): string { switch (propName) { case 'interpolate': case 'scheme': case 'domainMid': if (!isColorChannel(channel)) { return log.message.cannotUseScalePropertyWithNonColor(channel); } return undefined; case 'align': case 'type': case 'bins': case 'domain': case 'domainMax': case 'domainMin': case 'range': case 'base': case 'exponent': case 'constant': case 'nice': case 'padding': case 'paddingInner': case 'paddingOuter': case 'rangeMax': case 'rangeMin': case 'reverse': case 'round': case 'clamp': case 'zero': return undefined; // GOOD! }}
export function scaleTypeSupportDataType(specifiedType: ScaleType, fieldDefType: Type): boolean { if (contains([ORDINAL, NOMINAL], fieldDefType)) { return specifiedType === undefined || hasDiscreteDomain(specifiedType); } else if (fieldDefType === TEMPORAL) { return contains([ScaleType.TIME, ScaleType.UTC, undefined], specifiedType); } else if (fieldDefType === QUANTITATIVE) { return contains( [ ScaleType.LOG, ScaleType.POW, ScaleType.SQRT, ScaleType.SYMLOG, ScaleType.QUANTILE, ScaleType.QUANTIZE, ScaleType.THRESHOLD, ScaleType.LINEAR, undefined ], specifiedType ); }
return true;}
export function channelSupportScaleType(channel: Channel, scaleType: ScaleType): boolean { if (!CHANNEL.isScaleChannel(channel)) { return false; } switch (channel) { case CHANNEL.X: case CHANNEL.Y: case CHANNEL.THETA: case CHANNEL.RADIUS: return isContinuousToContinuous(scaleType) || contains(['band', 'point'], scaleType); case CHANNEL.SIZE: // TODO: size and opacity can support ordinal with more modification case CHANNEL.STROKEWIDTH: case CHANNEL.OPACITY: case CHANNEL.FILLOPACITY: case CHANNEL.STROKEOPACITY: case CHANNEL.ANGLE: // Although it generally doesn't make sense to use band with size and opacity, // it can also work since we use band: 0.5 to get midpoint. return ( isContinuousToContinuous(scaleType) || isContinuousToDiscrete(scaleType) || contains(['band', 'point', 'ordinal'], scaleType) ); case CHANNEL.COLOR: case CHANNEL.FILL: case CHANNEL.STROKE: return scaleType !== 'band'; // band does not make sense with color case CHANNEL.STROKEDASH: return scaleType === 'ordinal' || isContinuousToDiscrete(scaleType); case CHANNEL.SHAPE: return scaleType === 'ordinal'; // shape = lookup only }}