text_buffer_manager.cpp 26 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901
  1. /*
  2. * Copyright 2013 Jeremie Roy. All rights reserved.
  3. * License: http://www.opensource.org/licenses/BSD-2-Clause
  4. */
  5. #include "../common.h"
  6. #include <bgfx.h>
  7. #include <stddef.h> // offsetof
  8. #include <memory.h> // memcpy
  9. #include <wchar.h> // wcslen
  10. #include "text_buffer_manager.h"
  11. #include "utf8.h"
  12. #include "../cube_atlas.h"
  13. #include "vs_font_basic.bin.h"
  14. #include "fs_font_basic.bin.h"
  15. #include "vs_font_distance_field.bin.h"
  16. #include "fs_font_distance_field.bin.h"
  17. #include "vs_font_distance_field_subpixel.bin.h"
  18. #include "fs_font_distance_field_subpixel.bin.h"
  19. #define MAX_BUFFERED_CHARACTERS (8192 - 5)
  20. class TextBuffer
  21. {
  22. public:
  23. /// TextBuffer is bound to a fontManager for glyph retrieval
  24. /// @remark the ownership of the manager is not taken
  25. TextBuffer(FontManager* _fontManager);
  26. ~TextBuffer();
  27. void setStyle(uint32_t _flags = STYLE_NORMAL)
  28. {
  29. m_styleFlags = _flags;
  30. }
  31. void setTextColor(uint32_t _rgba = 0x000000FF)
  32. {
  33. m_textColor = toABGR(_rgba);
  34. }
  35. void setBackgroundColor(uint32_t _rgba = 0x000000FF)
  36. {
  37. m_backgroundColor = toABGR(_rgba);
  38. }
  39. void setOverlineColor(uint32_t _rgba = 0x000000FF)
  40. {
  41. m_overlineColor = toABGR(_rgba);
  42. }
  43. void setUnderlineColor(uint32_t _rgba = 0x000000FF)
  44. {
  45. m_underlineColor = toABGR(_rgba);
  46. }
  47. void setStrikeThroughColor(uint32_t _rgba = 0x000000FF)
  48. {
  49. m_strikeThroughColor = toABGR(_rgba);
  50. }
  51. void setPenPosition(float _x, float _y)
  52. {
  53. m_penX = _x; m_penY = _y;
  54. }
  55. /// Append an ASCII/utf-8 string to the buffer using current pen
  56. /// position and color.
  57. void appendText(FontHandle _fontHandle, const char* _string, const char* _end = NULL);
  58. /// Append a wide char unicode string to the buffer using current pen
  59. /// position and color.
  60. void appendText(FontHandle _fontHandle, const wchar_t* _string, const wchar_t* _end = NULL);
  61. /// Append a whole face of the atlas cube, mostly used for debugging
  62. /// and visualizing atlas.
  63. void appendAtlasFace(uint16_t _faceIndex);
  64. /// Clear the text buffer and reset its state (pen/color)
  65. void clearTextBuffer();
  66. /// Get pointer to the vertex buffer to submit it to the graphic card.
  67. const uint8_t* getVertexBuffer()
  68. {
  69. return (uint8_t*) m_vertexBuffer;
  70. }
  71. /// Number of vertex in the vertex buffer.
  72. uint32_t getVertexCount() const
  73. {
  74. return m_vertexCount;
  75. }
  76. /// Size in bytes of a vertex.
  77. uint32_t getVertexSize() const
  78. {
  79. return sizeof(TextVertex);
  80. }
  81. /// get a pointer to the index buffer to submit it to the graphic
  82. const uint16_t* getIndexBuffer() const
  83. {
  84. return m_indexBuffer;
  85. }
  86. /// number of index in the index buffer
  87. uint32_t getIndexCount() const
  88. {
  89. return m_indexCount;
  90. }
  91. /// Size in bytes of an index.
  92. uint32_t getIndexSize() const
  93. {
  94. return sizeof(uint16_t);
  95. }
  96. uint32_t getTextColor() const
  97. {
  98. return toABGR(m_textColor);
  99. }
  100. TextRectangle getRectangle() const
  101. {
  102. return m_rectangle;
  103. }
  104. private:
  105. void appendGlyph(FontHandle _handle, CodePoint _codePoint);
  106. void verticalCenterLastLine(float _txtDecalY, float _top, float _bottom);
  107. static uint32_t toABGR(uint32_t _rgba)
  108. {
  109. return ( ( (_rgba >> 0) & 0xff) << 24)
  110. | ( ( (_rgba >> 8) & 0xff) << 16)
  111. | ( ( (_rgba >> 16) & 0xff) << 8)
  112. | ( ( (_rgba >> 24) & 0xff) << 0)
  113. ;
  114. }
  115. uint32_t m_styleFlags;
  116. // color states
  117. uint32_t m_textColor;
  118. uint32_t m_backgroundColor;
  119. uint32_t m_overlineColor;
  120. uint32_t m_underlineColor;
  121. uint32_t m_strikeThroughColor;
  122. //position states
  123. float m_penX;
  124. float m_penY;
  125. float m_originX;
  126. float m_originY;
  127. float m_lineAscender;
  128. float m_lineDescender;
  129. float m_lineGap;
  130. TextRectangle m_rectangle;
  131. FontManager* m_fontManager;
  132. void setVertex(uint32_t _i, float _x, float _y, uint32_t _rgba, uint8_t _style = STYLE_NORMAL)
  133. {
  134. m_vertexBuffer[_i].x = _x;
  135. m_vertexBuffer[_i].y = _y;
  136. m_vertexBuffer[_i].rgba = _rgba;
  137. m_styleBuffer[_i] = _style;
  138. }
  139. struct TextVertex
  140. {
  141. float x, y;
  142. int16_t u, v, w, t;
  143. uint32_t rgba;
  144. };
  145. TextVertex* m_vertexBuffer;
  146. uint16_t* m_indexBuffer;
  147. uint8_t* m_styleBuffer;
  148. uint32_t m_indexCount;
  149. uint32_t m_lineStartIndex;
  150. uint16_t m_vertexCount;
  151. };
  152. TextBuffer::TextBuffer(FontManager* _fontManager)
  153. : m_styleFlags(STYLE_NORMAL)
  154. , m_textColor(0xffffffff)
  155. , m_backgroundColor(0xffffffff)
  156. , m_overlineColor(0xffffffff)
  157. , m_underlineColor(0xffffffff)
  158. , m_strikeThroughColor(0xffffffff)
  159. , m_penX(0)
  160. , m_penY(0)
  161. , m_originX(0)
  162. , m_originY(0)
  163. , m_lineAscender(0)
  164. , m_lineDescender(0)
  165. , m_lineGap(0)
  166. , m_fontManager(_fontManager)
  167. , m_vertexBuffer(new TextVertex[MAX_BUFFERED_CHARACTERS * 4])
  168. , m_indexBuffer(new uint16_t[MAX_BUFFERED_CHARACTERS * 6])
  169. , m_styleBuffer(new uint8_t[MAX_BUFFERED_CHARACTERS * 4])
  170. , m_indexCount(0)
  171. , m_lineStartIndex(0)
  172. , m_vertexCount(0)
  173. {
  174. m_rectangle.width = 0;
  175. m_rectangle.height = 0;
  176. }
  177. TextBuffer::~TextBuffer()
  178. {
  179. delete [] m_vertexBuffer;
  180. delete [] m_indexBuffer;
  181. delete [] m_styleBuffer;
  182. }
  183. void TextBuffer::appendText(FontHandle _fontHandle, const char* _string, const char* _end)
  184. {
  185. if (m_vertexCount == 0)
  186. {
  187. m_originX = m_penX;
  188. m_originY = m_penY;
  189. m_lineDescender = 0;
  190. m_lineAscender = 0;
  191. m_lineGap = 0;
  192. }
  193. CodePoint codepoint = 0;
  194. uint32_t state = 0;
  195. for (; *_string && _string<_end ; ++_string)
  196. {
  197. if (utf8_decode(&state, (uint32_t*)&codepoint, *_string) == UTF8_ACCEPT )
  198. {
  199. appendGlyph(_fontHandle, codepoint);
  200. }
  201. }
  202. BX_CHECK(state == UTF8_ACCEPT, "The string is not well-formed");
  203. }
  204. void TextBuffer::appendText(FontHandle _fontHandle, const wchar_t* _string, const wchar_t* _end)
  205. {
  206. if (m_vertexCount == 0)
  207. {
  208. m_originX = m_penX;
  209. m_originY = m_penY;
  210. m_lineDescender = 0;
  211. m_lineAscender = 0;
  212. m_lineGap = 0;
  213. }
  214. if (_end == NULL)
  215. {
  216. _end = _string + (uint32_t) wcslen(_string);
  217. }
  218. BX_CHECK(_end >= _string);
  219. for (const wchar_t* _current = _string; _current < _end; ++_current)
  220. {
  221. uint32_t _codePoint = *_current;
  222. appendGlyph(_fontHandle, _codePoint);
  223. }
  224. }
  225. void TextBuffer::appendAtlasFace(uint16_t _faceIndex)
  226. {
  227. if( m_vertexCount/4 >= MAX_BUFFERED_CHARACTERS)
  228. {
  229. return;
  230. }
  231. float x0 = m_penX;
  232. float y0 = m_penY;
  233. float x1 = x0 + (float)m_fontManager->getAtlas()->getTextureSize();
  234. float y1 = y0 + (float)m_fontManager->getAtlas()->getTextureSize();
  235. m_fontManager->getAtlas()->packFaceLayerUV(_faceIndex
  236. , (uint8_t*)m_vertexBuffer
  237. , sizeof(TextVertex) * m_vertexCount + offsetof(TextVertex, u)
  238. , sizeof(TextVertex)
  239. );
  240. setVertex(m_vertexCount + 0, x0, y0, m_backgroundColor);
  241. setVertex(m_vertexCount + 1, x0, y1, m_backgroundColor);
  242. setVertex(m_vertexCount + 2, x1, y1, m_backgroundColor);
  243. setVertex(m_vertexCount + 3, x1, y0, m_backgroundColor);
  244. m_indexBuffer[m_indexCount + 0] = m_vertexCount + 0;
  245. m_indexBuffer[m_indexCount + 1] = m_vertexCount + 1;
  246. m_indexBuffer[m_indexCount + 2] = m_vertexCount + 2;
  247. m_indexBuffer[m_indexCount + 3] = m_vertexCount + 0;
  248. m_indexBuffer[m_indexCount + 4] = m_vertexCount + 2;
  249. m_indexBuffer[m_indexCount + 5] = m_vertexCount + 3;
  250. m_vertexCount += 4;
  251. m_indexCount += 6;
  252. }
  253. void TextBuffer::clearTextBuffer()
  254. {
  255. m_penX = 0;
  256. m_penY = 0;
  257. m_originX = 0;
  258. m_originY = 0;
  259. m_vertexCount = 0;
  260. m_indexCount = 0;
  261. m_lineStartIndex = 0;
  262. m_lineAscender = 0;
  263. m_lineDescender = 0;
  264. m_lineGap = 0;
  265. m_rectangle.width = 0;
  266. m_rectangle.height = 0;
  267. }
  268. void TextBuffer::appendGlyph(FontHandle _handle, CodePoint _codePoint)
  269. {
  270. const GlyphInfo* glyph = m_fontManager->getGlyphInfo(_handle, _codePoint);
  271. BX_WARN(NULL != glyph, "Glyph not found (font handle %d, code point %d)", _handle.idx, _codePoint);
  272. if (NULL == glyph)
  273. {
  274. return;
  275. }
  276. const FontInfo& font = m_fontManager->getFontInfo(_handle);
  277. if( m_vertexCount/4 >= MAX_BUFFERED_CHARACTERS)
  278. {
  279. return;
  280. }
  281. if (_codePoint == L'\n')
  282. {
  283. m_penX = m_originX;
  284. m_penY += m_lineGap + m_lineAscender -m_lineDescender;
  285. m_lineGap = font.lineGap;
  286. m_lineDescender = font.descender;
  287. m_lineAscender = font.ascender;
  288. m_lineStartIndex = m_vertexCount;
  289. return;
  290. }
  291. //is there a change of font size that require the text on the left to be centered again ?
  292. if (font.ascender > m_lineAscender
  293. || (font.descender < m_lineDescender) )
  294. {
  295. if (font.descender < m_lineDescender)
  296. {
  297. m_lineDescender = font.descender;
  298. m_lineGap = font.lineGap;
  299. }
  300. float txtDecals = (font.ascender - m_lineAscender);
  301. m_lineAscender = font.ascender;
  302. m_lineGap = font.lineGap;
  303. verticalCenterLastLine( (txtDecals), (m_penY - m_lineAscender), (m_penY + m_lineAscender - m_lineDescender + m_lineGap) );
  304. }
  305. float kerning = 0 * font.scale;
  306. m_penX += kerning;
  307. const GlyphInfo& blackGlyph = m_fontManager->getBlackGlyph();
  308. const Atlas* atlas = m_fontManager->getAtlas();
  309. if (m_styleFlags & STYLE_BACKGROUND
  310. && m_backgroundColor & 0xFF000000)
  311. {
  312. float x0 = (m_penX - kerning);
  313. float y0 = (m_penY);
  314. float x1 = ( (float)x0 + (glyph->advance_x) );
  315. float y1 = (m_penY + m_lineAscender - m_lineDescender + m_lineGap);
  316. atlas->packUV(blackGlyph.regionIndex
  317. , (uint8_t*)m_vertexBuffer
  318. , sizeof(TextVertex) * m_vertexCount + offsetof(TextVertex, u)
  319. , sizeof(TextVertex)
  320. );
  321. const uint16_t vertexCount = m_vertexCount;
  322. setVertex(vertexCount + 0, x0, y0, m_backgroundColor, STYLE_BACKGROUND);
  323. setVertex(vertexCount + 1, x0, y1, m_backgroundColor, STYLE_BACKGROUND);
  324. setVertex(vertexCount + 2, x1, y1, m_backgroundColor, STYLE_BACKGROUND);
  325. setVertex(vertexCount + 3, x1, y0, m_backgroundColor, STYLE_BACKGROUND);
  326. m_indexBuffer[m_indexCount + 0] = vertexCount + 0;
  327. m_indexBuffer[m_indexCount + 1] = vertexCount + 1;
  328. m_indexBuffer[m_indexCount + 2] = vertexCount + 2;
  329. m_indexBuffer[m_indexCount + 3] = vertexCount + 0;
  330. m_indexBuffer[m_indexCount + 4] = vertexCount + 2;
  331. m_indexBuffer[m_indexCount + 5] = vertexCount + 3;
  332. m_vertexCount += 4;
  333. m_indexCount += 6;
  334. }
  335. if (m_styleFlags & STYLE_UNDERLINE
  336. && m_underlineColor & 0xFF000000)
  337. {
  338. float x0 = (m_penX - kerning);
  339. float y0 = (m_penY + m_lineAscender - m_lineDescender * 0.5f);
  340. float x1 = ( (float)x0 + (glyph->advance_x) );
  341. float y1 = y0 + font.underlineThickness;
  342. atlas->packUV(blackGlyph.regionIndex
  343. , (uint8_t*)m_vertexBuffer
  344. , sizeof(TextVertex) * m_vertexCount + offsetof(TextVertex, u)
  345. , sizeof(TextVertex)
  346. );
  347. setVertex(m_vertexCount + 0, x0, y0, m_underlineColor, STYLE_UNDERLINE);
  348. setVertex(m_vertexCount + 1, x0, y1, m_underlineColor, STYLE_UNDERLINE);
  349. setVertex(m_vertexCount + 2, x1, y1, m_underlineColor, STYLE_UNDERLINE);
  350. setVertex(m_vertexCount + 3, x1, y0, m_underlineColor, STYLE_UNDERLINE);
  351. m_indexBuffer[m_indexCount + 0] = m_vertexCount + 0;
  352. m_indexBuffer[m_indexCount + 1] = m_vertexCount + 1;
  353. m_indexBuffer[m_indexCount + 2] = m_vertexCount + 2;
  354. m_indexBuffer[m_indexCount + 3] = m_vertexCount + 0;
  355. m_indexBuffer[m_indexCount + 4] = m_vertexCount + 2;
  356. m_indexBuffer[m_indexCount + 5] = m_vertexCount + 3;
  357. m_vertexCount += 4;
  358. m_indexCount += 6;
  359. }
  360. if (m_styleFlags & STYLE_OVERLINE
  361. && m_overlineColor & 0xFF000000)
  362. {
  363. float x0 = (m_penX - kerning);
  364. float y0 = (m_penY);
  365. float x1 = ( (float)x0 + (glyph->advance_x) );
  366. float y1 = y0 + font.underlineThickness;
  367. m_fontManager->getAtlas()->packUV(blackGlyph.regionIndex
  368. , (uint8_t*)m_vertexBuffer
  369. , sizeof(TextVertex) * m_vertexCount + offsetof(TextVertex, u)
  370. , sizeof(TextVertex)
  371. );
  372. setVertex(m_vertexCount + 0, x0, y0, m_overlineColor, STYLE_OVERLINE);
  373. setVertex(m_vertexCount + 1, x0, y1, m_overlineColor, STYLE_OVERLINE);
  374. setVertex(m_vertexCount + 2, x1, y1, m_overlineColor, STYLE_OVERLINE);
  375. setVertex(m_vertexCount + 3, x1, y0, m_overlineColor, STYLE_OVERLINE);
  376. m_indexBuffer[m_indexCount + 0] = m_vertexCount + 0;
  377. m_indexBuffer[m_indexCount + 1] = m_vertexCount + 1;
  378. m_indexBuffer[m_indexCount + 2] = m_vertexCount + 2;
  379. m_indexBuffer[m_indexCount + 3] = m_vertexCount + 0;
  380. m_indexBuffer[m_indexCount + 4] = m_vertexCount + 2;
  381. m_indexBuffer[m_indexCount + 5] = m_vertexCount + 3;
  382. m_vertexCount += 4;
  383. m_indexCount += 6;
  384. }
  385. if (m_styleFlags & STYLE_STRIKE_THROUGH
  386. && m_strikeThroughColor & 0xFF000000)
  387. {
  388. float x0 = (m_penX - kerning);
  389. float y0 = (m_penY + 0.666667f * font.ascender);
  390. float x1 = ( (float)x0 + (glyph->advance_x) );
  391. float y1 = y0 + font.underlineThickness;
  392. atlas->packUV(blackGlyph.regionIndex
  393. , (uint8_t*)m_vertexBuffer
  394. , sizeof(TextVertex) * m_vertexCount + offsetof(TextVertex, u)
  395. , sizeof(TextVertex)
  396. );
  397. setVertex(m_vertexCount + 0, x0, y0, m_strikeThroughColor, STYLE_STRIKE_THROUGH);
  398. setVertex(m_vertexCount + 1, x0, y1, m_strikeThroughColor, STYLE_STRIKE_THROUGH);
  399. setVertex(m_vertexCount + 2, x1, y1, m_strikeThroughColor, STYLE_STRIKE_THROUGH);
  400. setVertex(m_vertexCount + 3, x1, y0, m_strikeThroughColor, STYLE_STRIKE_THROUGH);
  401. m_indexBuffer[m_indexCount + 0] = m_vertexCount + 0;
  402. m_indexBuffer[m_indexCount + 1] = m_vertexCount + 1;
  403. m_indexBuffer[m_indexCount + 2] = m_vertexCount + 2;
  404. m_indexBuffer[m_indexCount + 3] = m_vertexCount + 0;
  405. m_indexBuffer[m_indexCount + 4] = m_vertexCount + 2;
  406. m_indexBuffer[m_indexCount + 5] = m_vertexCount + 3;
  407. m_vertexCount += 4;
  408. m_indexCount += 6;
  409. }
  410. float x0 = m_penX + (glyph->offset_x);
  411. float y0 = (m_penY + m_lineAscender + (glyph->offset_y) );
  412. float x1 = (x0 + glyph->width);
  413. float y1 = (y0 + glyph->height);
  414. atlas->packUV(glyph->regionIndex
  415. , (uint8_t*)m_vertexBuffer
  416. , sizeof(TextVertex) * m_vertexCount + offsetof(TextVertex, u)
  417. , sizeof(TextVertex)
  418. );
  419. setVertex(m_vertexCount + 0, x0, y0, m_textColor);
  420. setVertex(m_vertexCount + 1, x0, y1, m_textColor);
  421. setVertex(m_vertexCount + 2, x1, y1, m_textColor);
  422. setVertex(m_vertexCount + 3, x1, y0, m_textColor);
  423. m_indexBuffer[m_indexCount + 0] = m_vertexCount + 0;
  424. m_indexBuffer[m_indexCount + 1] = m_vertexCount + 1;
  425. m_indexBuffer[m_indexCount + 2] = m_vertexCount + 2;
  426. m_indexBuffer[m_indexCount + 3] = m_vertexCount + 0;
  427. m_indexBuffer[m_indexCount + 4] = m_vertexCount + 2;
  428. m_indexBuffer[m_indexCount + 5] = m_vertexCount + 3;
  429. m_vertexCount += 4;
  430. m_indexCount += 6;
  431. m_penX += glyph->advance_x;
  432. if (m_penX > m_rectangle.width)
  433. {
  434. m_rectangle.width = m_penX;
  435. }
  436. if ( (m_penY +m_lineAscender - m_lineDescender+m_lineGap) > m_rectangle.height)
  437. {
  438. m_rectangle.height = (m_penY +m_lineAscender - m_lineDescender+m_lineGap);
  439. }
  440. }
  441. void TextBuffer::verticalCenterLastLine(float _dy, float _top, float _bottom)
  442. {
  443. for (uint32_t ii = m_lineStartIndex; ii < m_vertexCount; ii += 4)
  444. {
  445. if (m_styleBuffer[ii] == STYLE_BACKGROUND)
  446. {
  447. m_vertexBuffer[ii + 0].y = _top;
  448. m_vertexBuffer[ii + 1].y = _bottom;
  449. m_vertexBuffer[ii + 2].y = _bottom;
  450. m_vertexBuffer[ii + 3].y = _top;
  451. }
  452. else
  453. {
  454. m_vertexBuffer[ii + 0].y += _dy;
  455. m_vertexBuffer[ii + 1].y += _dy;
  456. m_vertexBuffer[ii + 2].y += _dy;
  457. m_vertexBuffer[ii + 3].y += _dy;
  458. }
  459. }
  460. }
  461. TextBufferManager::TextBufferManager(FontManager* _fontManager)
  462. : m_fontManager(_fontManager)
  463. {
  464. m_textBuffers = new BufferCache[MAX_TEXT_BUFFER_COUNT];
  465. const bgfx::Memory* vs_font_basic;
  466. const bgfx::Memory* fs_font_basic;
  467. const bgfx::Memory* vs_font_distance_field;
  468. const bgfx::Memory* fs_font_distance_field;
  469. const bgfx::Memory* vs_font_distance_field_subpixel;
  470. const bgfx::Memory* fs_font_distance_field_subpixel;
  471. switch (bgfx::getRendererType() )
  472. {
  473. case bgfx::RendererType::Direct3D9:
  474. vs_font_basic = bgfx::makeRef(vs_font_basic_dx9, sizeof(vs_font_basic_dx9) );
  475. fs_font_basic = bgfx::makeRef(fs_font_basic_dx9, sizeof(fs_font_basic_dx9) );
  476. vs_font_distance_field = bgfx::makeRef(vs_font_distance_field_dx9, sizeof(vs_font_distance_field_dx9) );
  477. fs_font_distance_field = bgfx::makeRef(fs_font_distance_field_dx9, sizeof(fs_font_distance_field_dx9) );
  478. vs_font_distance_field_subpixel = bgfx::makeRef(vs_font_distance_field_subpixel_dx9, sizeof(vs_font_distance_field_subpixel_dx9) );
  479. fs_font_distance_field_subpixel = bgfx::makeRef(fs_font_distance_field_subpixel_dx9, sizeof(fs_font_distance_field_subpixel_dx9) );
  480. break;
  481. case bgfx::RendererType::Direct3D11:
  482. vs_font_basic = bgfx::makeRef(vs_font_basic_dx11, sizeof(vs_font_basic_dx11) );
  483. fs_font_basic = bgfx::makeRef(fs_font_basic_dx11, sizeof(fs_font_basic_dx11) );
  484. vs_font_distance_field = bgfx::makeRef(vs_font_distance_field_dx11, sizeof(vs_font_distance_field_dx11) );
  485. fs_font_distance_field = bgfx::makeRef(fs_font_distance_field_dx11, sizeof(fs_font_distance_field_dx11) );
  486. vs_font_distance_field_subpixel = bgfx::makeRef(vs_font_distance_field_subpixel_dx11, sizeof(vs_font_distance_field_subpixel_dx11) );
  487. fs_font_distance_field_subpixel = bgfx::makeRef(fs_font_distance_field_subpixel_dx11, sizeof(fs_font_distance_field_subpixel_dx11) );
  488. break;
  489. default:
  490. vs_font_basic = bgfx::makeRef(vs_font_basic_glsl, sizeof(vs_font_basic_glsl) );
  491. fs_font_basic = bgfx::makeRef(fs_font_basic_glsl, sizeof(fs_font_basic_glsl) );
  492. vs_font_distance_field = bgfx::makeRef(vs_font_distance_field_glsl, sizeof(vs_font_distance_field_glsl) );
  493. fs_font_distance_field = bgfx::makeRef(fs_font_distance_field_glsl, sizeof(fs_font_distance_field_glsl) );
  494. vs_font_distance_field_subpixel = bgfx::makeRef(vs_font_distance_field_subpixel_glsl, sizeof(vs_font_distance_field_subpixel_glsl) );
  495. fs_font_distance_field_subpixel = bgfx::makeRef(fs_font_distance_field_subpixel_glsl, sizeof(fs_font_distance_field_subpixel_glsl) );
  496. break;
  497. }
  498. m_basicProgram = bgfx::createProgram(
  499. bgfx::createShader(vs_font_basic)
  500. , bgfx::createShader(fs_font_basic)
  501. , true
  502. );
  503. m_distanceProgram = bgfx::createProgram(
  504. bgfx::createShader(vs_font_distance_field)
  505. , bgfx::createShader(fs_font_distance_field)
  506. , true
  507. );
  508. m_distanceSubpixelProgram = bgfx::createProgram(
  509. bgfx::createShader(vs_font_distance_field_subpixel)
  510. , bgfx::createShader(fs_font_distance_field_subpixel)
  511. , true
  512. );
  513. m_vertexDecl
  514. .begin()
  515. .add(bgfx::Attrib::Position, 2, bgfx::AttribType::Float)
  516. .add(bgfx::Attrib::TexCoord0, 4, bgfx::AttribType::Int16, true)
  517. .add(bgfx::Attrib::Color0, 4, bgfx::AttribType::Uint8, true)
  518. .end();
  519. u_texColor = bgfx::createUniform("u_texColor", bgfx::UniformType::Uniform1iv);
  520. }
  521. TextBufferManager::~TextBufferManager()
  522. {
  523. BX_CHECK(m_textBufferHandles.getNumHandles() == 0, "All the text buffers must be destroyed before destroying the manager");
  524. delete [] m_textBuffers;
  525. bgfx::destroyUniform(u_texColor);
  526. bgfx::destroyProgram(m_basicProgram);
  527. bgfx::destroyProgram(m_distanceProgram);
  528. bgfx::destroyProgram(m_distanceSubpixelProgram);
  529. }
  530. TextBufferHandle TextBufferManager::createTextBuffer(uint32_t _type, BufferType::Enum _bufferType)
  531. {
  532. uint16_t textIdx = m_textBufferHandles.alloc();
  533. BufferCache& bc = m_textBuffers[textIdx];
  534. bc.textBuffer = new TextBuffer(m_fontManager);
  535. bc.fontType = _type;
  536. bc.bufferType = _bufferType;
  537. bc.indexBufferHandleIdx = bgfx::invalidHandle;
  538. bc.vertexBufferHandleIdx = bgfx::invalidHandle;
  539. TextBufferHandle ret = {textIdx};
  540. return ret;
  541. }
  542. void TextBufferManager::destroyTextBuffer(TextBufferHandle _handle)
  543. {
  544. BX_CHECK(bgfx::isValid(_handle), "Invalid handle used");
  545. BufferCache& bc = m_textBuffers[_handle.idx];
  546. m_textBufferHandles.free(_handle.idx);
  547. delete bc.textBuffer;
  548. bc.textBuffer = NULL;
  549. if (bc.vertexBufferHandleIdx == bgfx::invalidHandle)
  550. {
  551. return;
  552. }
  553. switch (bc.bufferType)
  554. {
  555. case BufferType::Static:
  556. {
  557. bgfx::IndexBufferHandle ibh;
  558. bgfx::VertexBufferHandle vbh;
  559. ibh.idx = bc.indexBufferHandleIdx;
  560. vbh.idx = bc.vertexBufferHandleIdx;
  561. bgfx::destroyIndexBuffer(ibh);
  562. bgfx::destroyVertexBuffer(vbh);
  563. }
  564. break;
  565. case BufferType::Dynamic:
  566. bgfx::DynamicIndexBufferHandle ibh;
  567. bgfx::DynamicVertexBufferHandle vbh;
  568. ibh.idx = bc.indexBufferHandleIdx;
  569. vbh.idx = bc.vertexBufferHandleIdx;
  570. bgfx::destroyDynamicIndexBuffer(ibh);
  571. bgfx::destroyDynamicVertexBuffer(vbh);
  572. break;
  573. case BufferType::Transient: // destroyed every frame
  574. break;
  575. }
  576. }
  577. void TextBufferManager::submitTextBuffer(TextBufferHandle _handle, uint8_t _id, int32_t _depth)
  578. {
  579. BX_CHECK(bgfx::isValid(_handle), "Invalid handle used");
  580. BufferCache& bc = m_textBuffers[_handle.idx];
  581. uint32_t indexSize = bc.textBuffer->getIndexCount() * bc.textBuffer->getIndexSize();
  582. uint32_t vertexSize = bc.textBuffer->getVertexCount() * bc.textBuffer->getVertexSize();
  583. if (0 == indexSize || 0 == vertexSize)
  584. {
  585. return;
  586. }
  587. bgfx::setTexture(0, u_texColor, m_fontManager->getAtlas()->getTextureHandle() );
  588. switch (bc.fontType)
  589. {
  590. case FONT_TYPE_ALPHA:
  591. bgfx::setProgram(m_basicProgram);
  592. bgfx::setState(0
  593. | BGFX_STATE_RGB_WRITE
  594. | BGFX_STATE_BLEND_FUNC(BGFX_STATE_BLEND_SRC_ALPHA, BGFX_STATE_BLEND_INV_SRC_ALPHA)
  595. );
  596. break;
  597. case FONT_TYPE_DISTANCE:
  598. bgfx::setProgram(m_distanceProgram);
  599. bgfx::setState(0
  600. | BGFX_STATE_RGB_WRITE
  601. | BGFX_STATE_BLEND_FUNC(BGFX_STATE_BLEND_SRC_ALPHA, BGFX_STATE_BLEND_INV_SRC_ALPHA)
  602. );
  603. break;
  604. case FONT_TYPE_DISTANCE_SUBPIXEL:
  605. bgfx::setProgram(m_distanceSubpixelProgram);
  606. bgfx::setState(0
  607. | BGFX_STATE_RGB_WRITE
  608. | BGFX_STATE_BLEND_FUNC(BGFX_STATE_BLEND_FACTOR, BGFX_STATE_BLEND_INV_SRC_COLOR)
  609. , bc.textBuffer->getTextColor()
  610. );
  611. break;
  612. }
  613. switch (bc.bufferType)
  614. {
  615. case BufferType::Static:
  616. {
  617. bgfx::IndexBufferHandle ibh;
  618. bgfx::VertexBufferHandle vbh;
  619. if (bgfx::invalidHandle == bc.vertexBufferHandleIdx)
  620. {
  621. ibh = bgfx::createIndexBuffer(
  622. bgfx::copy(bc.textBuffer->getIndexBuffer(), indexSize)
  623. );
  624. vbh = bgfx::createVertexBuffer(
  625. bgfx::copy(bc.textBuffer->getVertexBuffer(), vertexSize)
  626. , m_vertexDecl
  627. );
  628. bc.vertexBufferHandleIdx = vbh.idx;
  629. bc.indexBufferHandleIdx = ibh.idx;
  630. }
  631. else
  632. {
  633. vbh.idx = bc.vertexBufferHandleIdx;
  634. ibh.idx = bc.indexBufferHandleIdx;
  635. }
  636. bgfx::setVertexBuffer(vbh, 0, bc.textBuffer->getVertexCount() );
  637. bgfx::setIndexBuffer(ibh, bc.textBuffer->getIndexCount() );
  638. }
  639. break;
  640. case BufferType::Dynamic:
  641. {
  642. bgfx::DynamicIndexBufferHandle ibh;
  643. bgfx::DynamicVertexBufferHandle vbh;
  644. if (bgfx::invalidHandle == bc.vertexBufferHandleIdx )
  645. {
  646. ibh = bgfx::createDynamicIndexBuffer(
  647. bgfx::copy(bc.textBuffer->getIndexBuffer(), indexSize)
  648. );
  649. vbh = bgfx::createDynamicVertexBuffer(
  650. bgfx::copy(bc.textBuffer->getVertexBuffer(), vertexSize)
  651. , m_vertexDecl
  652. );
  653. bc.indexBufferHandleIdx = ibh.idx;
  654. bc.vertexBufferHandleIdx = vbh.idx;
  655. }
  656. else
  657. {
  658. ibh.idx = bc.indexBufferHandleIdx;
  659. vbh.idx = bc.vertexBufferHandleIdx;
  660. bgfx::updateDynamicIndexBuffer(ibh
  661. , bgfx::copy(bc.textBuffer->getIndexBuffer(), indexSize)
  662. );
  663. bgfx::updateDynamicVertexBuffer(vbh
  664. , bgfx::copy(bc.textBuffer->getVertexBuffer(), vertexSize)
  665. );
  666. }
  667. bgfx::setVertexBuffer(vbh, bc.textBuffer->getVertexCount() );
  668. bgfx::setIndexBuffer(ibh, bc.textBuffer->getIndexCount() );
  669. }
  670. break;
  671. case BufferType::Transient:
  672. {
  673. bgfx::TransientIndexBuffer tib;
  674. bgfx::TransientVertexBuffer tvb;
  675. bgfx::allocTransientIndexBuffer(&tib, bc.textBuffer->getIndexCount() );
  676. bgfx::allocTransientVertexBuffer(&tvb, bc.textBuffer->getVertexCount(), m_vertexDecl);
  677. memcpy(tib.data, bc.textBuffer->getIndexBuffer(), indexSize);
  678. memcpy(tvb.data, bc.textBuffer->getVertexBuffer(), vertexSize);
  679. bgfx::setVertexBuffer(&tvb, 0, bc.textBuffer->getVertexCount() );
  680. bgfx::setIndexBuffer(&tib, 0, bc.textBuffer->getIndexCount() );
  681. }
  682. break;
  683. }
  684. bgfx::submit(_id, _depth);
  685. }
  686. void TextBufferManager::setStyle(TextBufferHandle _handle, uint32_t _flags)
  687. {
  688. BX_CHECK(bgfx::isValid(_handle), "Invalid handle used");
  689. BufferCache& bc = m_textBuffers[_handle.idx];
  690. bc.textBuffer->setStyle(_flags);
  691. }
  692. void TextBufferManager::setTextColor(TextBufferHandle _handle, uint32_t _rgba)
  693. {
  694. BX_CHECK(bgfx::isValid(_handle), "Invalid handle used");
  695. BufferCache& bc = m_textBuffers[_handle.idx];
  696. bc.textBuffer->setTextColor(_rgba);
  697. }
  698. void TextBufferManager::setBackgroundColor(TextBufferHandle _handle, uint32_t _rgba)
  699. {
  700. BX_CHECK(bgfx::isValid(_handle), "Invalid handle used");
  701. BufferCache& bc = m_textBuffers[_handle.idx];
  702. bc.textBuffer->setBackgroundColor(_rgba);
  703. }
  704. void TextBufferManager::setOverlineColor(TextBufferHandle _handle, uint32_t _rgba)
  705. {
  706. BX_CHECK(bgfx::isValid(_handle), "Invalid handle used");
  707. BufferCache& bc = m_textBuffers[_handle.idx];
  708. bc.textBuffer->setOverlineColor(_rgba);
  709. }
  710. void TextBufferManager::setUnderlineColor(TextBufferHandle _handle, uint32_t _rgba)
  711. {
  712. BX_CHECK(bgfx::isValid(_handle), "Invalid handle used");
  713. BufferCache& bc = m_textBuffers[_handle.idx];
  714. bc.textBuffer->setUnderlineColor(_rgba);
  715. }
  716. void TextBufferManager::setStrikeThroughColor(TextBufferHandle _handle, uint32_t _rgba)
  717. {
  718. BX_CHECK(bgfx::isValid(_handle), "Invalid handle used");
  719. BufferCache& bc = m_textBuffers[_handle.idx];
  720. bc.textBuffer->setStrikeThroughColor(_rgba);
  721. }
  722. void TextBufferManager::setPenPosition(TextBufferHandle _handle, float _x, float _y)
  723. {
  724. BX_CHECK(bgfx::isValid(_handle), "Invalid handle used");
  725. BufferCache& bc = m_textBuffers[_handle.idx];
  726. bc.textBuffer->setPenPosition(_x, _y);
  727. }
  728. void TextBufferManager::appendText(TextBufferHandle _handle, FontHandle _fontHandle, const char* _string, const char* _end)
  729. {
  730. BX_CHECK(bgfx::isValid(_handle), "Invalid handle used");
  731. BufferCache& bc = m_textBuffers[_handle.idx];
  732. bc.textBuffer->appendText(_fontHandle, _string, _end);
  733. }
  734. void TextBufferManager::appendText(TextBufferHandle _handle, FontHandle _fontHandle, const wchar_t* _string, const wchar_t* _end)
  735. {
  736. BX_CHECK(bgfx::isValid(_handle), "Invalid handle used");
  737. BufferCache& bc = m_textBuffers[_handle.idx];
  738. bc.textBuffer->appendText(_fontHandle, _string, _end);
  739. }
  740. void TextBufferManager::appendAtlasFace(TextBufferHandle _handle, uint16_t _faceIndex)
  741. {
  742. BX_CHECK(bgfx::isValid(_handle), "Invalid handle used");
  743. BufferCache& bc = m_textBuffers[_handle.idx];
  744. bc.textBuffer->appendAtlasFace(_faceIndex);
  745. }
  746. void TextBufferManager::clearTextBuffer(TextBufferHandle _handle)
  747. {
  748. BX_CHECK(bgfx::isValid(_handle), "Invalid handle used");
  749. BufferCache& bc = m_textBuffers[_handle.idx];
  750. bc.textBuffer->clearTextBuffer();
  751. }
  752. TextRectangle TextBufferManager::getRectangle(TextBufferHandle _handle) const
  753. {
  754. BX_CHECK(bgfx::isValid(_handle), "Invalid handle used");
  755. BufferCache& bc = m_textBuffers[_handle.idx];
  756. return bc.textBuffer->getRectangle();
  757. }