/** * Copyright (c) 2019 mol* contributors, licensed under MIT, See LICENSE file for more info. * * @author Alexander Rose */ import { Loci } from '../../../mol-model/loci'; import { RuntimeContext } from '../../../mol-task'; import { Text } from '../../../mol-geo/geometry/text/text'; import { ParamDefinition as PD } from '../../../mol-util/param-definition'; import { ShapeRepresentation } from '../representation'; import { Representation, RepresentationParamsGetter, RepresentationContext } from '../../representation'; import { Shape } from '../../../mol-model/shape'; import { TextBuilder } from '../../../mol-geo/geometry/text/text-builder'; import { Sphere3D } from '../../../mol-math/geometry'; import { lociLabel } from '../../../mol-theme/label'; import { LociLabelTextParams } from './common'; export interface LabelData { infos: { loci: Loci, label?: string }[] } const TextParams = { ...LociLabelTextParams, }; type TextParams = typeof TextParams const LabelVisuals = { 'text': (ctx: RepresentationContext, getParams: RepresentationParamsGetter) => ShapeRepresentation(getTextShape, Text.Utils, { modifyState: s => ({ ...s, pickable: false }) }), }; export const LabelParams = { ...TextParams, scaleByRadius: PD.Boolean(true), visuals: PD.MultiSelect(['text'], PD.objectToOptions(LabelVisuals)), }; export type LabelParams = typeof LabelParams export type LabelProps = PD.Values // const tmpSphere = Sphere3D(); function label(info: { loci: Loci, label?: string }, condensed = false) { return info.label || lociLabel(info.loci, { hidePrefix: true, htmlStyling: false, condensed }); } function getLabelName(data: LabelData) { return data.infos.length === 1 ? label(data.infos[0]) : `${data.infos.length} Labels`; } // function buildText(data: LabelData, props: LabelProps, text?: Text): Text { const builder = TextBuilder.create(props, 128, 64, text); const customLabel = props.customText.trim(); for (let i = 0, il = data.infos.length; i < il; ++i) { const info = data.infos[i]; const sphere = Loci.getBoundingSphere(info.loci, tmpSphere); if (!sphere) continue; const { center, radius } = sphere; const text = customLabel || label(info, true); builder.add(text, center[0], center[1], center[2], props.scaleByRadius ? radius / 0.9 : 0, props.scaleByRadius ? Math.max(1, radius) : 1, i); } return builder.getText(); } function getTextShape(ctx: RuntimeContext, data: LabelData, props: LabelProps, shape?: Shape) { const text = buildText(data, props, shape && shape.geometry); const name = getLabelName(data); const customLabel = props.customText.trim(); const getLabel = customLabel ? function (groupId: number) { return customLabel; } : function (groupId: number) { return label(data.infos[groupId]); }; return Shape.create(name, data, text, () => props.textColor, () => props.textSize, getLabel); } // export type LabelRepresentation = Representation export function LabelRepresentation(ctx: RepresentationContext, getParams: RepresentationParamsGetter): LabelRepresentation { return Representation.createMulti('Label', ctx, getParams, Representation.StateBuilder, LabelVisuals as unknown as Representation.Def); }