From 577489bf96ff7a6e1de70b59af219e8f64285821 Mon Sep 17 00:00:00 2001 From: Ketmar Dark Date: Mon, 12 Mar 2018 23:30:07 +0200 Subject: [PATCH] nanovega/blendish/svg fixes (small) --- blendish.d | 17 +++++------ nanovega.d | 87 +++++++++++++++++++++++------------------------------- svg.d | 6 ++++ 3 files changed, 51 insertions(+), 59 deletions(-) diff --git a/blendish.d b/blendish.d index fa38f04..6215ef1 100644 --- a/blendish.d +++ b/blendish.d @@ -1702,31 +1702,30 @@ public void bndJoinAreaOverlay (NVGContext ctx, float x, float y, float w, float /// returns the ideal width for a label with given icon and text public float bndLabelWidth(T=char) (NVGContext ctx, int iconid, const(T)[] label) if (isAnyCharType!T) { - int w = BND_PAD_LEFT+BND_PAD_RIGHT; + float w = BND_PAD_LEFT+BND_PAD_RIGHT; if (iconid >= 0) w += BND_ICON_SHEET_RES; - if (label && bndFont >= 0) { + if (label.length && bndFont >= 0) { ctx.fontFaceId(bndFont); ctx.fontSize(BND_LABEL_FONT_SIZE); - w += cast(int)ctx.textBounds(1, 1, label, null); + w += ctx.textBounds(1, 1, label, null); } - //{ import core.stdc.stdio : printf; printf("len=%u; w=%d\n", cast(uint)label.length, cast(int)w); } - return w; + return cast(float)cast(int)(w+0.5); } /// returns the height for a label with given icon, text and width; this function is primarily useful in conjunction with multiline labels and textboxes public float bndLabelHeight(T=char) (NVGContext ctx, int iconid, const(T)[] label, float width) if (isAnyCharType!T) { - int h = BND_WIDGET_HEIGHT; + float h = BND_WIDGET_HEIGHT; width -= BND_TEXT_RADIUS*2; if (iconid >= 0) width -= BND_ICON_SHEET_RES; - if (label && (bndFont >= 0)) { + if (label.length && bndFont >= 0) { ctx.fontFaceId(bndFont); ctx.fontSize(BND_LABEL_FONT_SIZE); float[4] bounds = void; ctx.textBoxBounds(1, 1, width, label, bounds[]); - int bh = cast(int)(bounds[3]-bounds[1])+BND_TEXT_PAD_DOWN; + float bh = (bounds[3]-bounds[1])+BND_TEXT_PAD_DOWN; if (bh > h) h = bh; } - return h; + return cast(float)cast(int)(h+0.5); } diff --git a/nanovega.d b/nanovega.d index 0526077..b215c8b 100644 --- a/nanovega.d +++ b/nanovega.d @@ -1559,7 +1559,11 @@ public int height (NVGContext ctx) pure nothrow @trusted @nogc { pragma(inline, /// valid only inside [beginFrame]/[endFrame] /// Group: context_management -public float devicePixelRatio (NVGContext ctx) pure nothrow @trusted @nogc { pragma(inline, true); return (ctx !is null ? ctx.mDeviceRatio : float.nan); } +public float devicePixelRatio (NVGContext ctx) pure nothrow @trusted @nogc { pragma(inline, true); return (ctx !is null ? ctx.devicePxRatio : float.nan); } + +/// Returns `true` if [beginFrame] was called, and neither [endFrame], nor [cancelFrame] were. +/// Group: context_management +public bool inFrame (NVGContext ctx) pure nothrow @trusted @nogc { pragma(inline, true); return (ctx !is null && ctx.contextAlive ? ctx.nstates > 0 : false); } // path autoregistration @@ -1629,7 +1633,6 @@ private: // internals NVGMatrix gpuAffine; int mWidth, mHeight; - float mDeviceRatio; // image manager int imageCount; // number of alive images in this context bool contextAlive; // context can be dead, but still contain some images @@ -1824,7 +1827,7 @@ NVGCompositeOperationState nvg__compositeOperationState (NVGCompositeOperation o NVGstate* nvg__getState (NVGContext ctx) pure nothrow @trusted @nogc { pragma(inline, true); - return &ctx.states.ptr[ctx.nstates-1]; + return &ctx.states.ptr[ctx.nstates-(ctx.nstates > 0 ? 1 : 0)]; } // Constructor called by the render back-end. @@ -1854,6 +1857,7 @@ NVGContext createInternal (NVGparams* params) nothrow @trusted @nogc { ctx.reset(); nvg__setDevicePixelRatio(ctx, 1.0f); + ctx.mWidth = ctx.mHeight = 0; if (!ctx.params.renderCreate(ctx.params.userPtr)) goto error; @@ -1965,6 +1969,7 @@ public void beginFrame (NVGContext ctx, int windowWidth, int windowHeight, float ctx.drawCallCount, ctx.fillTriCount, ctx.strokeTriCount, ctx.textTriCount, ctx.fillTriCount+ctx.strokeTriCount+ctx.textTriCount); */ + if (ctx.nstates > 0) ctx.cancelFrame(); if (windowWidth < 1) windowWidth = 1; if (windowHeight < 1) windowHeight = 1; @@ -1981,7 +1986,6 @@ public void beginFrame (NVGContext ctx, int windowWidth, int windowHeight, float ctx.params.renderViewport(ctx.params.userPtr, windowWidth, windowHeight); ctx.mWidth = windowWidth; ctx.mHeight = windowHeight; - ctx.mDeviceRatio = devicePixelRatio; ctx.recset = null; ctx.recstart = -1; @@ -2003,9 +2007,8 @@ public void beginFrame (NVGContext ctx, int windowWidth, int windowHeight, float /// Group: frame_management public void cancelFrame (NVGContext ctx) nothrow @trusted @nogc { ctx.cancelRecording(); - ctx.mWidth = 0; - ctx.mHeight = 0; - ctx.mDeviceRatio = 0; + //ctx.mWidth = 0; + //ctx.mHeight = 0; // cancel render queue ctx.params.renderCancel(ctx.params.userPtr); // clear saved states (this may free some textures) @@ -2018,9 +2021,8 @@ public void cancelFrame (NVGContext ctx) nothrow @trusted @nogc { public void endFrame (NVGContext ctx) nothrow @trusted @nogc { if (ctx.recset !is null) ctx.recset.takeCurrentPickScene(ctx); ctx.stopRecording(); - ctx.mWidth = 0; - ctx.mHeight = 0; - ctx.mDeviceRatio = 0; + //ctx.mWidth = 0; + //ctx.mHeight = 0; // flush render queue NVGstate* state = nvg__getState(ctx); ctx.params.renderFlush(ctx.params.userPtr); @@ -8392,15 +8394,13 @@ public float textBounds(T) (NVGContext ctx, float x, float y, const(T)[] str, fl if (isAnyCharType!T) { NVGstate* state = nvg__getState(ctx); - float scale = nvg__getFontScale(state)*ctx.devicePxRatio; - float invscale = 1.0f/scale; - float width; if (state.fontId == FONS_INVALID) { bounds[] = 0; return 0; } + immutable float scale = nvg__getFontScale(state)*ctx.devicePxRatio; fonsSetSize(ctx.fs, state.fontSize*scale); fonsSetSpacing(ctx.fs, state.letterSpacing*scale); fonsSetBlur(ctx.fs, state.fontBlur*scale); @@ -8408,7 +8408,8 @@ if (isAnyCharType!T) fonsSetFont(ctx.fs, state.fontId); float[4] b = void; - width = fonsTextBounds(ctx.fs, x*scale, y*scale, str, b[]); + immutable float width = fonsTextBounds(ctx.fs, x*scale, y*scale, str, b[]); + immutable float invscale = 1.0f/scale; if (bounds.length) { // use line bounds for height fonsLineBounds(ctx.fs, y*scale, b.ptr+1, b.ptr+3); @@ -11180,54 +11181,40 @@ if (isAnyCharType!T) miny = maxy = y; startx = x; - static if (is(T == char)) { - foreach (char ch; str) { + foreach (T ch; str) { + static if (T.sizeof == 1) { //if (fons__decutf8(&utf8state, &codepoint, *cast(const(ubyte)*)str)) continue; mixin(DecUtfMixin!("utf8state", "codepoint", "(cast(ubyte)ch)")); if (utf8state) continue; - glyph = fons__getGlyph(stash, font, codepoint, isize, iblur, FONS_GLYPH_BITMAP_OPTIONAL); - if (glyph !is null) { - fons__getQuad(stash, font, prevGlyphIndex, glyph, isize/10.0f, scale, state.spacing, &x, &y, &q); - if (q.x0 < minx) minx = q.x0; - if (q.x1 > maxx) maxx = q.x1; - if (stash.params.flags&FONS_ZERO_TOPLEFT) { - if (q.y0 < miny) miny = q.y0; - if (q.y1 > maxy) maxy = q.y1; - } else { - if (q.y1 < miny) miny = q.y1; - if (q.y0 > maxy) maxy = q.y0; - } - prevGlyphIndex = glyph.index; - } else { - prevGlyphIndex = -1; - } - } - } else { - foreach (T ch; str) { - static if (is(T == dchar)) { + } else { + static if (T.sizeof == 4) { if (ch > dchar.max) ch = 0xFFFD; } codepoint = cast(uint)ch; - glyph = fons__getGlyph(stash, font, codepoint, isize, iblur, FONS_GLYPH_BITMAP_OPTIONAL); - if (glyph !is null) { - fons__getQuad(stash, font, prevGlyphIndex, glyph, isize/10.0f, scale, state.spacing, &x, &y, &q); - if (q.x0 < minx) minx = q.x0; - if (q.x1 > maxx) maxx = q.x1; - if (stash.params.flags&FONS_ZERO_TOPLEFT) { - if (q.y0 < miny) miny = q.y0; - if (q.y1 > maxy) maxy = q.y1; - } else { - if (q.y1 < miny) miny = q.y1; - if (q.y0 > maxy) maxy = q.y0; - } - prevGlyphIndex = glyph.index; + } + glyph = fons__getGlyph(stash, font, codepoint, isize, iblur, FONS_GLYPH_BITMAP_OPTIONAL); + if (glyph !is null) { + //{ import core.stdc.stdio; printf("0: x=%g; y=%g\n", cast(double)x, cast(double)y); } + fons__getQuad(stash, font, prevGlyphIndex, glyph, isize/10.0f, scale, state.spacing, &x, &y, &q); + //{ import core.stdc.stdio; printf("1: x=%g; y=%g\n", cast(double)x, cast(double)y); } + if (q.x0 < minx) minx = q.x0; + if (q.x1 > maxx) maxx = q.x1; + if (stash.params.flags&FONS_ZERO_TOPLEFT) { + if (q.y0 < miny) miny = q.y0; + if (q.y1 > maxy) maxy = q.y1; } else { - prevGlyphIndex = -1; + if (q.y1 < miny) miny = q.y1; + if (q.y0 > maxy) maxy = q.y0; } + prevGlyphIndex = glyph.index; + } else { + //{ import core.stdc.stdio; printf("NO GLYPH FOR 0x%04x\n", cast(uint)codepoint); } + prevGlyphIndex = -1; } } advance = x-startx; + //{ import core.stdc.stdio; printf("***: x=%g; startx=%g; advance=%g\n", cast(double)x, cast(double)startx, cast(double)advance); } // Align horizontally if (state.talign.left) { diff --git a/svg.d b/svg.d index 6fefc51..f6adba0 100644 --- a/svg.d +++ b/svg.d @@ -4029,6 +4029,12 @@ void nsvg__flattenCubicBez (NSVGrasterizer r, in float x1, in float y1, in float immutable x1234 = (x123+x234)*0.5f; immutable y1234 = (y123+y234)*0.5f; + // "taxicab" / "manhattan" check for flat curves + if (fabsf(x1+x3-x2-x2)+fabsf(y1+y3-y2-y2)+fabsf(x2+x4-x3-x3)+fabsf(y2+y4-y3-y3) < r.tessTol/4) { + nsvg__addPathPoint(r, x1234, y1234, type); + return; + } + nsvg__flattenCubicBez(r, x1, y1, x12, y12, x123, y123, x1234, y1234, level+1, 0); nsvg__flattenCubicBez(r, x1234, y1234, x234, y234, x34, y34, x4, y4, level+1, type); }