diff --git a/src/D2DLibExport/D2DDevice.cs b/src/D2DLibExport/D2DDevice.cs index cea5bffe7..42742ef77 100644 --- a/src/D2DLibExport/D2DDevice.cs +++ b/src/D2DLibExport/D2DDevice.cs @@ -101,19 +101,19 @@ public D2DRectangleGeometry CreateRectangleGeometry(FLOAT width, FLOAT height) public D2DRectangleGeometry CreateRectangleGeometry(D2DRect rect) { HANDLE rectGeometryHandle = D2D.CreateRectangleGeometry(this.Handle, ref rect); - return new D2DRectangleGeometry(this.Handle, rectGeometryHandle); + return new D2DRectangleGeometry(this, rectGeometryHandle); } public D2DPathGeometry CreatePathGeometry() { HANDLE geoHandle = D2D.CreatePathGeometry(this.Handle); - return new D2DPathGeometry(this.Handle, geoHandle); + return new D2DPathGeometry(this, geoHandle); } public D2DGeometry CreateEllipseGeometry(D2DPoint origin, D2DSize size) { var ellipse = new D2DEllipse(origin, size); - return new D2DGeometry(this.Handle, D2D.CreateEllipseGeometry(this.Handle, ref ellipse)); + return new D2DGeometry(this, D2D.CreateEllipseGeometry(this.Handle, ref ellipse)); } public D2DGeometry CreatePieGeometry(D2DPoint origin, D2DSize size, float startAngle, float endAngle) @@ -143,6 +143,29 @@ public D2DGeometry CreatePieGeometry(D2DPoint origin, D2DSize size, float startA return path; } + public D2DPathGeometry CreateTextPathGeometry(string text, string fontName, float fontSize, + D2DFontWeight fontWeight = D2DFontWeight.Normal, + D2DFontStyle fontStyle = D2DFontStyle.Normal, + D2DFontStretch fontStretch = D2DFontStretch.Normal) + { + var fontFace = D2D.CreateFontFace(this.Handle, fontName, fontWeight, fontStyle, fontStretch); + if (fontFace == IntPtr.Zero) + { + throw new CreateFontFaceFailedException(fontName); + } + + var pathHandler = D2D.CreateTextPathGeometry(this.Handle, text, fontFace, fontSize); + + D2D.DestroyFontFace(fontFace); + + if (pathHandler == IntPtr.Zero) + { + throw new CreatePathGeometryFailedException(); + } + + return new D2DPathGeometry(this, pathHandler); + } + public D2DBitmap? LoadBitmap(byte[] buffer) { return this.LoadBitmap(buffer, 0, (uint)buffer.Length); diff --git a/src/D2DLibExport/D2DGeometry.cs b/src/D2DLibExport/D2DGeometry.cs index cf3926445..57189980a 100644 --- a/src/D2DLibExport/D2DGeometry.cs +++ b/src/D2DLibExport/D2DGeometry.cs @@ -26,12 +26,12 @@ namespace unvell.D2DLib { public class D2DGeometry : D2DObject { - internal HANDLE DeviceHandle { get; private set; } + internal D2DDevice Device { get; private set; } - internal D2DGeometry(HANDLE deviceHandle, HANDLE geoHandle) + internal D2DGeometry(D2DDevice device, HANDLE geoHandle) : base(geoHandle) { - this.DeviceHandle = deviceHandle; + this.Device = device; } // FIXME: TO be implemented diff --git a/src/D2DLibExport/D2DGraphics.cs b/src/D2DLibExport/D2DGraphics.cs index 4adfda282..149a32614 100644 --- a/src/D2DLibExport/D2DGraphics.cs +++ b/src/D2DLibExport/D2DGraphics.cs @@ -435,6 +435,38 @@ public void DrawText(string text, D2DColor color, string fontName, float fontSiz fontWeight, fontStyle, fontStretch, halign, valign); } + public void DrawStrokedText(string text, D2DPoint location, + D2DColor strokeColor, float strokeWidth, + D2DColor fillColor, + string fontName, float fontSize, + D2DFontWeight fontWeight = D2DFontWeight.Normal, + D2DFontStyle fontStyle = D2DFontStyle.Normal, + D2DFontStretch fontStretch = D2DFontStretch.Normal) + { + this.DrawStrokedText(text, location.x, location.y, strokeColor, strokeWidth, fillColor, + fontName, fontSize, fontWeight, fontStyle, fontStretch); + } + + public void DrawStrokedText(string text, float x, float y, + D2DColor strokeColor, float strokeWidth, + D2DColor fillColor, + string fontName, float fontSize, + D2DFontWeight fontWeight = D2DFontWeight.Normal, + D2DFontStyle fontStyle = D2DFontStyle.Normal, + D2DFontStretch fontStretch = D2DFontStretch.Normal) + { + using (var textPath = this.Device.CreateTextPathGeometry(text, fontName, fontSize, + fontWeight, fontStyle, fontStretch)) + { + this.TranslateTransform(x, y); + + this.FillPath(textPath, fillColor); + this.DrawPath(textPath, strokeColor, strokeWidth); + + this.TranslateTransform(-x, -y); + } + } + public D2DSize MeasureText(string text, string fontName, float fontSize, D2DSize placeSize) { D2DSize outputSize = placeSize; diff --git a/src/D2DLibExport/D2DLib.cs b/src/D2DLibExport/D2DLib.cs index 3d0907bea..cb47ad13d 100644 --- a/src/D2DLibExport/D2DLib.cs +++ b/src/D2DLibExport/D2DLib.cs @@ -201,6 +201,19 @@ public static extern void MeasureText([In] HANDLE ctx, [In] string text, [In] st [In] DWriteTextAlignment halign = DWriteTextAlignment.Leading, [In] DWriteParagraphAlignment valign = DWriteParagraphAlignment.Near); + [DllImport(DLL_NAME, CharSet = CharSet.Unicode, CallingConvention = CallingConvention.Cdecl)] + public static extern HANDLE CreateFontFace([In] HANDLE context, [In] string fontName, + [In] D2DFontWeight fontWeight = D2DFontWeight.Normal, + [In] D2DFontStyle fontStyle = D2DFontStyle.Normal, + [In] D2DFontStretch fontStretch = D2DFontStretch.Normal); + + [DllImport(DLL_NAME, CharSet = CharSet.Unicode, CallingConvention = CallingConvention.Cdecl)] + public static extern void DestroyFontFace(HANDLE fontFaceHandle); + + [DllImport(DLL_NAME, CharSet = CharSet.Unicode, CallingConvention = CallingConvention.Cdecl)] + public static extern HANDLE CreateTextPathGeometry(HANDLE ctx, [In] string text, + HANDLE fontFaceHandle, FLOAT fontSize); + #endregion // Text #region Geometry diff --git a/src/D2DLibExport/D2DPathGeometry.cs b/src/D2DLibExport/D2DPathGeometry.cs index e9024c7ef..a2f910cc6 100644 --- a/src/D2DLibExport/D2DPathGeometry.cs +++ b/src/D2DLibExport/D2DPathGeometry.cs @@ -26,15 +26,15 @@ namespace unvell.D2DLib { public class D2DPathGeometry : D2DGeometry { - internal D2DPathGeometry(HANDLE deviceHandle, HANDLE pathHandle) - : base(deviceHandle, pathHandle) + internal D2DPathGeometry(D2DDevice device, HANDLE pathHandle) + : base(device, pathHandle) { } - public void SetStartPoint(FLOAT x, FLOAT y) - { - this.SetStartPoint(new D2DPoint(x, y)); - } + public void SetStartPoint(FLOAT x, FLOAT y) + { + this.SetStartPoint(new D2DPoint(x, y)); + } public void SetStartPoint(D2DPoint startPoint) { @@ -51,15 +51,15 @@ public void AddBeziers(D2DBezierSegment[] bezierSegments) D2D.AddPathBeziers(this.Handle, bezierSegments); } - // TODO: unnecessary API and it doesn't work very well, consider to remove - //public void AddEllipse(D2DEllipse ellipse) - //{ - // D2D.AddPathEllipse(this.Handle, ref ellipse); - //} + // TODO: unnecessary API and it doesn't work very well, consider to remove + //public void AddEllipse(D2DEllipse ellipse) + //{ + // D2D.AddPathEllipse(this.Handle, ref ellipse); + //} - public void AddArc(D2DPoint endPoint, D2DSize size, FLOAT sweepAngle, - D2DArcSize arcSize = D2DArcSize.Small, - D2DSweepDirection sweepDirection = D2DSweepDirection.Clockwise) + public void AddArc(D2DPoint endPoint, D2DSize size, FLOAT sweepAngle, + D2DArcSize arcSize = D2DArcSize.Small, + D2DSweepDirection sweepDirection = D2DSweepDirection.Clockwise) { D2D.AddPathArc(this.Handle, endPoint, size, sweepAngle, arcSize, sweepDirection); } @@ -81,8 +81,8 @@ public void ClosePath() public override void Dispose() { - if (this.Handle != IntPtr.Zero) D2D.DestroyPathGeometry(this.Handle); - this.handle = IntPtr.Zero; + if (this.Handle != IntPtr.Zero) D2D.DestroyPathGeometry(this.Handle); + this.handle = IntPtr.Zero; } } } diff --git a/src/D2DLibExport/D2DPieGeometry.cs b/src/D2DLibExport/D2DPieGeometry.cs index c64c973f1..1d60920a9 100644 --- a/src/D2DLibExport/D2DPieGeometry.cs +++ b/src/D2DLibExport/D2DPieGeometry.cs @@ -26,8 +26,8 @@ namespace unvell.D2DLib { public class D2DPieGeometry : D2DGeometry { - internal D2DPieGeometry(HANDLE deviceHandle, HANDLE pathHandle) - : base(deviceHandle, pathHandle) + internal D2DPieGeometry(D2DDevice device, HANDLE pathHandle) + : base(device, pathHandle) { } diff --git a/src/D2DLibExport/D2DRectangleGeometry.cs b/src/D2DLibExport/D2DRectangleGeometry.cs index 3cdd3d6d8..e71546343 100644 --- a/src/D2DLibExport/D2DRectangleGeometry.cs +++ b/src/D2DLibExport/D2DRectangleGeometry.cs @@ -26,8 +26,8 @@ namespace unvell.D2DLib { public class D2DRectangleGeometry : D2DGeometry { - internal D2DRectangleGeometry(HANDLE deviceHandle, HANDLE geoHandle) - : base(deviceHandle, geoHandle) + internal D2DRectangleGeometry(D2DDevice device, HANDLE geoHandle) + : base(device, geoHandle) { } } diff --git a/src/D2DLibExport/Exceptions.cs b/src/D2DLibExport/Exceptions.cs new file mode 100644 index 000000000..4b42dc61f --- /dev/null +++ b/src/D2DLibExport/Exceptions.cs @@ -0,0 +1,40 @@ +/* + * MIT License + * + * Copyright (c) 2009-2022 Jingwood, unvell.com. All right reserved. + * + * Permission is hereby granted, free of charge, to any person obtaining a copy + * of this software and associated documentation files (the "Software"), to deal + * in the Software without restriction, including without limitation the rights + * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell + * copies of the Software, and to permit persons to whom the Software is + * furnished to do so, subject to the following conditions: + * + * The above copyright notice and this permission notice shall be included in all + * copies or substantial portions of the Software. + * + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE + * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, + * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE + * SOFTWARE. + */ +namespace unvell.D2DLib +{ + public class CreateFontFaceFailedException : Exception + { + public CreateFontFaceFailedException(string fontName) : + base("Create font face failed by specified font name: " + fontName) + { + } + } + + public class CreatePathGeometryFailedException : Exception + { + public CreatePathGeometryFailedException() + { + } + } +} diff --git a/src/Examples/SampleCode/StrokedText.cs b/src/Examples/SampleCode/StrokedText.cs new file mode 100644 index 000000000..317e8cd84 --- /dev/null +++ b/src/Examples/SampleCode/StrokedText.cs @@ -0,0 +1,91 @@ +/* + * MIT License + * + * Copyright (c) 2009-2022 Jingwood, unvell.com. All right reserved. + * + * Permission is hereby granted, free of charge, to any person obtaining a copy + * of this software and associated documentation files (the "Software"), to deal + * in the Software without restriction, including without limitation the rights + * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell + * copies of the Software, and to permit persons to whom the Software is + * furnished to do so, subject to the following conditions: + * + * The above copyright notice and this permission notice shall be included in all + * copies or substantial portions of the Software. + * + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE + * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, + * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE + * SOFTWARE. + */ + +namespace unvell.D2DLib.Examples.SampleCode +{ + public partial class StrokedText : ExampleForm + { + public StrokedText() + { + Text = "Stroked Text Drawing - d2dlib Examples"; + BackColor = Color.White; + } + + D2DPathGeometry textPath1, textPath2; + + protected override void OnLoad(EventArgs e) + { + base.OnLoad(e); + + // Exception will be thrown when the specified font not found in the Windows OS + textPath1 = this.Device.CreateTextPathGeometry("Hello World", "Arial", 36, D2DFontWeight.ExtraBold); + } + + protected override void OnRender(D2DGraphics g) + { + base.OnRender(g); + + // + // Example 1: Create a path geometry from text and render it + // + // change the location to draw the text + g.TranslateTransform(100, 100); + + g.FillPath(this.textPath1, D2DColor.Yellow); + g.DrawPath(this.textPath1, D2DColor.Blue, 3); + + // restore the location after drawing text + g.TranslateTransform(-100, -100); + + + + // + // Example 2: Use a helper method to draw a stroked text in simplest way + // + g.DrawStrokedText("Stroked text rendered using D2DLib", 100, 200, + D2DColor.DarkOliveGreen, 2, D2DColor.LightCyan, "Consolas", 24); + + + + // Example 3: Text in non-English language (Unicode fonts) + // The characters in the text must exist in the font. + // This case only works in a Japanese language Windows. + // + //var textPath2 = this.Device.CreateTextPathGeometry("こんにちは、世界", "MS Gothic", 28); + + //g.TranslateTransform(100, 300); + //g.DrawPath(textPath2, D2DColor.DarkGreen); + //g.TranslateTransform(-100, -300); + + } + + protected override void Dispose(bool disposing) + { + base.Dispose(disposing); + + this.textPath1?.Dispose(); + //this.textPath2?.Dispose(); + } + } +} diff --git a/src/d2dlib/Context.h b/src/d2dlib/Context.h index 036c460a6..b295dad33 100644 --- a/src/d2dlib/Context.h +++ b/src/d2dlib/Context.h @@ -1,4 +1,4 @@ -/* +/* * MIT License * * Copyright (c) 2009-2021 Jingwood, unvell.com. All right reserved. @@ -176,4 +176,4 @@ extern "C" D2DLIB_API void TestDraw(HANDLE context); } -#endif /* __D2DLIB_H__ */ \ No newline at end of file +#endif /* __D2DLIB_H__ */ diff --git a/src/d2dlib/Geometry.cpp b/src/d2dlib/Geometry.cpp index a729e2066..b9ab40277 100644 --- a/src/d2dlib/Geometry.cpp +++ b/src/d2dlib/Geometry.cpp @@ -1,4 +1,4 @@ -/* +/* * MIT License * * Copyright (c) 2009-2021 Jingwood, unvell.com. All right reserved. @@ -445,4 +445,4 @@ void GetGeometryTransformedBounds(HANDLE pathCtx, __in D2D1_MATRIX_3X2_F* mat3x2 { D2DPathContext* pathContext = reinterpret_cast(pathCtx); pathContext->path->GetBounds(mat3x2, rect); -} \ No newline at end of file +} diff --git a/src/d2dlib/Text.cpp b/src/d2dlib/Text.cpp index 7e93b91a6..01de3a5af 100644 --- a/src/d2dlib/Text.cpp +++ b/src/d2dlib/Text.cpp @@ -1,4 +1,4 @@ -/* +/* * MIT License * * Copyright (c) 2009-2021 Jingwood, unvell.com. All right reserved. @@ -36,12 +36,8 @@ D2DLIB_API void DrawString(HANDLE ctx, LPCWSTR text, D2D1_COLOR_F color, ID2D1SolidColorBrush* brush = NULL; IDWriteTextFormat* textFormat = NULL; - HRESULT hr = context->writeFactory->CreateTextFormat(fontName, - NULL, - fontWeight, fontStyle, fontStretch, - fontSize, - L"", //locale - &textFormat); + HRESULT hr = context->writeFactory->CreateTextFormat(fontName, NULL, + fontWeight, fontStyle, fontStretch, fontSize, L"", &textFormat); if (SUCCEEDED(hr) && textFormat != NULL) { @@ -97,6 +93,133 @@ D2DLIB_API HANDLE CreateTextLayout(HANDLE ctx, LPCWSTR text, LPCWSTR fontName, F return NULL; } +D2DLIB_API HANDLE CreateFontFace(HANDLE ctx, LPCWSTR fontName, + DWRITE_FONT_WEIGHT fontWeight, DWRITE_FONT_STYLE fontStyle, DWRITE_FONT_STRETCH fontStretch) { + + RetrieveContext(ctx); + HRESULT hr = NULL; + D2DFontFace* d2dFontFace = NULL; + + IDWriteFontCollection* coll; + hr = context->writeFactory->GetSystemFontCollection(&coll); + + if (SUCCEEDED(hr)) + { + UINT32 fontIndex; + BOOL fontIndexFound; + coll->FindFamilyName(fontName, &fontIndex, &fontIndexFound); + + if (fontIndexFound) { + + IDWriteFontFamily* fontFamily; + hr = coll->GetFontFamily(fontIndex, &fontFamily); + + if (SUCCEEDED(hr)) + { + IDWriteFont* font; + hr = fontFamily->GetFirstMatchingFont(fontWeight, fontStretch, fontStyle, &font); + + if (SUCCEEDED(hr)) { + IDWriteFontFace* fontFace; + hr = font->CreateFontFace(&fontFace); + + if (SUCCEEDED(hr)) { + d2dFontFace = new D2DFontFace(); + d2dFontFace->font = font; + d2dFontFace->fontFace = fontFace; + } + } + + SafeRelease(&fontFamily); + } + } + } + + SafeRelease(&coll); + + return d2dFontFace; +} + +void DestroyFontFace(HANDLE fontFaceHandle) { + D2DFontFace* fontFace = reinterpret_cast(fontFaceHandle); + + if (fontFace != NULL) { + SafeRelease(&fontFace->font); + SafeRelease(&fontFace->fontFace); + delete fontFace; + } +} + +HANDLE CreateTextPathGeometry(HANDLE ctx, LPCWSTR text, HANDLE fontFaceHandle, FLOAT fontSize) { + + RetrieveContext(ctx); + D2DFontFace* fontFaceWrap = reinterpret_cast(fontFaceHandle); + + if (fontFaceWrap == NULL) { + return NULL; + } + + HRESULT hr = NULL; + D2DPathContext* pathContext = NULL; + IDWriteFontFace* fontFace = fontFaceWrap->fontFace; + + int textLength = wcslen(text); + + UINT* codePoints = new UINT[textLength]; + UINT16* glyphIndices = new UINT16[textLength]; + ZeroMemory(codePoints, sizeof(UINT) * textLength); + ZeroMemory(glyphIndices, sizeof(UINT16) * textLength); + + for (int i = 0; i < textLength; i++) + { + codePoints[i] = text[i]; + } + + hr = fontFace->GetGlyphIndicesW(codePoints, textLength, glyphIndices); + + if (SUCCEEDED(hr)) { + + ID2D1PathGeometry* path = NULL; + ID2D1GeometrySink* sink = NULL; + + hr = context->factory->CreatePathGeometry(&path); + if (SUCCEEDED(hr)) { + + hr = path->Open(&sink); + if (SUCCEEDED(hr)) { + + hr = fontFace->GetGlyphRunOutline(fontSize * 96.0 / 72.0, glyphIndices, + NULL, NULL, textLength, FALSE, FALSE, sink); + + //sink->SetFillMode(D2D1_FILL_MODE_WINDING); + + sink->Close(); + SafeRelease(&sink); + + if (SUCCEEDED(hr)) { + pathContext = new D2DPathContext(); + + pathContext->path = path; + pathContext->geometry = pathContext->path; + pathContext->d2context = context; + } + } + + if (pathContext == NULL) { + SafeRelease(&path); + } + } + } + + delete[] codePoints; + codePoints = NULL; + delete[] glyphIndices; + glyphIndices = NULL; + + return (HANDLE)pathContext; +} + + D2DLIB_API void MeasureText(HANDLE ctx, LPCWSTR text, LPCWSTR fontName, FLOAT fontSize, D2D1_SIZE_F* size, DWRITE_FONT_WEIGHT fontWeight, DWRITE_FONT_STYLE fontStyle, DWRITE_FONT_STRETCH fontStretch) { RetrieveContext(ctx); diff --git a/src/d2dlib/Text.h b/src/d2dlib/Text.h index 5aeae5da3..3a4da4cb3 100644 --- a/src/d2dlib/Text.h +++ b/src/d2dlib/Text.h @@ -26,6 +26,17 @@ #include "Context.h" +typedef struct D2DFontFace { + //WCHAR fontName[128]; + //FLOAT fontSize; + //DWRITE_FONT_WEIGHT fontWeight; + //DWRITE_FONT_STYLE fontStyle; + //DWRITE_FONT_STRETCH fontStretch; + //IDWriteFontFamily* fontFamily; + IDWriteFont* font; + IDWriteFontFace* fontFace; +}; + extern "C" { D2DLIB_API void DrawString(HANDLE handle, LPCWSTR text, D2D1_COLOR_F color, @@ -52,4 +63,13 @@ extern "C" D2DLIB_API void DrawGlyphRun(HANDLE ctx, D2D1_POINT_2F baselineOrigin, const DWRITE_GLYPH_RUN *glyphRun, D2D1_COLOR_F color, DWRITE_MEASURING_MODE measuringMode = DWRITE_MEASURING_MODE_NATURAL); + + D2DLIB_API HANDLE CreateFontFace(HANDLE ctx, LPCWSTR fontName, + DWRITE_FONT_WEIGHT fontWeight = DWRITE_FONT_WEIGHT_NORMAL, + DWRITE_FONT_STYLE fontStyle = DWRITE_FONT_STYLE_NORMAL, + DWRITE_FONT_STRETCH fontStretch = DWRITE_FONT_STRETCH_NORMAL); + + D2DLIB_API void DestroyFontFace(HANDLE fontFaceHandle); + + D2DLIB_API HANDLE CreateTextPathGeometry(HANDLE ctx, LPCWSTR text, HANDLE fontFaceHandle, FLOAT fontSize); }