text-builder.ts 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296
  1. /**
  2. * Copyright (c) 2019-2020 mol* contributors, licensed under MIT, See LICENSE file for more info.
  3. *
  4. * @author Alexander Rose <alexander.rose@weirdbyte.de>
  5. */
  6. import { ParamDefinition as PD } from '../../../mol-util/param-definition';
  7. import { ChunkedArray } from '../../../mol-data/util';
  8. import { Text } from './text';
  9. import { getFontAtlas } from './font-atlas';
  10. const quadIndices = new Uint16Array([
  11. 0, 1, 2,
  12. 1, 3, 2
  13. ])
  14. export interface TextBuilder {
  15. add(str: string, x: number, y: number, z: number, depth: number, scale: number, group: number): void
  16. getText(): Text
  17. }
  18. export namespace TextBuilder {
  19. export function create(props: Partial<PD.Values<Text.Params>> = {}, initialCount = 2048, chunkSize = 1024, text?: Text): TextBuilder {
  20. initialCount *= 2
  21. chunkSize *= 2
  22. const centers = ChunkedArray.create(Float32Array, 3, chunkSize, text ? text.centerBuffer.ref.value : initialCount);
  23. const mappings = ChunkedArray.create(Float32Array, 2, chunkSize, text ? text.mappingBuffer.ref.value : initialCount);
  24. const depths = ChunkedArray.create(Float32Array, 1, chunkSize, text ? text.depthBuffer.ref.value : initialCount);
  25. const indices = ChunkedArray.create(Uint32Array, 3, chunkSize, text ? text.indexBuffer.ref.value : initialCount);
  26. const groups = ChunkedArray.create(Float32Array, 1, chunkSize, text ? text.groupBuffer.ref.value : initialCount);
  27. const tcoords = ChunkedArray.create(Float32Array, 2, chunkSize, text ? text.tcoordBuffer.ref.value : initialCount);
  28. const p = { ...PD.getDefaultValues(Text.Params), ...props }
  29. const { attachment, background, backgroundMargin, tether, tetherLength, tetherBaseWidth } = p
  30. const fontAtlas = getFontAtlas(p)
  31. const margin = (1 / 2.5) * backgroundMargin
  32. const outline = fontAtlas.buffer / fontAtlas.lineHeight
  33. const add = (x: number, y: number, z: number, depth: number, group: number) => {
  34. ChunkedArray.add3(centers, x, y, z);
  35. ChunkedArray.add(depths, depth);
  36. ChunkedArray.add(groups, group);
  37. }
  38. return {
  39. add: (str: string, x: number, y: number, z: number, depth: number, scale: number, group: number) => {
  40. let bWidth = 0
  41. const nChar = str.length
  42. // calculate width
  43. for (let iChar = 0; iChar < nChar; ++iChar) {
  44. const c = fontAtlas.get(str[iChar])
  45. bWidth += c.nw - 2 * outline
  46. }
  47. const bHeight = 1 / 1.25
  48. // attachment
  49. let yShift: number, xShift: number
  50. // vertical
  51. if (attachment.startsWith('top')) {
  52. yShift = bHeight
  53. } else if (attachment.startsWith('middle')) {
  54. yShift = bHeight / 2
  55. } else {
  56. yShift = 0 // "bottom"
  57. }
  58. // horizontal
  59. if (attachment.endsWith('right')) {
  60. xShift = bWidth
  61. } else if (attachment.endsWith('center')) {
  62. xShift = bWidth / 2
  63. } else {
  64. xShift = 0 // "left"
  65. }
  66. if (tether) {
  67. switch (attachment) {
  68. case 'bottom-left':
  69. xShift -= tetherLength / 2 + margin + 0.1
  70. yShift -= tetherLength / 2 + margin
  71. break
  72. case 'bottom-center':
  73. yShift -= tetherLength + margin
  74. break
  75. case 'bottom-right':
  76. xShift += tetherLength / 2 + margin + 0.1
  77. yShift -= tetherLength / 2 + margin
  78. break
  79. case 'middle-left':
  80. xShift -= tetherLength + margin + 0.1
  81. break
  82. case 'middle-center':
  83. break
  84. case 'middle-right':
  85. xShift += tetherLength + margin + 0.1
  86. break
  87. case 'top-left':
  88. xShift -= tetherLength / 2 + margin + 0.1
  89. yShift += tetherLength / 2 + margin
  90. break
  91. case 'top-center':
  92. yShift += tetherLength + margin
  93. break
  94. case 'top-right':
  95. xShift += tetherLength / 2 + margin + 0.1
  96. yShift += tetherLength / 2 + margin
  97. break
  98. }
  99. }
  100. const xLeft = (-xShift - margin - 0.1) * scale
  101. const xRight = (bWidth - xShift + margin + 0.1) * scale
  102. const yTop = (bHeight - yShift + margin) * scale
  103. const yBottom = (-yShift - margin) * scale
  104. // background
  105. if (background) {
  106. ChunkedArray.add2(mappings, xLeft, yTop) // top left
  107. ChunkedArray.add2(mappings, xLeft, yBottom) // bottom left
  108. ChunkedArray.add2(mappings, xRight, yTop) // top right
  109. ChunkedArray.add2(mappings, xRight, yBottom) // bottom right
  110. const offset = centers.elementCount
  111. for (let i = 0; i < 4; ++i) {
  112. ChunkedArray.add2(tcoords, 10, 10)
  113. add(x, y, z, depth, group)
  114. }
  115. ChunkedArray.add3(indices, offset + quadIndices[0], offset + quadIndices[1], offset + quadIndices[2])
  116. ChunkedArray.add3(indices, offset + quadIndices[3], offset + quadIndices[4], offset + quadIndices[5])
  117. }
  118. if (tether) {
  119. let xTip: number, yTip: number
  120. let xBaseA: number, yBaseA: number
  121. let xBaseB: number, yBaseB: number
  122. let xBaseCenter: number, yBaseCenter: number
  123. const scaledTetherLength = tetherLength * scale
  124. const scaledTetherBaseWidth = tetherBaseWidth * scale
  125. switch (attachment) {
  126. case 'bottom-left':
  127. xTip = xLeft - scaledTetherLength / 2
  128. xBaseA = xLeft + scaledTetherBaseWidth / 2
  129. xBaseB = xLeft
  130. xBaseCenter = xLeft
  131. yTip = yBottom - scaledTetherLength / 2
  132. yBaseA = yBottom
  133. yBaseB = yBottom + scaledTetherBaseWidth / 2
  134. yBaseCenter = yBottom
  135. break
  136. case 'bottom-center':
  137. xTip = 0
  138. xBaseA = scaledTetherBaseWidth / 2
  139. xBaseB = -scaledTetherBaseWidth / 2
  140. xBaseCenter = 0
  141. yTip = yBottom - scaledTetherLength
  142. yBaseA = yBottom
  143. yBaseB = yBottom
  144. yBaseCenter = yBottom
  145. break
  146. case 'bottom-right':
  147. xTip = xRight + scaledTetherLength / 2
  148. xBaseA = xRight
  149. xBaseB = xRight - scaledTetherBaseWidth / 2
  150. xBaseCenter = xRight
  151. yTip = yBottom - scaledTetherLength / 2
  152. yBaseA = yBottom + scaledTetherBaseWidth / 2
  153. yBaseB = yBottom
  154. yBaseCenter = yBottom
  155. break
  156. case 'middle-left':
  157. xTip = xLeft - scaledTetherLength
  158. xBaseA = xLeft
  159. xBaseB = xLeft
  160. xBaseCenter = xLeft
  161. yTip = 0
  162. yBaseA = -scaledTetherBaseWidth / 2
  163. yBaseB = scaledTetherBaseWidth / 2
  164. yBaseCenter = 0
  165. break
  166. case 'middle-center':
  167. xTip = 0
  168. xBaseA = 0
  169. xBaseB = 0
  170. xBaseCenter = 0
  171. yTip = 0
  172. yBaseA = 0
  173. yBaseB = 0
  174. yBaseCenter = 0
  175. break
  176. case 'middle-right':
  177. xTip = xRight + scaledTetherLength
  178. xBaseA = xRight
  179. xBaseB = xRight
  180. xBaseCenter = xRight
  181. yTip = 0
  182. yBaseA = scaledTetherBaseWidth / 2
  183. yBaseB = -scaledTetherBaseWidth / 2
  184. yBaseCenter = 0
  185. break
  186. case 'top-left':
  187. xTip = xLeft - scaledTetherLength / 2
  188. xBaseA = xLeft + scaledTetherBaseWidth / 2
  189. xBaseB = xLeft
  190. xBaseCenter = xLeft
  191. yTip = yTop + scaledTetherLength / 2
  192. yBaseA = yTop
  193. yBaseB = yTop - scaledTetherBaseWidth / 2
  194. yBaseCenter = yTop
  195. break
  196. case 'top-center':
  197. xTip = 0
  198. xBaseA = scaledTetherBaseWidth / 2
  199. xBaseB = -scaledTetherBaseWidth / 2
  200. xBaseCenter = 0
  201. yTip = yTop + scaledTetherLength
  202. yBaseA = yTop
  203. yBaseB = yTop
  204. yBaseCenter = yTop
  205. break
  206. case 'top-right':
  207. xTip = xRight + scaledTetherLength / 2
  208. xBaseA = xRight
  209. xBaseB = xRight - scaledTetherBaseWidth / 2
  210. xBaseCenter = xRight
  211. yTip = yTop + scaledTetherLength / 2
  212. yBaseA = yTop - scaledTetherBaseWidth / 2
  213. yBaseB = yTop
  214. yBaseCenter = yTop
  215. break
  216. default:
  217. throw new Error('unsupported attachment')
  218. }
  219. ChunkedArray.add2(mappings, xTip, yTip) // tip
  220. ChunkedArray.add2(mappings, xBaseA, yBaseA) // base A
  221. ChunkedArray.add2(mappings, xBaseB, yBaseB) // base B
  222. ChunkedArray.add2(mappings, xBaseCenter, yBaseCenter) // base center
  223. const offset = centers.elementCount
  224. for (let i = 0; i < 4; ++i) {
  225. ChunkedArray.add2(tcoords, 10, 10)
  226. add(x, y, z, depth, group)
  227. }
  228. ChunkedArray.add3(indices, offset, offset + 1, offset + 3)
  229. ChunkedArray.add3(indices, offset, offset + 3, offset + 2)
  230. }
  231. xShift += outline
  232. yShift += outline
  233. let xadvance = 0
  234. for (let iChar = 0; iChar < nChar; ++iChar) {
  235. const c = fontAtlas.get(str[iChar])
  236. const left = (xadvance - xShift) * scale
  237. const right = (xadvance + c.nw - xShift) * scale
  238. const top = (c.nh - yShift) * scale
  239. const bottom = (-yShift) * scale
  240. ChunkedArray.add2(mappings, left, top)
  241. ChunkedArray.add2(mappings, left, bottom)
  242. ChunkedArray.add2(mappings, right, top)
  243. ChunkedArray.add2(mappings, right, bottom)
  244. const texWidth = fontAtlas.texture.width
  245. const texHeight = fontAtlas.texture.height
  246. ChunkedArray.add2(tcoords, c.x / texWidth, c.y / texHeight) // top left
  247. ChunkedArray.add2(tcoords, c.x / texWidth, (c.y + c.h) / texHeight) // bottom left
  248. ChunkedArray.add2(tcoords, (c.x + c.w) / texWidth, c.y / texHeight) // top right
  249. ChunkedArray.add2(tcoords, (c.x + c.w) / texWidth, (c.y + c.h) / texHeight) // bottom right
  250. xadvance += c.nw - 2 * outline
  251. const offset = centers.elementCount
  252. for (let i = 0; i < 4; ++i) add(x, y, z, depth, group)
  253. ChunkedArray.add3(indices, offset + quadIndices[0], offset + quadIndices[1], offset + quadIndices[2])
  254. ChunkedArray.add3(indices, offset + quadIndices[3], offset + quadIndices[4], offset + quadIndices[5])
  255. }
  256. },
  257. getText: () => {
  258. const ft = fontAtlas.texture
  259. const cb = ChunkedArray.compact(centers, true) as Float32Array
  260. const mb = ChunkedArray.compact(mappings, true) as Float32Array
  261. const db = ChunkedArray.compact(depths, true) as Float32Array
  262. const ib = ChunkedArray.compact(indices, true) as Uint32Array
  263. const gb = ChunkedArray.compact(groups, true) as Float32Array
  264. const tb = ChunkedArray.compact(tcoords, true) as Float32Array
  265. return Text.create(ft, cb, mb, db, ib, gb, tb, indices.elementCount / 2, text)
  266. }
  267. }
  268. }
  269. }