fabric
Version:
Object model for HTML5 canvas, and SVG-to-canvas parser. Backed by jsdom and node-canvas.
95 lines (84 loc) • 2.59 kB
text/typescript
import { BaseFilter } from './BaseFilter';
import type { T2DPipelineState, TWebGLPipelineState } from './typedefs';
import { isWebGLPipelineState } from './utils';
import { classRegistry } from '../ClassRegistry';
type ComposedOwnProps = {
subFilters: BaseFilter<string, object, object>[];
};
type ComposedSerializedProps = {
subFilters: Record<string, unknown>[];
};
/**
* A container class that knows how to apply a sequence of filters to an input image.
*/
export class Composed extends BaseFilter<
'Composed',
ComposedOwnProps,
ComposedSerializedProps
> {
/**
* A non sparse array of filters to apply
*/
declare subFilters: ComposedOwnProps['subFilters'];
static type = 'Composed';
constructor(
options: { subFilters?: BaseFilter<string, object>[] } & Record<
string,
any
> = {},
) {
super(options);
this.subFilters = options.subFilters || [];
}
/**
* Apply this container's filters to the input image provided.
*
* @param {Object} options
* @param {Number} options.passes The number of filters remaining to be applied.
*/
applyTo(options: TWebGLPipelineState | T2DPipelineState) {
if (isWebGLPipelineState(options)) {
options.passes += this.subFilters.length - 1;
}
this.subFilters.forEach((filter) => {
filter.applyTo(options);
});
}
/**
* Serialize this filter into JSON.
* @returns {Object} A JSON representation of this filter.
*/
toObject() {
return {
type: this.type,
subFilters: this.subFilters.map((filter) => filter.toObject()),
};
}
isNeutralState() {
return !this.subFilters.some((filter) => !filter.isNeutralState());
}
/**
* Deserialize a JSON definition of a ComposedFilter into a concrete instance.
* @static
* @param {oject} object Object to create an instance from
* @param {object} [options]
* @param {AbortSignal} [options.signal] handle aborting `BlendImage` filter loading, see https://developer.mozilla.org/en-US/docs/Web/API/AbortController/signal
* @returns {Promise<Composed>}
*/
static fromObject(
object: Record<string, any>,
options: { signal: AbortSignal },
): Promise<Composed> {
return Promise.all(
((object.subFilters || []) as BaseFilter<string, object>[]).map(
(filter) =>
classRegistry
.getClass<typeof BaseFilter>(filter.type)
.fromObject(filter, options),
),
).then(
(enlivedFilters) => new this({ subFilters: enlivedFilters }) as Composed,
);
}
}
classRegistry.setClass(Composed);