123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296 |
- /**
- * Copyright (c) 2019-2020 mol* contributors, licensed under MIT, See LICENSE file for more info.
- *
- * @author Alexander Rose <alexander.rose@weirdbyte.de>
- */
- import { ParamDefinition as PD } from '../../../mol-util/param-definition';
- import { ChunkedArray } from '../../../mol-data/util';
- import { Text } from './text';
- import { getFontAtlas } from './font-atlas';
- const quadIndices = new Uint16Array([
- 0, 1, 2,
- 1, 3, 2
- ])
- export interface TextBuilder {
- add(str: string, x: number, y: number, z: number, depth: number, scale: number, group: number): void
- getText(): Text
- }
- export namespace TextBuilder {
- export function create(props: Partial<PD.Values<Text.Params>> = {}, initialCount = 2048, chunkSize = 1024, text?: Text): TextBuilder {
- initialCount *= 2
- chunkSize *= 2
- const centers = ChunkedArray.create(Float32Array, 3, chunkSize, text ? text.centerBuffer.ref.value : initialCount);
- const mappings = ChunkedArray.create(Float32Array, 2, chunkSize, text ? text.mappingBuffer.ref.value : initialCount);
- const depths = ChunkedArray.create(Float32Array, 1, chunkSize, text ? text.depthBuffer.ref.value : initialCount);
- const indices = ChunkedArray.create(Uint32Array, 3, chunkSize, text ? text.indexBuffer.ref.value : initialCount);
- const groups = ChunkedArray.create(Float32Array, 1, chunkSize, text ? text.groupBuffer.ref.value : initialCount);
- const tcoords = ChunkedArray.create(Float32Array, 2, chunkSize, text ? text.tcoordBuffer.ref.value : initialCount);
- const p = { ...PD.getDefaultValues(Text.Params), ...props }
- const { attachment, background, backgroundMargin, tether, tetherLength, tetherBaseWidth } = p
- const fontAtlas = getFontAtlas(p)
- const margin = (1 / 2.5) * backgroundMargin
- const outline = fontAtlas.buffer / fontAtlas.lineHeight
- const add = (x: number, y: number, z: number, depth: number, group: number) => {
- ChunkedArray.add3(centers, x, y, z);
- ChunkedArray.add(depths, depth);
- ChunkedArray.add(groups, group);
- }
- return {
- add: (str: string, x: number, y: number, z: number, depth: number, scale: number, group: number) => {
- let bWidth = 0
- const nChar = str.length
- // calculate width
- for (let iChar = 0; iChar < nChar; ++iChar) {
- const c = fontAtlas.get(str[iChar])
- bWidth += c.nw - 2 * outline
- }
- const bHeight = 1 / 1.25
- // attachment
- let yShift: number, xShift: number
- // vertical
- if (attachment.startsWith('top')) {
- yShift = bHeight
- } else if (attachment.startsWith('middle')) {
- yShift = bHeight / 2
- } else {
- yShift = 0 // "bottom"
- }
- // horizontal
- if (attachment.endsWith('right')) {
- xShift = bWidth
- } else if (attachment.endsWith('center')) {
- xShift = bWidth / 2
- } else {
- xShift = 0 // "left"
- }
- if (tether) {
- switch (attachment) {
- case 'bottom-left':
- xShift -= tetherLength / 2 + margin + 0.1
- yShift -= tetherLength / 2 + margin
- break
- case 'bottom-center':
- yShift -= tetherLength + margin
- break
- case 'bottom-right':
- xShift += tetherLength / 2 + margin + 0.1
- yShift -= tetherLength / 2 + margin
- break
- case 'middle-left':
- xShift -= tetherLength + margin + 0.1
- break
- case 'middle-center':
- break
- case 'middle-right':
- xShift += tetherLength + margin + 0.1
- break
- case 'top-left':
- xShift -= tetherLength / 2 + margin + 0.1
- yShift += tetherLength / 2 + margin
- break
- case 'top-center':
- yShift += tetherLength + margin
- break
- case 'top-right':
- xShift += tetherLength / 2 + margin + 0.1
- yShift += tetherLength / 2 + margin
- break
- }
- }
- const xLeft = (-xShift - margin - 0.1) * scale
- const xRight = (bWidth - xShift + margin + 0.1) * scale
- const yTop = (bHeight - yShift + margin) * scale
- const yBottom = (-yShift - margin) * scale
- // background
- if (background) {
- ChunkedArray.add2(mappings, xLeft, yTop) // top left
- ChunkedArray.add2(mappings, xLeft, yBottom) // bottom left
- ChunkedArray.add2(mappings, xRight, yTop) // top right
- ChunkedArray.add2(mappings, xRight, yBottom) // bottom right
- const offset = centers.elementCount
- for (let i = 0; i < 4; ++i) {
- ChunkedArray.add2(tcoords, 10, 10)
- add(x, y, z, depth, group)
- }
- ChunkedArray.add3(indices, offset + quadIndices[0], offset + quadIndices[1], offset + quadIndices[2])
- ChunkedArray.add3(indices, offset + quadIndices[3], offset + quadIndices[4], offset + quadIndices[5])
- }
- if (tether) {
- let xTip: number, yTip: number
- let xBaseA: number, yBaseA: number
- let xBaseB: number, yBaseB: number
- let xBaseCenter: number, yBaseCenter: number
- const scaledTetherLength = tetherLength * scale
- const scaledTetherBaseWidth = tetherBaseWidth * scale
- switch (attachment) {
- case 'bottom-left':
- xTip = xLeft - scaledTetherLength / 2
- xBaseA = xLeft + scaledTetherBaseWidth / 2
- xBaseB = xLeft
- xBaseCenter = xLeft
- yTip = yBottom - scaledTetherLength / 2
- yBaseA = yBottom
- yBaseB = yBottom + scaledTetherBaseWidth / 2
- yBaseCenter = yBottom
- break
- case 'bottom-center':
- xTip = 0
- xBaseA = scaledTetherBaseWidth / 2
- xBaseB = -scaledTetherBaseWidth / 2
- xBaseCenter = 0
- yTip = yBottom - scaledTetherLength
- yBaseA = yBottom
- yBaseB = yBottom
- yBaseCenter = yBottom
- break
- case 'bottom-right':
- xTip = xRight + scaledTetherLength / 2
- xBaseA = xRight
- xBaseB = xRight - scaledTetherBaseWidth / 2
- xBaseCenter = xRight
- yTip = yBottom - scaledTetherLength / 2
- yBaseA = yBottom + scaledTetherBaseWidth / 2
- yBaseB = yBottom
- yBaseCenter = yBottom
- break
- case 'middle-left':
- xTip = xLeft - scaledTetherLength
- xBaseA = xLeft
- xBaseB = xLeft
- xBaseCenter = xLeft
- yTip = 0
- yBaseA = -scaledTetherBaseWidth / 2
- yBaseB = scaledTetherBaseWidth / 2
- yBaseCenter = 0
- break
- case 'middle-center':
- xTip = 0
- xBaseA = 0
- xBaseB = 0
- xBaseCenter = 0
- yTip = 0
- yBaseA = 0
- yBaseB = 0
- yBaseCenter = 0
- break
- case 'middle-right':
- xTip = xRight + scaledTetherLength
- xBaseA = xRight
- xBaseB = xRight
- xBaseCenter = xRight
- yTip = 0
- yBaseA = scaledTetherBaseWidth / 2
- yBaseB = -scaledTetherBaseWidth / 2
- yBaseCenter = 0
- break
- case 'top-left':
- xTip = xLeft - scaledTetherLength / 2
- xBaseA = xLeft + scaledTetherBaseWidth / 2
- xBaseB = xLeft
- xBaseCenter = xLeft
- yTip = yTop + scaledTetherLength / 2
- yBaseA = yTop
- yBaseB = yTop - scaledTetherBaseWidth / 2
- yBaseCenter = yTop
- break
- case 'top-center':
- xTip = 0
- xBaseA = scaledTetherBaseWidth / 2
- xBaseB = -scaledTetherBaseWidth / 2
- xBaseCenter = 0
- yTip = yTop + scaledTetherLength
- yBaseA = yTop
- yBaseB = yTop
- yBaseCenter = yTop
- break
- case 'top-right':
- xTip = xRight + scaledTetherLength / 2
- xBaseA = xRight
- xBaseB = xRight - scaledTetherBaseWidth / 2
- xBaseCenter = xRight
- yTip = yTop + scaledTetherLength / 2
- yBaseA = yTop - scaledTetherBaseWidth / 2
- yBaseB = yTop
- yBaseCenter = yTop
- break
- default:
- throw new Error('unsupported attachment')
- }
- ChunkedArray.add2(mappings, xTip, yTip) // tip
- ChunkedArray.add2(mappings, xBaseA, yBaseA) // base A
- ChunkedArray.add2(mappings, xBaseB, yBaseB) // base B
- ChunkedArray.add2(mappings, xBaseCenter, yBaseCenter) // base center
- const offset = centers.elementCount
- for (let i = 0; i < 4; ++i) {
- ChunkedArray.add2(tcoords, 10, 10)
- add(x, y, z, depth, group)
- }
- ChunkedArray.add3(indices, offset, offset + 1, offset + 3)
- ChunkedArray.add3(indices, offset, offset + 3, offset + 2)
- }
- xShift += outline
- yShift += outline
- let xadvance = 0
- for (let iChar = 0; iChar < nChar; ++iChar) {
- const c = fontAtlas.get(str[iChar])
- const left = (xadvance - xShift) * scale
- const right = (xadvance + c.nw - xShift) * scale
- const top = (c.nh - yShift) * scale
- const bottom = (-yShift) * scale
- ChunkedArray.add2(mappings, left, top)
- ChunkedArray.add2(mappings, left, bottom)
- ChunkedArray.add2(mappings, right, top)
- ChunkedArray.add2(mappings, right, bottom)
- const texWidth = fontAtlas.texture.width
- const texHeight = fontAtlas.texture.height
- ChunkedArray.add2(tcoords, c.x / texWidth, c.y / texHeight) // top left
- ChunkedArray.add2(tcoords, c.x / texWidth, (c.y + c.h) / texHeight) // bottom left
- ChunkedArray.add2(tcoords, (c.x + c.w) / texWidth, c.y / texHeight) // top right
- ChunkedArray.add2(tcoords, (c.x + c.w) / texWidth, (c.y + c.h) / texHeight) // bottom right
- xadvance += c.nw - 2 * outline
- const offset = centers.elementCount
- for (let i = 0; i < 4; ++i) add(x, y, z, depth, group)
- ChunkedArray.add3(indices, offset + quadIndices[0], offset + quadIndices[1], offset + quadIndices[2])
- ChunkedArray.add3(indices, offset + quadIndices[3], offset + quadIndices[4], offset + quadIndices[5])
- }
- },
- getText: () => {
- const ft = fontAtlas.texture
- const cb = ChunkedArray.compact(centers, true) as Float32Array
- const mb = ChunkedArray.compact(mappings, true) as Float32Array
- const db = ChunkedArray.compact(depths, true) as Float32Array
- const ib = ChunkedArray.compact(indices, true) as Uint32Array
- const gb = ChunkedArray.compact(groups, true) as Float32Array
- const tb = ChunkedArray.compact(tcoords, true) as Float32Array
- return Text.create(ft, cb, mb, db, ib, gb, tb, indices.elementCount / 2, text)
- }
- }
- }
- }
|