gds_font.c 9.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330
  1. /**
  2. * Copyright (c) 2017-2018 Tara Keeling
  3. *
  4. * This software is released under the MIT License.
  5. * https://opensource.org/licenses/MIT
  6. */
  7. #include <stdio.h>
  8. #include <string.h>
  9. #include <stdint.h>
  10. #include <math.h>
  11. #include "gds_private.h"
  12. #include "gds.h"
  13. #include "gds_font.h"
  14. #include "gds_draw.h"
  15. #include "gds_err.h"
  16. #include "esp_spiffs.h"
  17. #include "esp_log.h"
  18. #include "esp_heap_caps.h"
  19. #include "tools.h"
  20. static const char * TAG = "gds_font";
  21. struct GDS_FontDef * Font_droid_sans_fallback_11x13 = NULL;
  22. struct GDS_FontDef * Font_line_1 = NULL;
  23. struct GDS_FontDef * Font_line_2 = NULL;
  24. // struct GDS_FontDef * Font_droid_sans_fallback_15x17 = NULL;
  25. // struct GDS_FontDef * Font_droid_sans_fallback_24x28 = NULL;
  26. // struct GDS_FontDef * Font_droid_sans_mono_7x13 = NULL;
  27. // struct GDS_FontDef * Font_droid_sans_mono_13x24 = NULL;
  28. // struct GDS_FontDef * Font_droid_sans_mono_16x31 = NULL;
  29. // struct GDS_FontDef * Font_liberation_mono_9x15 = NULL;
  30. // struct GDS_FontDef * Font_liberation_mono_13x21 = NULL;
  31. // struct GDS_FontDef * Font_liberation_mono_17x30 = NULL;
  32. // struct GDS_FontDef * Font_Tarable7Seg_16x32 = NULL;
  33. // struct GDS_FontDef * Font_Tarable7Seg_32x64 = NULL;
  34. static bool LoadFont(struct GDS_FontDef ** fontPtr, const char * fileName){
  35. if(!fontPtr){
  36. ESP_LOGE(TAG, "Invalid pointer for LoadFont");
  37. return false;
  38. }
  39. char font_file_name[CONFIG_SPIFFS_OBJ_NAME_LEN+1]={0};
  40. snprintf(font_file_name,sizeof(font_file_name),"/spiffs/fonts/%s",fileName);
  41. // Allocate DMA-capable memory for the font
  42. struct GDS_FontDef* loadedFont = load_file_dma(NULL,font_file_name);
  43. // Check if allocation succeeded
  44. if (loadedFont == NULL) {
  45. ESP_LOGE(TAG, "Failed to load font");
  46. return false;
  47. }
  48. // Update the pointer
  49. *fontPtr = loadedFont;
  50. ESP_LOGI(TAG, "Successfully loaded font: %s", fileName);
  51. return true;
  52. }
  53. bool gds_init_fonts() {
  54. bool success = true;
  55. // Load the Font_droid_sans_fallback_11x13
  56. if (!LoadFont(&Font_droid_sans_fallback_11x13, "droid_sans_fb_11x13.bin")) {
  57. success = false;
  58. }
  59. // Load the Font_line_1
  60. if (!LoadFont(&Font_line_1, "line_1.bin")) {
  61. success = false;
  62. }
  63. // Load the Font_line_2
  64. if (!LoadFont(&Font_line_2, "line_2.bin")) {
  65. success = false;
  66. }
  67. return success;
  68. }
  69. static int RoundUpFontHeight( const struct GDS_FontDef* Font ) {
  70. int Height = Font->Height;
  71. if ( ( Height % 8 ) != 0 ) {
  72. return ( ( Height + 7 ) / 8 ) * 8;
  73. }
  74. return Height;
  75. }
  76. static const uint8_t* GetCharPtr( const struct GDS_FontDef* Font, char Character ) {
  77. return &Font->FontData[( Character - Font->StartChar ) * ( ( Font->Width * ( RoundUpFontHeight( Font ) / 8 ) ) + 1 )];
  78. }
  79. void GDS_FontDrawChar( struct GDS_Device* Device, char Character, int x, int y, int Color ) {
  80. const uint8_t* GlyphData = NULL;
  81. int GlyphColumnLen = 0;
  82. int CharStartX = 0;
  83. int CharStartY = 0;
  84. int CharWidth = 0;
  85. int CharHeight = 0;
  86. int CharEndX = 0;
  87. int CharEndY = 0;
  88. int OffsetX = 0;
  89. int OffsetY = 0;
  90. int YByte = 0;
  91. int YBit = 0;
  92. int i = 0;
  93. NullCheck( ( GlyphData = GetCharPtr( Device->Font, Character ) ), return );
  94. if ( Character >= Device->Font->StartChar && Character <= Device->Font->EndChar ) {
  95. /* The first byte in the glyph data is the width of the character in pixels, skip over */
  96. GlyphData++;
  97. GlyphColumnLen = RoundUpFontHeight( Device->Font ) / 8;
  98. CharWidth = GDS_FontGetCharWidth( Device, Character );
  99. CharHeight = GDS_FontGetHeight( Device );
  100. CharStartX = x;
  101. CharStartY = y;
  102. CharEndX = CharStartX + CharWidth;
  103. CharEndY = CharStartY + CharHeight;
  104. /* If the character is partially offscreen offset the end by
  105. * distance between (coord) and 0.
  106. */
  107. OffsetX = ( CharStartX < 0 ) ? abs( CharStartX ) : 0;
  108. OffsetY = ( CharStartY < 0 ) ? abs( CharStartY ) : 0;
  109. /* This skips into the proper column within the glyph data */
  110. GlyphData+= ( OffsetX * GlyphColumnLen );
  111. CharStartX+= OffsetX;
  112. CharStartY+= OffsetY;
  113. /* Do not attempt to draw if this character is entirely offscreen */
  114. if ( CharEndX < 0 || CharStartX >= Device->TextWidth || CharEndY < 0 || CharStartY >= Device->Height ) {
  115. ClipDebug( x, y );
  116. return;
  117. }
  118. /* Do not attempt to draw past the end of the screen */
  119. CharEndX = ( CharEndX >= Device->TextWidth ) ? Device->TextWidth - 1 : CharEndX;
  120. CharEndY = ( CharEndY >= Device->Height ) ? Device->Height - 1 : CharEndY;
  121. Device->Dirty = true;
  122. for ( x = CharStartX; x < CharEndX; x++ ) {
  123. for ( y = CharStartY, i = 0; y < CharEndY && i < CharHeight; y++, i++ ) {
  124. YByte = ( i + OffsetY ) / 8;
  125. YBit = ( i + OffsetY ) & 0x07;
  126. if ( GlyphData[ YByte ] & BIT( YBit ) ) {
  127. DrawPixel( Device, x, y, Color );
  128. }
  129. }
  130. GlyphData+= GlyphColumnLen;
  131. }
  132. }
  133. }
  134. const struct GDS_FontDef* GDS_SetFont( struct GDS_Device* Display, const struct GDS_FontDef* Font ) {
  135. const struct GDS_FontDef* OldFont = Display->Font;
  136. Display->FontForceProportional = false;
  137. Display->FontForceMonospace = false;
  138. Display->Font = Font;
  139. return OldFont;
  140. }
  141. void GDS_FontForceProportional( struct GDS_Device* Display, bool Force ) {
  142. Display->FontForceProportional = Force;
  143. }
  144. void GDS_FontForceMonospace( struct GDS_Device* Display, bool Force ) {
  145. Display->FontForceMonospace = Force;
  146. }
  147. int GDS_FontGetWidth( struct GDS_Device* Display ) {
  148. return Display->Font->Width;
  149. }
  150. int GDS_FontGetHeight( struct GDS_Device* Display ) {
  151. return Display->Font->Height;
  152. }
  153. int GDS_FontGetCharWidth( struct GDS_Device* Display, char Character ) {
  154. const uint8_t* CharPtr = NULL;
  155. int Width = 0;
  156. if ( Character >= Display->Font->StartChar && Character <= Display->Font->EndChar ) {
  157. CharPtr = GetCharPtr( Display->Font, Character );
  158. Width = ( Display->Font->Monospace == true ) ? Display->Font->Width : *CharPtr;
  159. if ( Display->FontForceMonospace == true ) {
  160. Width = Display->Font->Width;
  161. }
  162. if ( Display->FontForceProportional == true ) {
  163. Width = *CharPtr;
  164. }
  165. }
  166. return Width;
  167. }
  168. int GDS_FontGetMaxCharsPerRow( struct GDS_Device* Display ) {
  169. return Display->TextWidth / Display->Font->Width;
  170. }
  171. int GDS_FontGetMaxCharsPerColumn( struct GDS_Device* Display ) {
  172. return Display->Height / Display->Font->Height;
  173. }
  174. int GDS_FontGetCharHeight( struct GDS_Device* Display ) {
  175. return Display->Font->Height;
  176. }
  177. int GDS_FontMeasureString( struct GDS_Device* Display, const char* Text ) {
  178. int Width = 0;
  179. int Len = 0;
  180. NullCheck( Text, return 0 );
  181. for ( Len = strlen( Text ); Len >= 0; Len--, Text++ ) {
  182. if ( *Text >= Display->Font->StartChar && *Text <= Display->Font->EndChar ) {
  183. Width+= GDS_FontGetCharWidth( Display, *Text );
  184. }
  185. }
  186. return Width;
  187. }
  188. void GDS_FontDrawString( struct GDS_Device* Display, int x, int y, const char* Text, int Color ) {
  189. int Len = 0;
  190. int i = 0;
  191. NullCheck( Text, return );
  192. for ( Len = strlen( Text ), i = 0; i < Len; i++ ) {
  193. GDS_FontDrawChar( Display, *Text, x, y, Color );
  194. x+= GDS_FontGetCharWidth( Display, *Text );
  195. Text++;
  196. }
  197. }
  198. void GDS_FontDrawAnchoredString( struct GDS_Device* Display, TextAnchor Anchor, const char* Text, int Color ) {
  199. int x = 0;
  200. int y = 0;
  201. NullCheck( Text, return );
  202. GDS_FontGetAnchoredStringCoords( Display, &x, &y, Anchor, Text );
  203. GDS_FontDrawString( Display, x, y, Text, Color );
  204. }
  205. void GDS_FontGetAnchoredStringCoords( struct GDS_Device* Display, int* OutX, int* OutY, TextAnchor Anchor, const char* Text ) {
  206. int StringWidth = 0;
  207. int StringHeight = 0;
  208. NullCheck( OutX, return );
  209. NullCheck( OutY, return );
  210. NullCheck( Text, return );
  211. StringWidth = GDS_FontMeasureString( Display, Text );
  212. StringHeight = GDS_FontGetCharHeight( Display );
  213. switch ( Anchor ) {
  214. case TextAnchor_East: {
  215. *OutY = ( Display->Height / 2 ) - ( StringHeight / 2 );
  216. *OutX = ( Display->TextWidth - StringWidth );
  217. break;
  218. }
  219. case TextAnchor_West: {
  220. *OutY = ( Display->Height / 2 ) - ( StringHeight / 2 );
  221. *OutX = 0;
  222. break;
  223. }
  224. case TextAnchor_North: {
  225. *OutX = ( Display->TextWidth / 2 ) - ( StringWidth / 2 );
  226. *OutY = 0;
  227. break;
  228. }
  229. case TextAnchor_South: {
  230. *OutX = ( Display->TextWidth / 2 ) - ( StringWidth / 2 );
  231. *OutY = ( Display->Height - StringHeight );
  232. break;
  233. }
  234. case TextAnchor_NorthEast: {
  235. *OutX = ( Display->TextWidth - StringWidth );
  236. *OutY = 0;
  237. break;
  238. }
  239. case TextAnchor_NorthWest: {
  240. *OutY = 0;
  241. *OutX = 0;
  242. break;
  243. }
  244. case TextAnchor_SouthEast: {
  245. *OutY = ( Display->Height - StringHeight );
  246. *OutX = ( Display->TextWidth - StringWidth );
  247. break;
  248. }
  249. case TextAnchor_SouthWest: {
  250. *OutY = ( Display->Height - StringHeight );
  251. *OutX = 0;
  252. break;
  253. }
  254. case TextAnchor_Center: {
  255. *OutY = ( Display->Height / 2 ) - ( StringHeight / 2 );
  256. *OutX = ( Display->TextWidth / 2 ) - ( StringWidth / 2 );
  257. break;
  258. }
  259. default: {
  260. *OutX = 128;
  261. *OutY = 64;
  262. break;
  263. }
  264. };
  265. }