1 module gfm.sdl2.sdlttf; 2 3 import std..string; 4 5 import derelict.sdl2.sdl, 6 derelict.sdl2.ttf, 7 derelict.util.exception; 8 9 static if( __VERSION__ >= 2067 ) 10 import std.experimental.logger; 11 else 12 import std.historical.logger; 13 14 import gfm.sdl2.sdl, 15 gfm.sdl2.surface; 16 17 /// SDL_ttf library wrapper. 18 final class SDLTTF 19 { 20 public 21 { 22 /// Loads the SDL_ttf library. 23 /// Throws: $(D SDL2Exception) on error. 24 this(SDL2 sdl2) 25 { 26 _sdl2 = sdl2; // force loading of SDL first 27 _logger = sdl2._logger; 28 _SDLTTFInitialized = false; 29 30 try 31 { 32 DerelictSDL2ttf.load(); 33 } 34 catch(DerelictException e) 35 { 36 throw new SDL2Exception(e.msg); 37 } 38 39 int res = TTF_Init(); 40 if (res != 0) 41 throwSDL2TTFException("TTF_Init"); 42 43 _SDLTTFInitialized = true; 44 } 45 46 /// Releases the SDL_ttf library. 47 ~this() 48 { 49 if (_SDLTTFInitialized) 50 { 51 debug ensureNotInGC("SDLTTF"); 52 _SDLTTFInitialized = false; 53 TTF_Quit(); 54 } 55 } 56 deprecated("Use .destroy instead") void close(){} 57 } 58 59 private 60 { 61 Logger _logger; 62 SDL2 _sdl2; 63 bool _SDLTTFInitialized; 64 65 void throwSDL2TTFException(string callThatFailed) 66 { 67 string message = format("%s failed: %s", callThatFailed, getErrorString()); 68 throw new SDL2Exception(message); 69 } 70 71 const(char)[] getErrorString() 72 { 73 return fromStringz(TTF_GetError()); 74 } 75 } 76 } 77 78 /// SDL_ttf loaded font wrapper. 79 final class SDLFont 80 { 81 public 82 { 83 /// Loads a font from a file. 84 /// Params: 85 /// ptSize = font size in 72 dpi ("This basically translates to pixel height" says the doc). 86 /// Throws: $(D SDL2Exception) on error. 87 this(SDLTTF sdlttf, string filename, int ptSize) 88 { 89 _sdlttf = sdlttf; 90 _font = TTF_OpenFont(toStringz(filename), ptSize); 91 if (_font is null) 92 _sdlttf.throwSDL2TTFException("TTF_OpenFont"); 93 } 94 95 /// Releases the SDL resource. 96 ~this() 97 { 98 if (_font !is null) 99 { 100 debug ensureNotInGC("SDLFont"); 101 TTF_CloseFont(_font); 102 _font = null; 103 } 104 } 105 deprecated("Use .destroy instead") void close(){} 106 107 /// Returns: Font style. 108 int style() 109 { 110 return TTF_GetFontStyle(_font); 111 } 112 113 /// Set font style. 114 int setStyle(int newStyle) 115 { 116 if (newStyle != TTF_GetFontStyle(_font)) 117 TTF_SetFontStyle(_font, newStyle); 118 return newStyle; 119 } 120 121 /// Returns: Font hinting. 122 int hinting() 123 { 124 return TTF_GetFontHinting(_font); 125 } 126 127 /// Set font hinting. 128 int setHinting(int newHinting) 129 { 130 if (newHinting != TTF_GetFontHinting(_font)) 131 TTF_SetFontHinting(_font, newHinting); 132 return newHinting; 133 } 134 135 /// Returns: Font outline. 136 int outline() 137 { 138 return TTF_GetFontOutline(_font); 139 } 140 141 /// Set font outline. 142 int setOutline(int newOutline) 143 { 144 if (newOutline != TTF_GetFontOutline(_font)) 145 TTF_SetFontOutline(_font, newOutline); 146 return newOutline; 147 } 148 149 /// Returns: true if kerning is enabled. 150 bool getKerning() 151 { 152 return TTF_GetFontKerning(_font) != 0; 153 } 154 155 /// Enables/Disables font kerning. 156 bool setKerning(bool enabled) 157 { 158 TTF_SetFontKerning(_font, enabled ? 1 : 0); 159 return enabled; 160 } 161 162 /// Returns: Maximum height of a glyph in pixels. 163 int height() 164 { 165 return TTF_FontAscent(_font); 166 } 167 168 /// Returns: Height above baseline in pixels. 169 int ascent() 170 { 171 return TTF_FontAscent(_font); 172 } 173 174 /// Returns: Height below baseline. 175 int descent() 176 { 177 return TTF_FontDescent(_font); 178 } 179 180 /// Returns: Line skip, the recommended pixel interval between two lines. 181 int lineSkip() 182 { 183 return TTF_FontLineSkip(_font); 184 } 185 186 /// Returns: Size of text in pixels if rendered with this font. 187 SDL_Point measureText(string text) 188 { 189 int w, h; 190 TTF_SizeUTF8(_font, toStringz(text), &w, &h); 191 return SDL_Point(w, h); 192 } 193 194 /// Create a 32-bit ARGB surface and render the given character at high quality, 195 /// using alpha blending to dither the font with the given color. 196 /// Throws: $(D SDL2Exception) on error. 197 SDL2Surface renderGlyphBlended(dchar ch, SDL_Color color) 198 { 199 return checkedSurface(TTF_RenderGlyph_Blended(_font, cast(ushort)ch, color)); 200 } 201 202 /// Create a 32-bit ARGB surface and render the given text at high quality, 203 /// using alpha blending to dither the font with the given color. 204 /// Throws: $(D SDL2Exception) on error. 205 SDL2Surface renderTextBlended(string text, SDL_Color color) 206 { 207 return checkedSurface(TTF_RenderUTF8_Blended(_font, toStringz(text), color)); 208 } 209 210 /// Create an 8-bit palettized surface and render the given text at fast 211 /// quality with the given font and color. 212 /// Throws: $(D SDL2Exception) on error. 213 SDL2Surface renderTextSolid(string text, SDL_Color color) 214 { 215 return checkedSurface(TTF_RenderUTF8_Solid(_font, toStringz(text), color)); 216 } 217 218 /// Create an 8-bit palettized surface and render the given text at high 219 /// quality with the given font and colors. 220 /// Throws: $(D SDL2Exception) on error. 221 SDL2Surface renderTextShaded(string text, SDL_Color fg, SDL_Color bg) 222 { 223 return checkedSurface(TTF_RenderUTF8_Shaded(_font, toStringz(text), fg, bg)); 224 } 225 226 /// Create a 32-bit ARGB surface and render the given text at high quality, 227 /// using alpha blending to dither the font with the given color. 228 /// Uses multi-line text wrapping. 229 /// Throws: $(D SDL2Exception) on error. 230 SDL2Surface renderTextBlendedWrapped(string text, SDL_Color color, uint wrapLength) 231 { 232 return checkedSurface(TTF_RenderUTF8_Blended_Wrapped(_font, toStringz(text), color, wrapLength)); 233 } 234 } 235 236 private 237 { 238 SDLTTF _sdlttf; 239 TTF_Font *_font; 240 241 SDL2Surface checkedSurface(SDL_Surface* s) 242 { 243 if (s is null) 244 _sdlttf.throwSDL2TTFException("TTF_Render"); 245 return new SDL2Surface(_sdlttf._sdl2, s, SDL2Surface.Owned.YES); 246 } 247 } 248 }