fabric
Version:
Object model for HTML5 canvas, and SVG-to-canvas parser. Backed by jsdom and node-canvas.
88 lines (77 loc) • 2.51 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>[] } & 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.
* @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>[]).map((filter) =>
classRegistry
.getClass<typeof BaseFilter>(filter.type)
.fromObject(filter, options),
),
).then((enlivedFilters) => new this({ subFilters: enlivedFilters }));
}
}
classRegistry.setClass(Composed);