1 /**
2 * ae.ui.video.renderer
3 *
4 * License:
5 * This Source Code Form is subject to the terms of
6 * the Mozilla Public License, v. 2.0. If a copy of
7 * the MPL was not distributed with this file, You
8 * can obtain one at http://mozilla.org/MPL/2.0/.
9 *
10 * Authors:
11 * Vladimir Panteleev <ae@cy.md>
12 */13 14 moduleae.ui.video.renderer;
15 16 publicimportae.utils.graphics.color;
17 publicimportae.utils.graphics.image;
18 publicimportae.utils.graphics.view;
19 20 importae.utils.meta : enumLength;
21 22 /// Abstract class for a video renderer.23 classRenderer24 {
25 aliasCOLOR = BGRX; /// The color type we use for all rendering.26 27 /// BGRX/BGRA-only.28 structBitmap29 {
30 staticassert(COLOR.sizeof == uint.sizeof);
31 /// `ae.utils.graphics.view` implementation.32 aliasStorageType = PlainStorageUnit!COLOR;
33 34 StorageType* pixels; /// ditto35 /// ditto36 xy_tw, h, stride;
37 38 inout(StorageType)[] scanline(xy_ty) inout39 {
40 assert(y>=0 && y<h);
41 returnpixels[stride*y..stride*(y+1)];
42 } /// ditto43 44 mixinDirectView;
45 }
46 47 /// True when this renderer can lock quickly (usually when it's rendering in software).48 /*immutable*/boolcanFastLock;
49 50 /// Lock a 32-bit, BGRX/BGRA surface51 abstractBitmapfastLock();
52 53 /// ditto54 abstractBitmaplock();
55 56 /// Unlock what was previously locked57 abstractvoidunlock();
58 59 /// Finalize rendering and present it to the user (flip buffers etc.)60 abstractvoidpresent();
61 62 /// Destroy any bound resources63 abstractvoidshutdown();
64 65 // **********************************************************************66 67 /// Get geometry.68 abstract @propertyuintwidth();
69 abstract @propertyuintheight(); /// ditto70 71 /// Set a single pixel.72 abstractvoidputPixel(intx, inty, COLORcolor);
73 74 /// Set some pixels.75 structPixel { intx, /***/y; /***/COLORcolor; /***/ }
76 voidputPixels(Pixel[] pixels)
77 {
78 foreach (refpixel; pixels)
79 putPixel(pixel.tupleof);
80 } /// ditto81 82 /// Draw a straight line.83 abstractvoidline(floatx0, floaty0, floatx1, floaty1, COLORcolor);
84 /// Draw a vertical line.85 voidvline(intx, inty0, inty1, COLORcolor) { line(x, y0, x, y1, color); }
86 /// Draw a horizontal line.87 voidhline(intx0, intx1, inty, COLORcolor) { line(x0, y, x1, y, color); }
88 89 /// Draw a filled rectangle.90 abstractvoidfillRect(intx0, inty0, intx1, inty1, COLORcolor);
91 abstractvoidfillRect(floatx0, floaty0, floatx1, floaty1, COLORcolor); /// ditto92 93 /// Clear the entire surface.94 abstractvoidclear();
95 96 /// Draw a texture.97 abstractvoiddraw(intx, inty, TextureSourcesource, intu0, intv0, intu1, intv1);
98 99 /// Draw a texture with scaling.100 abstractvoiddraw(floatx0, floaty0, floatx1, floaty1, TextureSourcesource, intu0, intv0, intu1, intv1);
101 }
102 103 /// Uniquely identify private data owned by different renderers104 enumRenderers105 {
106 SDLSoftware, ///107 SDLOpenGL, ///108 SDL2, ///109 }
110 111 /// Base class for all renderer-specific texture data112 classTextureRenderData113 {
114 /// If `true`, the `TextureSource` has been destroyed,115 /// and so should this instance (in the render thread)>116 booldestroyed;
117 118 /// Uploaded version number of the texture.119 /// If it does not match `TextureSource.textureVersion`,120 /// it needs to be updated.121 uinttextureVersion = 0;
122 123 /// Set to `true` when any `TextureRenderData`124 /// needs to be destroyed.125 staticsharedboolcleanupNeeded;
126 }
127 128 /// Base class for logical textures129 classTextureSource130 {
131 // TODO: make this extensible for external renderer implementations.132 /// Renderer-specific texture data.133 TextureRenderData[enumLength!Renderers] renderData;
134 135 /// Source version number of the texture.136 uinttextureVersion = 1;
137 138 /// Common type used by `drawTo` / `getPixels`139 aliasImageRef!(Renderer.COLOR) TextureCanvas;
140 141 /// Request the contents of this `TextureSource`.142 /// Used when the target pixel memory is already allocated143 abstractvoiddrawTo(TextureCanvasdest);
144 145 /// Request the contents of this `TextureSource`.146 /// Used when a pointer is needed to existing pixel memory147 abstractTextureCanvasgetPixels();
148 149 ~this() @nogc150 {
151 foreach (r; renderData)
152 if (r)
153 r.destroyed = true;
154 TextureRenderData.cleanupNeeded = true;
155 }
156 }
157 158 /// Implementation of `TextureSource`159 /// using an `ae.utils.graphics.image.Image`.160 classImageTextureSource : TextureSource161 {
162 Image!(Renderer.COLOR) image; /// The image.163 164 overridevoiddrawTo(TextureCanvasdest)
165 {
166 image.blitTo(dest);
167 } ///168 169 overrideTextureCanvasgetPixels()
170 {
171 returnimage.toRef();
172 } ///173 }
174 175 /// Base class for `TextureSource` implementations176 /// where pixel data is calculated on request.177 classProceduralTextureSource : TextureSource178 {
179 privateImage!(Renderer.COLOR) cachedImage;
180 181 /// Query the size of the procedural texture182 abstractvoidgetSize(outintwidth, outintheight);
183 184 /// Implementation of `getPixels` using185 /// `drawTo` and a cached copy.186 overrideTextureCanvasgetPixels()
187 {
188 if (!cachedImage.w)
189 {
190 intw, h;
191 getSize(w, h);
192 cachedImage.size(w, h);
193 drawTo(cachedImage.toRef());
194 }
195 returncachedImage.toRef();
196 }
197 }